• Android多线程研究(6)——多线程之间数据隔离


    在上一篇《Android多线程研究(5)——线程之间共享数据》中对线程之间的数据共享进行了学习和研究,这一篇我们来看看如何解决多个线程之间的数据隔离问题,什么是数据隔离呢?比如说我们现在开启了两个线程,这两个线程都要同时给同一个全局变量data赋值,各个线程操作它赋值后的变量数据,这里就需要用到隔离。先看一段代码:

    import java.util.Random;
    
    
    public class ThreadLocalTest {
    	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() {
    					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 int get(){
    			System.out.println("A from " + Thread.currentThread().getName() +
    					" has get data: " + data);
    			return data;
    		}
    	}
    	
    	static class B{
    		public int get(){
    			System.out.println("B from " + Thread.currentThread().getName() +
    					" has get data: " + data);
    			return data;
    		}
    	}
    }
    
    运行结果:

    从上面我们可以看到Thread-0和Thread-1都在操作变量data,但是两个线程之间没有做到对数据操作的隔离,所以输出结果中两个线程共用了一个data变量。

    我们将上面代码修改如下:

    import java.util.HashMap;
    import java.util.Map;
    import java.util.Random;
    
    
    public class ThreadLocalTest {
    	//private static int data = 0;
    	private static Map<Thread, Integer> map = 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() {
    					//data = new Random().nextInt();
    					int data = new Random().nextInt();
    					map.put(Thread.currentThread(), data);
    					System.out.println(Thread.currentThread().getName() +
    							" has put data: " + data);
    					new A().get();
    					new B().get();
    				}
    			}).start();
    		}
    	}
    	
    	static class A{
    		public int get(){
    			System.out.println("A from " + Thread.currentThread().getName() +
    					" has get data: " + map.get(Thread.currentThread()));
    			return map.get(Thread.currentThread());
    		}
    	}
    	
    	static class B{
    		public int get(){
    			System.out.println("B from " + Thread.currentThread().getName() +
    					" has get data: " + map.get(Thread.currentThread()));
    			return  map.get(Thread.currentThread());
    		}
    	}
    }
    
    输出结果:

    上面代码中我们用一个Map集合隔离了线程对data数据的操作,其实相当于创建了一个data数据的备份(双份的data)实现了线程之间数据的隔离,其实早在Java 1.2就引入了一个用来支持线程数据隔离的类(java.lang.ThreadLocal),下面我们来看看如何使用ThreadLocal实现线程之间的数据隔离。

    ThreadLocal中的三个方法:‘

    get() :返回当前线程的线程局部变量副本

    protected initialValue() :返回该线程局部变量的当前线程的初始值

    void set(Object value) :设置当前线程的线程局部变量副本的值

    其中initialValue方法是为子类写的方法,在一个线程第一次调用get()或者set()方法时执行,并且仅执行一次。

    import java.util.Random;
    
    
    public class ThreadLocalTest {
    	//private static int data = 0;
    	//private static Map<Thread, Integer> map = new HashMap<Thread, Integer>();
    	private static ThreadLocal<Integer> tl = new ThreadLocal<Integer>();
    	public static void main(String[] args) {
    		for(int i=0; i<2; i++){
    			new Thread(new Runnable() {
    				
    				@Override
    				public void run() {
    					//data = new Random().nextInt();
    					int data = new Random().nextInt();
    					//map.put(Thread.currentThread(), data);
    					tl.set(data);
    					System.out.println(Thread.currentThread().getName() +
    							" has put data: " + data);
    					new A().get();
    					new B().get();
    				}
    			}).start();
    		}
    	}
    	
    	static class A{
    		public int get(){
    			System.out.println("A from " + Thread.currentThread().getName() +
    					" has get data: " + tl.get());
    			return tl.get();
    		}
    	}
    	
    	static class B{
    		public int get(){
    			System.out.println("B from " + Thread.currentThread().getName() +
    					" has get data: " + tl.get());
    			return  tl.get();
    		}
    	}
    }
    
    运行结果:

    上面代码明显少了很多,其实ThreadLocal中底层也是用Map来存储变量副本实现的。




  • 相关阅读:
    python继承__init__函数
    Oracle 用户(user)和模式(schema)的区别【转】
    url增加签名验证防窜改
    Codeforces 每日一练 706C+575H+15C
    Codeforces每日一练 1194D+552C+1117D
    每日一练周赛#2 题解
    AtCoder Beginner Contest 160 题解(F待填坑)
    Codeforces每日一练 1030D+1154E+540D
    Codeforcs 每日一练 678C+527C+1012C
    Codeforces 每日一练922C+725D+1152D
  • 原文地址:https://www.cnblogs.com/lanzhi/p/6468929.html
Copyright © 2020-2023  润新知