Android并发编程高级面试题汇总最全最细面试题讲解持续更新中👊👊 👀你想要的面试题这里都有👀 👇👇👇
ThreadLocal是什么?
这道题想考察什么?
是否了解ThreadLocal与真实场景使用,是否熟悉ThreadLocal
考察的知识点
ThreadLocal的概念在项目中使用与基本知识
考生应该如何回答
ThreadLocal提供了线程本地变量,它可以保证访问到的变量属于当前线程,每个线程都保存有一个变量副本,每个线程的变量都不同。
ThreadLocal<String> threadLocal = new ThreadLocal<>();
threadLocal.set("享学");
System.out.println("主线程获取变量:"+threadLocal.get());
Thread thread = new Thread() {
@Override
public void run() {
super.run();
System.out.println("子线程获取变量:"+ threadLocal.get());
threadLocal.set("教育");
System.out.println("子线程获取变量:"+ threadLocal.get());
}
};
在上述代码中,主线程输出:享学,子线程第一次输出:null,第二次输出教育。ThreadLocal相当于提供了一种线程隔离,将变量与线程相绑定。
set
通过ThreadLocal#set
设置线程本地变量,set的实现为:
public void set(T value) {
Thread t = Thread.currentThread();
ThreadLocalMap map = getMap(t);
if (map != null)
map.set(this, value);
else
createMap(t, value);
}
通过Thread.currentThread()方法获取了当前的线程引用,并传给了getMap(Thread)方法获取一个ThreadLocalMap的实例。
ThreadLocalMap getMap(Thread t) {
return t.threadLocals;
}
可以看到getMap(Thread)方法直接返回Thread实例的成员变量threadLocals。它的定义在Thread内部:
public class Thread implements Runnable {
ThreadLocal.ThreadLocalMap threadLocals = null;
}
每个Thread里面都有一个ThreadLocal.ThreadLocalMap成员变量,也就是说每个线程通过ThreadLocal.ThreadLocalMap与ThreadLocal相绑定,这样可以确保每个线程访问到变量的都是本线程自己的。
获取了ThreadLocalMap实例以后,如果它不为空则调用ThreadLocalMap.ThreadLocalMap 的set方法设值;若为空则调用ThreadLocal 的createMap方法new一个ThreadLocalMap实例并赋给Thread.threadLocals。
void createMap(Thread t, T firstValue) {
// this = ThreadLocal
t.threadLocals = new ThreadLocalMap(this, firstValue);
}
get
而ThreadLocal 的 get 方法,源码如下:
public T get() {
Thread t = Thread.currentThread();
ThreadLocalMap map = getMap(t);
if (map != null) {
ThreadLocalMap.Entry e = map.getEntry(this);
if (e != null)
return (T)e.value;
}
return setInitialValue();
}
同样通过Thread.currentThread()方法获取了当前的线程引用,并传给了getMap(Thread)方法获取一个ThreadLocalMap的实例。 而如果从ThreadLocalMap未能找到当前线程的变量则返回setInitialValue
。
private T setInitialValue() {
T value = initialValue();
Thread t = Thread.currentThread();
ThreadLocalMap map = getMap(t);
if (map != null)
map.set(this, value);
else
createMap(t, value);
return value;
}
在setInitialValue中首先调用 initialValue()方法来获得一个value,然后执行ThreadLocal#set
同样的处理并返回这个value,也就是说可以通过重写ThreadLocal的initialValue方法能够实现在set变量值之前,使用get获取的就是这个initialValue返回的结果。
ThreadLocal<String> threadLocal = new ThreadLocal<String>(){
@Nullable
@Override
protected String initialValue() {
return "享学";
}
};
// 享学
String value = threadLocal.get();
在set/get中其实就是借助ThreadLocalMap实现线程与本地变量的绑定与获取。每个线程都有自己的一个ThreadLocalMap,ThreadLocalMap是一个映射集合,以ThreadLocal为key。
ThreadLocal简化的伪代码为:
class Thread extends Thread {
ThreadLocalMap threadLocals;
}
class ThreadLocal<T> {
public void set(T t) {
Thread thread = Thread.currentThread();
thread.threadLocals.put(this, t);
}
public T get() {
Thread thread = Thread.currentThread();
thread.threadLocals.get(this);
}
}
Java多线程对同一个对象进行操作(字节跳动)
这道题想考察什么?
是否了解Java多线程对同一个对象进行操作与真实场景使用,是否熟悉Java多线程对同一个对象进行操作?
考察的知识点
Java多线程对同一个对象进行操作的概念在项目中使用与基本知识
考生应该如何回答
在多线程环境下,多个线程操作同一对象,本质上就是线程安全问题。因此为了应对线程安全需要对多线程操作的对象加锁。
例如当我们遇到需求:实现三个窗口同时出售20张票。
程序分析:
1、票数要使用一个静态的值。
2、为保证不会出现卖出同一张票,要使用同步锁。
3、设计思路:创建一个站台类Station,继承Thread,重写run方法,在run方法内部执行售票操作。
售票要使用同步锁:即有一个站台卖这张票时,其他站台要等待这张票卖完才能继续卖票!
package com.multi_thread;
//站台类
public class Station extends Thread {
// 通过构造方法给线程名字赋值
public Station(String name) {
super(name);// 给线程起名字
}
// 为了保持票数的一直,票数要静态
static int tick = 20;
// 创建一个静态钥匙
static Object ob = "aa";// 值是任意的
@Override
public void run() {
while (tick > 0) {
// 这个很重要,必须使用一个锁,进去的人会把钥匙拿在手上,出来后把钥匙让出来
synchronized (ob) {
if (tick > 0) {
System.out.println(getName() + "卖出了第" + tick + "张票");
tick--;
} else {
System.out.println("票卖完了");
}
}
try {
// 休息一秒钟
sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
package com.multi_thread;
public class MainClass {
// java多线程同步所的使用
// 三个售票窗口同时出售10张票
public static void main(String[] args) {
// 实例化站台对象,并为每一个站台取名字
Station station1 = new Station("窗口1");
Station station2 = new Station("窗口2");
Station station3 = new Station("窗口3");
// 让每一个站台对象各自开始工作
station1.start();
station2.start();
station3.start();
}
}
程序运行结果:
窗口1卖出了第20张票
窗口3卖出了第19张票
窗口2卖出了第18张票
窗口2卖出了第17张票
窗口3卖出了第16张票
窗口1卖出了第15张票
窗口1卖出了第14张票
窗口3卖出了第13张票
窗口2卖出了第12张票
窗口1卖出了第11张票
窗口3卖出了第10张票
窗口2卖出了第9张票
窗口1卖出了第8张票
窗口3卖出了第7张票
窗口2卖出了第6张票
窗口1卖出了第5张票
窗口3卖出了第4张票
窗口2卖出了第3张票
窗口3卖出了第2张票
窗口1卖出了第1张票