1.线程范围内共享变量

  

1.1 前奏:

使用一个Map来实现线程范围内共享变量

  

public class ThreadScopeShareData {

    static Map<Thread, Integer> dataMap = new HashMap<Thread, 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()
+ " put data " + data);
dataMap.put(Thread.currentThread(), data);
new A().get();
new B().get();
}
}).start();
}
} static class A {
public void get() {
System.out.println(Thread.currentThread().getName() + " get data "
+ dataMap.get(Thread.currentThread()));
}
} static class B {
public void get() {
System.out.println(Thread.currentThread().getName() + "get data "
+ dataMap.get(Thread.currentThread()));
}
} }

  

1.2 ThreadLocal类实际上就是一种map

/**
* ThreadLocal 类 这里ThreadLocal存放一个变量,如果有多个变量, 可以先将多个变量封装为一个对象
*
* @author Administrator
*
*/
public class ThreadLocalTest {
static ThreadLocal<Integer> x = new 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()
+ " put data " + data);
x.set(data);
new A().get();
new B().get();
}
}).start();
}
} static class A {
public void get() {
System.out.println(Thread.currentThread().getName() + " get data "
+ x.get());
}
} static class B {
public void get() {
System.out.println(Thread.currentThread().getName() + "get data "
+ x.get());
}
} }

 2.线程范围内共享多个变量,可以将多个变量封装为一个对象

/**
* ThreadLocal 类 这里ThreadLocal存放一个变量,如果有多个变量, 可以先将多个变量封装为一个对象
*
* @author Administrator
*
*/
public class ThreadLocalTest {
static ThreadLocal<Integer> x = new 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()
+ " put data " + data);
x.set(data);
MyThreadScopeData myData = MyThreadScopeData.getThreadInstance();//获取与线程绑定的对象
myData.setName("name"+data);
myData.setAge(data);
System.out.println(Thread.currentThread().getName()
+ " put Object " + "name: "+myData.getName()+","+" age: "+myData.getAge());
new A().get();
new B().get();
}
}).start();
}
} static class A {
public void get() {
System.out.println(Thread.currentThread().getName() + " get data "
+ x.get());
MyThreadScopeData instance = MyThreadScopeData.getThreadInstance(); //直接获取与该线程相关的对象
System.out.println(Thread.currentThread().getName() + " get Object "
+ "name: "+instance.getName()+","+" age: "+instance.getAge());
}
} static class B {
public void get() {
System.out.println(Thread.currentThread().getName() + "get data "
+ x.get());
MyThreadScopeData instance = MyThreadScopeData.getThreadInstance(); //直接获取与该线程相关的对象
System.out.println(Thread.currentThread().getName() + " get Object "
+ "name: "+instance.getName()+","+" age: "+instance.getAge());
}
} } // 单例
class MyThreadScopeData { //类的实例是与线程相关的,那么类的设计就交由类自身完成,只要调用自然就是与线程有关的
private static ThreadLocal<MyThreadScopeData> map = new ThreadLocal<>(); private MyThreadScopeData() { } public static MyThreadScopeData getThreadInstance() { // 线程间是相互独立的,这里不需要考虑同步
MyThreadScopeData instance = map.get();
if (instance == null) {
instance = new MyThreadScopeData();
map.set(instance);
}
return instance;
} private String name;
private Integer age; /**
* @return the name
*/
public String getName() {
return name;
} /**
* @param name
* the name to set
*/
public void setName(String name) {
this.name = name;
} /**
* @return the age
*/
public Integer getAge() {
return age;
} /**
* @param age
* the age to set
*/
public void setAge(Integer age) {
this.age = age;
} }

打印结果

Thread-1 put data -723086824
Thread-0 put data 772514756
Thread-1 put Object name: name-723086824, age: -723086824
Thread-0 put Object name: name772514756, age: 772514756
Thread-0 get data 772514756
Thread-1 get data -723086824
Thread-0 get Object name: name772514756, age: 772514756
Thread-1 get Object name: name-723086824, age: -723086824
Thread-0get data 772514756
Thread-1get data -723086824
Thread-0 get Object name: name772514756, age: 772514756
Thread-1 get Object name: name-723086824, age: -723086824

类的实例是与线程相关的,那么类的设计就交由类自身完成,只要调用自然就是与线程有关的  strust2的主要思想就是这么设计的

参看JAVA API

ThreadLocal有一个 remove()方法

可以移除与该线程相关的变量

remove()
Removes the current thread's value for this thread-local variable.

补充:

  虚拟机的对应类 Runtime ,中有一个方法 addShutdownHook(Thread hook)

  addShutdownHook(Thread hook)
  Registers a new virtual-machine shutdown hook.

  例如可以写一个发送邮件的线程Thread,当虚拟机挂掉之前会调用传入的Thread,发送一封邮件。

线程中是不是也应该有这种机制,当一个线程挂掉之前可以执行一个之前注册好的事件,或者有一个监听器在监听线程的状态,从而进行回调

在获取到线程挂掉的通知,就可以把该线程相关的变量全部remove获取clear掉

多个线程访问共享对象和数据的方式 (启用4个线程,其中2个线程对j加1,2个线程对j减1)

 如果每个线程执行的代码相同,可以使用同一个Runnable对象,这个Runnable对象中有那个共享数据,例如卖票系统可以这个做。

第一种方式:将共享数据封装在另外一个对象中,然后将这个对象逐一传递给各个Runnable对象。每个线程对共享数据的操作方法也分配到那个对象身上去完成,这样容易实现针对该数据进行的各个操作的互斥和通信。

public class MutilThreadShareData {

    public static void main(String[] args) {
final ShareData1 data1 = new ShareData1(); //两个线程操作同一个对象
for(int i = 0;i<2;i++){
new Thread(new Runnable1(data1)).start();
new Thread(new Runnable2(data1)).start();
}
}
static class Runnable1 implements Runnable{
private ShareData1 data1; public Runnable1(ShareData1 data1) {
this.data1 = data1;
}
@Override
public void run() {
data1.increment();
}
}
static class Runnable2 implements Runnable{
private ShareData1 data1; public Runnable2(ShareData1 data1) {
this.data1 = data1;
}
@Override
public void run() {
data1.decrement();
}
}
} class ShareData1{
private int j = 0; void increment(){
j++;
System.out.println(j);
}
void decrement(){
j--;
System.out.println(j);
} }

  第二种: 将这些Runnable对象作为某一个类中的内部类,共享数据作为这个外部类中的成员变量,每个线程对共享数据的操作方法也分配给外部类,以便实现对共享数据进行的各个操作的互斥和通信,作为内部类的各个Runnable对象调用外部类的这些方法。

public class MutilThreadShareData {

    private static ShareData1 data1 = new ShareData1(); //两个线程操作同一个对象
public static void main(String[] args) {
for(int i = 0;i<2;i++){
new Thread(new Runnable() {
@Override
public void run() {
data1.increment();
}
}).start();
new Thread(new Runnable() { @Override
public void run() {
data1.decrement();
}
}).start();
}
}
} class ShareData1{
private int j = 0; void increment(){
j++;
System.out.println(j);
}
void decrement(){
j--;
System.out.println(j);
} }

第三种:

public class MutilThreadShareData {

    private static int j = 0;
public static void main(String[] args) {
for(int i = 0;i<2;i++){
new Thread(new Inc()).start();
new Thread(new Dec()).start();
}
}
static class Inc implements Runnable{
@Override
public void run() {
for(int i = 0;i<100;i++){
j++;
System.out.println(j);
}
}
}
static class Dec implements Runnable{
@Override
public void run() {
for(int i = 0;i<100;i++){
j--;
System.out.println(j);
}
}
}
}

一个外部类有两个内部类,两个内部类如何共享数据,都操作外部类的成员变量得到共享数据的目的

Java线程与并发库高级应用-线程范围内共享数据ThreadLocal类的更多相关文章

  1. Java多线程与并发库高级应用-线程池

    线程池 线程池的思想  线程池的概念与Executors类的应用 > 创建固定大小的线程池 > 创建缓存线程池 > 创建单一线程池(如何实现线程死掉后重新启动?) 关闭线程池 > ...

  2. Java多线程与并发库高级应用-java5线程并发库

    java5 中的线程并发库 主要在java.util.concurrent包中 还有 java.util.concurrent.atomic子包和java.util.concurrent.lock子包 ...

  3. Java多线程与并发库高级应用-工具类介绍

    java.util.concurrent.Lock 1.Lock比传统线程模型中的synchronized方式更加面向对象,与生活中的锁类似,锁本身也应该是一个对象.两个线程执行的代码片段要实现同步互 ...

  4. Java多线程与并发库高级应用-Callable与Future的应用

    Callable这种任务可以返回结果,返回的结果可以由Future去拿 >Future取得的结果类型和Callable返回的结果类型必须一致,这是通过泛型来实现的. >Completion ...

  5. Java多线程与并发库高级应用-传统线程机制回顾

    1.传统线程机制的回顾 1.1创建线程的两种传统方式 在Thread子类覆盖的run方法中编写运行代码 // 1.使用子类,把代码放到子类的run()中运行 Thread thread = new T ...

  6. Java多线程与并发库高级应用-传统线程同步通信技术

    面试题: 子线程循环10次,接着主线程循环100次,接着又回到子线程循环10次,接着又 主线程循环100次,如此循环50次,请写出程序 /** * 子线程循环10次,接着主线程循环100次,接着又回到 ...

  7. Java多线程与并发库高级应用-传统线程互斥技术

     线程安全问题: 多个线程操作同一份数据的时候,有可能会出现线程安全问题.可以用银行转账来解释. 模拟线程安全问题 /** * 启动两个线程分别打印两个名字,名字按照字符一个一个打印 * * @aut ...

  8. Java多线程与并发库高级应用-面试题

    第一题:现有的程序代码模拟产生了16个日志对象,并且需要运行16秒才能打印完这些日志,请在程序中增加4个线程去调用parseLog()方法来分头打印这16个日志对象,程序只需要运行4秒即可打印完这些日 ...

  9. Java多线程与并发库高级应用-同步集合

    ArrayBlockingQueue LinkedBlockingQueue 数组是连续的一片内存 链表是不连续的一片内存  传统方式下用Collections工具类提供的synchronizedCo ...

随机推荐

  1. BZOJ 1208: [HNOI2004]宠物收养所

    1208: [HNOI2004]宠物收养所 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 7514  Solved: 2982[Submit][Sta ...

  2. 10 Things Every Java Programmer Should Know about String

    String in Java is very special class and most frequently used class as well. There are lot many thin ...

  3. PAT 1010. 一元多项式求导 (25)

    设计函数求一元多项式的导数.(注:xn(n为整数)的一阶导数为n*xn-1.) 输入格式:以指数递降方式输入多项式非零项系数和指数(绝对值均为不超过1000的整数).数字间以空格分隔. 输出格式:以与 ...

  4. PAT 1003. 我要通过!(20) JAVA

    参考http://blog.csdn.net/bin8632/article/details/50216297 答案正确"是自动判题系统给出的最令人欢喜的回复.本题属于PAT的"答 ...

  5. Broadmann area (wiki)

    Source: https://en.wikipedia.org/wiki/Brodmann_area Lateral surface Medial serface Areas 3, 1 & ...

  6. Jquery操作下拉框(DropDownList)实现取值赋值

    Jquery操作下拉框(DropDownList)想必大家都有所接触吧,下面与大家分享下对DropDownList进行取值赋值的实现代码 1. 获取选中项: 获取选中项的Value值: $('sele ...

  7. mediaplayer与surfaceView,无法播放问题

    mediaplayer需要在surfaceView创建之后才能创建,不然会导致错误. surfaceholder = msurface.getHolder(); surfaceholder.setKe ...

  8. 详细学习ORACLE JOBS

    一点一点学习jobs的各个方面比较长,比较烦,但是应该看完后会对jobs比较好的应用 一.学习准备 开始dbms_job学习前,先认识一个参数job_queue_processes a.job_que ...

  9. 实现Linux与Windows下一致的命令行

    这其实是个非常简单的东西. 我们会写一些命令行的工具,一般跨平台的话,会用python或者perl写,比如叫foo.py,然后在Windows和Linux下调用这个脚本: Linux: foo.py ...

  10. Web软件安全攻击