android 线程范围内共享变量以及ThreadLocal的使用

来源:互联网 发布:血龙狂舞知乎 编辑:程序博客网 时间:2024/05/17 03:38

线程在java中是一个重头戏,算是比较难的一快,特别是并发哪一块,关于并发这一块,项目上几乎也没用到,今天是讲线程范围内的共享变量,突然听到这个概念,可能心里有点发愣,打个简单比方:有三个线程,每一个产生一个数据,有三个模块分别取获取每个线程产生的数据,在java中其实已经有现成的类给我们解决了此方案 哪就是ThreadLoacl类,我们先不用java提供的类先手动解决下

package com.kge;import java.util.Random;public class ThreadLocalDemo {private static int data = 0;public static void main(String[] args) {for(int i=0;i<4;i++){new Thread(new Runnable() {@Overridepublic void run() {data = new Random().nextInt();System.out.println(Thread.currentThread().getName()+"has put data:"+data);new A().get();new B().get();}}){}.start();}}static class A{public void get(){System.out.println("A from"+Thread.currentThread().getName()+"data="+data);}}static class B{public  void get(){System.out.println("B from"+Thread.currentThread().getName()+"data="+data);}}}

打印结果:


看到这个结果 此时心都凉了,怎么和我预期的效果不一样,其实仔细看代码不能发现这是线程安全问题导致的,稍微分析一下就知道了程序运行是这样的,


发现程序并不是按照我们正常从一而终这样顺序运行的,每当线程运行到给data变量赋值时,另外一个线程就进来了,依次类推,这样会导致前生成的data变量值会被后的值覆盖,哪我们就换个方案,我们把每一个线程产生的数据和这个线程一一对应,哪就用到集合存储了,代码修改后如下:

public class ThreadLocalDemo {private static int data = 0;private static Map<Thread,Integer> maps = new HashMap<Thread,Integer>();public static void main(String[] args) {for(int i=0;i<4;i++){new Thread(new Runnable() {@Overridepublic void run() {data = new Random().nextInt();System.out.println(Thread.currentThread().getName()+"has put data:"+data);maps.put(Thread.currentThread(),data);new A().get();new B().get();}}){}.start();}}static class A{public void get(){int data = maps.get(Thread.currentThread());System.out.println("A from"+Thread.currentThread().getName()+"data="+data);}}static class B{public  void get(){int data = maps.get(Thread.currentThread());System.out.println("B from"+Thread.currentThread().getName()+"data="+data);}}}


发现这个方案也是有问题的,问题出在哪呢?和之前分析的到底一样,哪现在怎么改呢?此刻无限的沉思之中,怎么办?还有什么办法呢?其实只要解决一个问题 就可以把所有的问题解决掉,哪就是变量data属于每个线程自己的属性,这个时候就是考察你基本功好不好的关键时刻了,其实只要把data变量不要做成类变量就可以了,

int data = new Random().nextInt();

好了问题解决了,现在改用java给我们提供好的ThreadLocal类

public class ThreadLocalDemo {private static int data = 0;static ThreadLocal<Integer> local = new ThreadLocal<Integer>();public static void main(String[] args) {for(int i=0;i<4;i++){new Thread(new Runnable() {@Overridepublic void run() {int data = new Random().nextInt();System.out.println(Thread.currentThread().getName()+"has put data:"+data);local.set(data);new A().get();new B().get();}}){}.start();}}static class A{public void get(){data = local.get();System.out.println("A from"+Thread.currentThread().getName()+"data="+data);}}static class B{public  void get(){data = local.get();System.out.println("B from"+Thread.currentThread().getName()+"data="+data);}}}
看看多么简单就那么几行代码搞定,这就是java语言的强大之处,什么事都给你封装好了,如果线程共享有很多个变量,这个时候怎么办,哪直接把这些变量封装到一个类中,因为ThreadLcoal类可以保存任意的值!


0 0
原创粉丝点击