ThreadLocal保证线程安全的原理

1、线程安全问题
    这里我们主要关注的是Servlet的线程安全,我们知道Servlet是用来处理用户http请求的。当web容器接收到一个对Servlet的请求时,web容器就会分配一个工作线程来处理请求,在执行时,如果又有一个请求进来,web同样会再分配一个线程去响应,而不管这个请求和上一个请求是不是对同一个Servlet的请求。Web容器出于效率和节省内存的考虑,在其中只会保存Servlet的一个实例,这也就意味着,任何Servlet类的代码会在一个多线程环境下执行。这样在Servlet类中定义一个变量并使用就会引发线程安全问题(正因为如此,我们创建Servlet的子类来处理请求时,是不能定义全局变量的)。
2、ThreadLocal如何保证线程安全
    ThreadLocal本身并没有承担存储每个线程中的数据的职责,它是通过操作每个线程内部的一个“副本”-ThreadLocalMap来实现线程之间的隔离,从而保证了线程安全。
    首先我们查看一下Thread类的源码:

ThreadLocal.ThreadLocalMap threadLocals = null;  


这个是设值的方法
 public void set(T value) {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null)
            map.set(this, value);
        else
            createMap(t, value);
    }


 其中的getMap()方法如下:
    ThreadLocalMap getMap(Thread t) {
        return t.threadLocals;
    }

可以看到ThreadLocal操作值的时候是取得当前线程的ThreadLocalMap对象,然后把值设置到了这个对象中,这样对于不同的线程得到的就是不同的ThreadLocalMap,那么向其中保存值或者修改值都只是会影响到当前线程,这样就保证了线程安全。

3、使用实例
    首先我们建立一个类,然后在其中保存一个静态ThreadLocal变量,这样的静态变量在程序的任何位置都可以访问的到。

可以看出同一个线程中的数据是递增的,也就是数据是线程安全的。

4、应用
    使用ThreadLocal来保证线程安全,这种方式被应用在了struts 2框架中(其他框架中也有使用,比如hibernate),我们可以扒开struts 2的源码看一看,ActionContext类中确实存在一个静态的ThreadLocal变量,关于ThreadLocal在struts 2中的应用,大家可以看看《struts 2 Internals》(struts 2技术内幕,陆舟 著,兹认为这是一本不可多得的好书,讲解struts 2非常深入而且易懂)这本书,synchronized关键字也用来解决多线程环境下访问变量的问题,这两者的区别在于ThreadLocal是用空间换取时间,synchronized关键字是用时间换空间。
    

public class SeqCount {
	  private static A a=new A();
	  private static ThreadLocal<Integer> seqCount = new ThreadLocal<Integer>(){
	        // 实现initialValue()
	        public Integer initialValue() {
	            return 0;
	        }
	    };

	    public int nextSeq(){
	        seqCount.set(seqCount.get() + 1);

	        return seqCount.get();
	    }

	    public static void main(String[] args){
	        SeqCount seqCount = new SeqCount();

	        SeqThread thread1 = new SeqThread(seqCount);
	        SeqThread thread2 = new SeqThread(seqCount);
	        SeqThread thread3 = new SeqThread(seqCount);
	        SeqThread thread4 = new SeqThread(seqCount);

	        thread1.start();
	        thread2.start();
	        thread3.start();
	        thread4.start();
	    }

	    private static class SeqThread extends Thread{
	        private SeqCount seqCount;

	        SeqThread(SeqCount seqCount){
	            this.seqCount = seqCount;
	        }

	        public void run() {
	            for(int i = 0 ; i < 3 ; i++){
	                System.out.println(Thread.currentThread().getName() + " seqCount :" + seqCount.nextSeq());
	            }
	        }
	    }
	    
	    
}





评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值