ThreadLocal的使用

本文详细介绍了ThreadLocal的作用和目的,即实现线程内的数据共享,并通过两个具体示例展示了如何使用ThreadLocal来为每个线程提供独立的数据副本。
ThreadLocal 的作用和目的:用于实现线程内的数据共享,即对于相同的程序代码,多个模块在同一个线程中运行时要共享一份数据,而在另外线程中运行时又共享另外一份数据。

            每个线程调用全局 ThreadLocal 对象的 set 方法,在 set 方法中,首先根据当前线程获取当前线程的ThreadLocalMap 对象,然后往这个 map 中插入一条记录, key 其实是 ThreadLocal 对象, value 是各自的 set方法传进去的值。也就是每个线程其实都有一份自己独享的 ThreadLocalMap 对象,该对象的 Key 是 ThreadLocal对象,值是用户设置的具体值。


ThreadLocal的使用示例:

//第一个示例:public class ThreadLocalNum{
    private static int nextNum = 0;    private static ThreadLocal threadLocalNum= new ThreadLocal() {
        protected synchronized Object initialValue() {            return new Integer(nextNum++);        }    }    //获取ThreadLocal中的独享对象    public static int get() {        return ((Integer) (threadLocalNum.get())).intValue();    }}
//第二个示例:
public class ThreadLocalDemo implements Runnable{
	ThreadLocal<Student> threadLocal = new ThreadLocal<Student>();//独享student的ThreadLocal

	public void run() {
		String currentThreadName = Thread.currentThread().getName();
		System.out.println(currentThreadName + "is running--------------------");
		Random random = new Random();
		int a = random.nextInt(100);
		System.out.println(currentThreadName + "is set age :" + a);
		Student stu = getStu();
		stu.setAge(a);
		System.out.println(currentThreadName + "is first get age :" + stu.getAge());
		try {
			Thread.sleep(1000);
		} catch (Exception e) {
			e.printStackTrace();
		}
		System.out.println(currentThreadName + "id second get age :" + stu.getAge());
	}

	
	public static void main(String[] args) {
		ThreadLocalDemo demo = new ThreadLocalDemo();
		Thread t1 = new Thread(demo,"t1111111");
		Thread t2 = new Thread(demo,"t222222222");
		t1.start();
		t2.start();
	}
	
	private Student getStu() {
		Student stu = threadLocal.get();
		if (stu == null) {
			stu = new Student();
			threadLocal.set(stu);
		}
		return stu;
	}	
}
class Student{
	private int age;

	public int getAge() {
		return age;
	}

	public void setAge(int age) {
		this.age = age;
	}
}



ThreadLocal能够存储数据,其作用域是线程,可通过以下几种方式使用: - **基本使用及原理**:每个ThreadLocal会创建一个ThreadLocalMap,以线程作为ThreadLocalMap的key,要存储的局部变量作为value,以此实现各个线程的局部变量隔离 [^2]。 - **结合业务对象使用**:将ThreadLocal放在业务对象里可体现高内聚,能让每一个线程都有一个独立的业务对象。使用时,通过特定方法如`ThreadLocalDemo.getInstance()`就能得到当前线程所需的值。例如以下代码: ```java package com.yanghs.test.traditional; /** * @author yanghs * @Description: * @date 2018/3/31 16:24 */ public class ThreadLocalTest { public static void main(String[] args) { for(int i=0; i<2;i++){ new Thread(new Runnable() { @Override public void run() { Double d = Math.random()*10; ThreadLocalDemo.getThreadInstance().setName("name"+d); new A().get(); new B().get(); } }).start(); } } static class A{ public void get(){ System.out.println(ThreadLocalDemo.getThreadInstance().getName()); } } static class B{ public void get(){ System.out.println(ThreadLocalDemo.getThreadInstance().getName()); } } } ``` Struts2的ActionContext就是使用这种方式绑定数据 [^3]。 - **应对特定需求的使用**:当需求是使用最高效的方式实现1000个时间的格式化时,可使用ThreadLocal创建线程级别变量,如创建10个SimpleDateFormat对象。不过具体使用该方案还是使用加锁方案,取决于线程及变量的复用率 [^4]。 - **避免内存泄漏的使用**:在线程池场景中,由于线程会重复使用,若忘记调用`remove()`,会导致ThreadLocal对象无法被回收。因此,要在`finally`块中调用`remove()`,示例代码如下: ```java try { // 使用 ThreadLocal } finally { threadLocal.remove(); } ``` 同时,使用`InheritableThreadLocal`(父子线程继承变量)时需谨慎 [^5]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值