首页 > 代码库 > 多线程之线程范围内的数据共享ThreadLocal

多线程之线程范围内的数据共享ThreadLocal

如果多个线程使用同一个数据,那么如何保证线程范围内的数据共享。

我们可以使用一个map来存储当前线程,以及其数据如下:

package andy.thread.traditional.test;

import java.util.HashMap;
import java.util.Map;
import java.util.Random;

/**
 * @author Zhang,Tianyou
 * @version 2014年11月8日 下午2:12:44
 */
// 线程范围内的共享数据
public class ThreadScopeSharedData {

	private static int data = http://www.mamicode.com/0;>
运行效果如下:

Thread-1 has put data :-591164568
Thread-0 has put data :2032497041
A from Thread-0 get data :2032497041
A from Thread-1 get data :-591164568
B from Thread-0 get data :2032497041
B from Thread-1 get data :-591164568

 


但JDK也为我们提供了另一种实现方式:ThreadLocal类,它能够实现各自线程局部变量,并且独立于变量的初始化副本。


public class ThreadLocal<T>
extends Object

该类提供了线程局部 (thread-local) 变量。这些变量不同于它们的普通对应物,因为访问某个变量(通过其 getset 方法)的每个线程都有自己的局部变量,它独立于变量的初始化副本。ThreadLocal 实例通常是类中的 private static 字段,它们希望将状态与某一个线程(例如,用户 ID 或事务 ID)相关联。

例如,以下类生成对每个线程唯一的局部标识符。线程 ID 是在第一次调用 UniqueThreadIdGenerator.getCurrentThreadId() 时分配的,在后续调用中不会更改。

  import java.util.concurrent.atomic.AtomicInteger;

 public class UniqueThreadIdGenerator {

     private static final AtomicInteger uniqueId = new AtomicInteger(0);

     private static final ThreadLocal < Integer > uniqueNum = 
         new ThreadLocal < Integer > () {
             @Override protected Integer initialValue() {
                 return uniqueId.getAndIncrement();
         }
     };
 
     public static int getCurrentThreadId() {
         return uniqueId.get();
     }
 } // UniqueThreadIdGenerator
 

每个线程都保持对其线程局部变量副本的隐式引用,只要线程是活动的并且 ThreadLocal 实例是可访问的;在线程消失之后,其线程局部实例的所有副本都会被垃圾回收(除非存在对这些副本的其他引用)。 



上面代码的更优雅的实现如下:


package andy.thread.traditional.test;

import java.util.Random;

/**
 * @author Zhang,Tianyou
 * @version 2014年11月8日 下午2:26:24
 */

public class ThreadLocalTest {

	private static int data = http://www.mamicode.com/0;>


其运行效果如下:

Thread-1 has put data :1322483490
A from Thread-1 Name :name = 1322483490age = 1322483490
B from Thread-1 Name :name = 1322483490age = 1322483490
Thread-0 has put data :1149801771
A from Thread-0 Name :name = 1149801771age = 1149801771
B from Thread-0 Name :name = 1149801771age = 1149801771



多线程之线程范围内的数据共享ThreadLocal