笔记摘要

所谓线程范围内共享数据,即对于相同的程序代码,多个模块在同一个线程中运行时要共享一份数据,而在另外线程中运行时又共享另外一份数据,API中为我们提供了一个操作线程范围内共享数据的类ThreadLocal,对于线程范围内共享数据的应用,在ThreadLocal的应用场景中进行了介绍,然后主要对它的使用进行讲解,演示了由单一数据的共享到将多个数据封装到一个对象中,然后进行共享。在开始先用一个Map集合简单实现线程范围内数据的共享

1. 使用Map实现线程范围内数据的共享

原理:将线程对象作为map的键存入,这样就保证了map对象的唯一,也就保证了线程内数据的唯一

关键: 明确一点,把当前线程对象作为map集合的键存进去

import java.util.HashMap;
import java.util.Map;
import java.util.Random;  public class ThreadScopeShareData {  private static int data = 0;    //定义一个全局的成员变量  private static Map<Thread, Integer> threadData = 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()   + " has put data :" + data);  //把当前线程对象作为键,就可以保证map对象的唯一,即保证线程内的数据唯一  threadData.put(Thread.currentThread(), data);  new A().get();  new B().get();  }  }).start();  }  }  //定义一个类模拟获取数据  static class A{  public void get(){  int data = threadData.get(Thread.currentThread());  System.out.println("A from " + Thread.currentThread().getName()   + " get data :" + data);  }  }  static class B{  public void get(){  int data = threadData.get(Thread.currentThread());            System.out.println("B from " + Thread.currentThread().getName()   + " get data :" + data);  }         }
}

打印结果

Thread-0 has put data:-49248136
Thread-1 has put data:311124475
A from Thread-0 get data:-49248136
A from Thread-1 get data:311124475
B from Thread-0 get data:-49248136
B from Thread-1 get data:311124475

2. ThreadLocal类

ThreadLocal的作用和目的:

用于实现线程内的数据共享,即对于相同的程序代码,多个模块在同一个线程中运行时要共享一份数据,而在另外线程中运行时又共享另外一份数据。

每个线程调用全局ThreadLocal对象的set方法,就相当于往其内部的map中增加一条记录,key分别是各自的线程,value是各自的set方法传进去的值。在线程结束时可以调用ThreadLocal.clear()方法,这样会更快释放内存,不调用也可以,因为线程结束后也可以自动释放相关的ThreadLocal变量。

3. ThreadLocal的应用场景

1、订单处理包含一系列操作:减少库存量、增加一条流水台账、修改总账,这几个操作要在同一个事务中完成,通常也即同一个线程中进行处理,如果累加公司应收款的操作失败了,则应该把前面的操作回滚,否则,提交所有操作,这要求这些操作使用相同的数据库连接对象,而这些操作的代码分别位于不同的模块类中。

2、 银行转账包含一系列操作:把转出帐户的余额减少,把转入帐户的余额增加,这两个操作要在同一个事务中完成,它们必须使用相同的数据库连接对象,转入和转出操作的代码分别是两个不同的帐户对象的方法。

3、例如Strut2的ActionContext,同一段代码被不同的线程调用运行时,该代码操作的数据是每个线程各自的状态和数据,对于不同的线程来说,getContext方法拿到的对象都不相同,对同一个线程来说,不管调用getContext方法多少次和在哪个模块中getContext方法,拿到的都是同一个。

线程范围内共享数据示意图

实现对ThreadLocal变量的封装, 让外界不要直接操作ThreadLocal变量由于对基本类型的数据的封装,这种应用相对很少见。而对对象类型的数据的封装,比较常见,即让某个类针对不同线程分别创建一个独立的实例对象。所以我们要对数据进行封装。

实现方式一

示例说明:

1、 该示例包含了对基本类型数据的共享和对象类型数据的共享

2、定义一个全局共享的ThreadLocal变量,然后启动多个线程向该ThreadLocal变量中存储一个随机值,接着各个线程调用另外其他多个类的方法,这多个类的方法中读取这个ThreadLocal变量的值,就可以看到多个类在同一个线程中共享同一份数据。

3、但这里每次存储数据时,都是使用同一个ThreadLocal对象,只是重新赋值而已

import java.util.HashMap;
import java.util.Map;
import java.util.Random;  public class ThreadLocalTest {  private static ThreadLocal<Integer> x = new ThreadLocal<Integer>();  //创建一个存储封装类对象的ThreadLocal  private static ThreadLocal<MyThreadScopeData> myThreadScopeData = new  ThreadLocal<MyThreadScopeData>();  private static int data = 0;  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);  x.set(data);      //共享多个数据  //将数据封装在myData对象中,并将myData作为myThreadScopeData的键  MyThreadScopeData myData = new MyThreadScopeData();  myData.setName("name "+data);  myData.setAge(data);  myThreadScopeData.set(myData);  new A().get();  new B().get();  }  }).start();  }  }  static class A{  public void get(){  int data = x.get();  System.out.println("A from "+Thread.currentThread().getName()+" get data :"+data);  //从myData中取出数据,并获取当前线程名,数据  MyThreadScopeData myData = myThreadScopeData.get();  System.out.println("A from "+Thread.currentThread().getName()+" getMyData: " +   myData.getName() + "," +myData.getAge());  }  }  static class B{  public void get(){  int data = x.get();  System.out.println("B from "+Thread.currentThread().getName()+" get data :"+data);  MyThreadScopeData myData = myThreadScopeData.get();  System.out.println("B from "+Thread.currentThread().getName()+" getMyData: " +   myData.getName() + "," +myData.getAge());  }  }
}  //封装数据的类
class MyThreadScopeData{  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-0has put data : 1317043235
Thread-1has put data : -969579752
A from Thread-0 get data :1317043235
A from Thread-1 get data :-969579752
A from Thread-0 getMyData: name 1317043235,1317043235
A from Thread-1 getMyData: name -969579752,-969579752
B from Thread-0 get data :1317043235
B from Thread-1 get data :-969579752
B from Thread-0 getMyData: name 1317043235,1317043235
B from Thread-1 getMyData: name -969579752,-969579752

实现方式二

示例说明:

这里模拟原始的单例模式,它们的区别是:单例模式中只有唯一的一个实例,而这里是每个线程拥有自己唯一的实例,只要是已经创建,就直接返回,保证每个线程拥有自己的唯一一份实例

优点:

这里可以返回每个线程自己唯一的实例对象,所以不必在外面定义,当在代码中的任意地方想获取到一个可以存储自己数据的线程实例的时候直接去调用getThreadInstance方法即可,直接定义在数据对象的内部,和数据关系更紧密,而方式一,则每次想存入数据的时候都需要在外面创建一个ThreadLocal对象用于存储数据。所以方式二更具封装性。

package cn.itcast.heima2;  import java.util.HashMap;
import java.util.Map;
import java.util.Random;  public class ThreadLocalTest {  //创建一个ThreadLocal对象  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);  x.set(data);    //往当前线程存入一条数据  //获取与当前线程绑定的实例并设置值  MyThreadScopeData.getThreadInstance().setName("name:" + data);  MyThreadScopeData.getThreadInstance().setAge(data);  new A().get();  new B().get();  }  }).start();  }  }  static class A{  public void get(){  int data = x.get();     //获取当前线程中的数据  System.out.println("A from " + Thread.currentThread().getName()   + " get data :" + data);  //获取与当前线程绑定的实例  MyThreadScopeData myData = MyThreadScopeData.getThreadInstance();  System.out.println("A from " + Thread.currentThread().getName()   + " getMyData: " + myData.getName() + "," +  myData.getAge());  }  }  static class B{  public void get(){  int data = x.get();           System.out.println("B from " + Thread.currentThread().getName()   + " get data :" + data);  MyThreadScopeData myData = MyThreadScopeData.getThreadInstance();  System.out.println("B from " + Thread.currentThread().getName()   + " getMyData: " + myData.getName() + ",age: " +  myData.getAge());             }         }
}  //一个绑定当前线程的类
class MyThreadScopeData{  private MyThreadScopeData(){}   //构造方法私有化  private static ThreadLocal<MyThreadScopeData> map = new ThreadLocal<MyThreadScopeData>();  //定义一个静态方法,返回各线程自己的实例   //这里不必用同步,因为每个线程都要创建自己的实例,所以没有线程安全问题。  public static /*synchronized*/ MyThreadScopeData getThreadInstance(){  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-0 has put data :2105117242
Thread-1 has put data :-368218341
A from Thread-1 get data :-368218341
A from Thread-1 getMyData: name:-368218341,-368218341
A from Thread-0 get data :2105117242
A from Thread-0 getMyData: name:2105117242,2105117242
B from Thread-0 get data :2105117242
B from Thread-1 get data :-368218341
B from Thread-0 getMyData: name:2105117242,age: 2105117242
B from Thread-1 getMyData: name:-368218341,age: -368218341

4. 总结

一个ThreadLocal代表一个变量,故其中只能放一个数据,有两个变量都要线程范围内共享,则要定义两个ThreadLocal对象,如果数据更多就很麻烦,可以先定义一个对象封装变量,然后在ThreadLocal中存储这一个对象,而这些操作都在提供线程数据类中完成

Java高并发编程:线程范围内共享数据相关推荐

  1. Java高并发编程详解系列-Java线程入门

    根据自己学的知识加上从各个网站上收集的资料分享一下关于java高并发编程的知识点.对于代码示例会以Maven工程的形式分享到个人的GitHub上面.   首先介绍一下这个系列的东西是什么,这个系列自己 ...

  2. Java高并发编程 (马士兵老师视频)笔记(一)同步器

    本篇主要总结同步器的相关例子:包括synchronized.volatile.原子变量类(AtomicXxx).CountDownLatch.ReentrantLock和ThreadLocal.还涉及 ...

  3. Java高并发入门-线程初步

    Java高并发入门-线程初步 线程与进程之间的关系 进程就是我们运行在计算机上的一个程序,对应Java程序来说就是运行在计算机上的Java应用程序,这个程序在运行的时候就会创建了一个进程,服务器上就会 ...

  4. Java高并发编程学习(三)java.util.concurrent包

    简介 我们已经学习了形成Java并发程序设计基础的底层构建块,但对于实际编程来说,应该尽可能远离底层结构.使用由并发处理的专业人士实现的较高层次的结构要方便得多.要安全得多.例如,对于许多线程问题,可 ...

  5. 高并发编程-线程通信_使用wait和notify进行线程间的通信2_多生产者多消费者导致程序假死原因分析

    文章目录 概述 jstack或者可视化工具检测是否死锁(没有) 原因分析 概述 高并发编程-线程通信_使用wait和notify进行线程间的通信 - 遗留问题 我们看到了 应用卡住了 .... 怀疑是 ...

  6. 29W 字总结阿里 Java 高并发编程:案例 + 源码 + 面试 + 系统架构设计

    下半年的跳槽季已经开始,好多同学已经拿到了不错的 Offer,同时还有一些同学对于 Java 高并发编程还缺少一些深入的理解,不过不用慌,今天老师分享的这份 27W 字的阿里巴巴 Java 高并发编程 ...

  7. 多线程学习(四)-线程范围内共享数据

    一:线程范围内共享数据: 如果每个线程执行的代码相同,可以使用同一个Runnable对象,这个Runnable对象中有那个共享数据,例如,买票系统就可以这么做. 如果每个线程执行的代码不同,这时候需要 ...

  8. Java高并发入门-线程初步(二)

    Java高并发入门详细讲解 上期回顾及问题总结 上次说了创建线程的两种常用方式,第三种方式在后面的更新中会讲解到.这里对于上一节的内容做个回顾. 在上一节中说到了创建多线程的问题,分析了Thread的 ...

  9. Java高并发编程:活跃性危险

    Java高并发程序中,不得不出现资源竞争以及一些其他严重的问题,比如死锁.线程饥饿.响应性问题和活锁问题.在安全性与活跃性之间通常存在依赖,我们使用加锁机制来确保线程安全,但是如果过度地使用加锁,则可 ...

最新文章

  1. mvc ajax教程,MVC3----AJAX辅助方法
  2. eureka-server详解
  3. 五款帮助创业者迅速熟悉互联网创业的在线学习工具
  4. Latex: 表格中 自动换行居中
  5. 无线WAPI网络AS鉴权服务器,WAPI是什么意思?苹果iPhone手机启用WAPI有什么作用?...
  6. 初识内存控制器和SDRAM【一文了解】
  7. [转]Windows10 中文版 英文语言包安装失败解决
  8. 【嵌入式基础常识】单片机
  9. Java学习路线:day1 Java语言概述
  10. Matplotlib绘制动图
  11. greatest least 函数
  12. Pytorch 如何 优化/调整 模型参数
  13. PyGame每日一练——五子棋小游戏
  14. AB测试(Test)——原理与实际案例手把手教学
  15. 父母的期许与自己的愿望
  16. CDN真的有用吗?效果好不好
  17. 比较 Java 枚举成员:== 或 equals()?
  18. 学习同学们的博客经验
  19. 英语文档学习颗粒归仓
  20. 网络营销广告的类型(CPC、CPM、CPA、CPS, CTR)

热门文章

  1. ASP.NET应用程序
  2. 数据库分页存储过程(4)
  3. 什么是QoS技术?—Vecloud微云
  4. Eclipse中使用Ctrl键卡顿响应慢的解决方法
  5. python 的基础 学习 第七天 is id 编码的补充
  6. 杭州(含嘉兴,绍兴,金华,湖州,义乌)Uber优步司机奖励政策(1月18日~1月24日)...
  7. GlusterFS常用命令小结
  8. GCD介绍(三): Dispatch Sources
  9. 如何设置GridView的内框线颜色
  10. SQL Server 命令行管理工具:SqlLocalDB.exe