Android并发编程高级面试题汇总(含详细解析 八)
  CpwfxCg9mmk0 2023年11月22日 13 0

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。

Android并发编程高级面试题汇总(含详细解析 八)_System

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张票

Android并发编程高级面试题汇总(含详细解析 八)_Java_02

【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 2023年11月22日 0

暂无评论

CpwfxCg9mmk0