如果多个线程使用同一个数据,那么如何保证线程范围内的数据共享。
我们可以使用一个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 = 0;
//ThreadLocal 提供了线程范围内的数据共享 只能存贮一个数据
private static Map<Thread, Integer> threadData = new HashMap<Thread, Integer>();
//private static ThreadLocal<Integer> x = new ThreadLocal<Integer>();
public static void main(String[] args) {
for (int i = 0; i < 2; i++) {
new Thread(new Runnable() {
@Override
public void run() {
int data = new Random().nextInt();
System.out.println(Thread.currentThread().getName()
+ " has put data :" + data);
threadData.put(Thread.currentThread(), data);
//x.set(data);
new A().get();
new B().get();
}
}).start();
}
}
static class A {
public void get() {
int data = threadData.get(Thread.currentThread());
//int data = x.get();
System.out.println("A from " + Thread.currentThread().getName()
+ " get data :" + data);
}
}
static class B {
public void get() {
int data = threadData.get(Thread.currentThread());
//int data = x.get();
System.out.println("B from " + Thread.currentThread().getName()
+ " get data :" + data);
}
}
}
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>
该类提供了线程局部 (thread-local) 变量。这些变量不同于它们的普通对应物,因为访问某个变量(通过其 get 或 set 方法)的每个线程都有自己的局部变量,它独立于变量的初始化副本。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 = 0;
// ThreadLocal 提供了线程范围内的数据共享
public static void main(String[] args) {
for (int i = 0; i < 2; i++) {
new Thread(new Runnable() {
@Override
public void run() {
int data = new Random().nextInt();
System.out.println(Thread.currentThread().getName()
+ " has put data :" + data);
MyThreadScopeData.getThreadScopeInstance().setName(
"name = " + data);
MyThreadScopeData.getThreadScopeInstance().setAge(data);
new A().get();
new B().get();
}
}).start();
}
}
static class A {
public void get() {
MyThreadScopeData myThreadScopeData = MyThreadScopeData
.getThreadScopeInstance();
System.out.println("A from " + Thread.currentThread().getName()
+ " Name :" + myThreadScopeData.getName() + "age = "
+ myThreadScopeData.getAge());
}
}
static class B {
public void get() {
MyThreadScopeData myThreadScopeData = MyThreadScopeData
.getThreadScopeInstance();
System.out.println("B from " + Thread.currentThread().getName()
+ " Name :" + myThreadScopeData.getName() + "age = "
+ myThreadScopeData.getAge());
}
}
}
class MyThreadScopeData {
private MyThreadScopeData() {
}
private static ThreadLocal<MyThreadScopeData> map = new ThreadLocal<MyThreadScopeData>();
// 此处不需要synchronized 应该每个线程操作各自的数据
public static MyThreadScopeData getThreadScopeInstance() {
// 返回此线程局部变量的当前线程副本中的值。
MyThreadScopeData instance = map.get();
if (instance == null) {
instance = new MyThreadScopeData();
// 将此线程局部变量的当前线程副本中的值设置为指定值。
map.set(instance);
}
return instance;
}
private String name;
private int age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
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
原文:http://blog.csdn.net/fengshizty/article/details/40921913