当前位置: 代码迷 >> Android >> Android多线程研究(六)——多线程之间数据隔离
  详细解决方案

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

热度:91   发布时间:2016-04-28 05:21:38.0
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来存储变量副本实现的。




  相关解决方案