`
m635674608
  • 浏览: 5041333 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

ThreadLocal 与 Synchronized 总结

    博客分类:
  • java
 
阅读更多

1.对于同步方法和对象:
无论synchronized关键字加在方法上还是对象上,它取得的锁都是对象,而不是把一段代码或函数当作锁――而且同步方法很可能还会被其他线程的不同对象访问。
解释一下“取得的锁都是对象”的意思:如果一个对象有多个synchronized方法,只要一个线程访问了这个对象中的一个synchronized方法,其它线程不能同时访问这个对象中任何一个synchronized方法,但是对于不是synchronized的方法可以访问。而对于其他线程里的另一个对象可以访问synchronized方法。

2.对于同步块:
synchronized(要锁定的对象) {……} 
当要锁定的对象为this时表示的是调用这个方法的对象。

3.对于同步static 方法:
不管调用方法的是类还是对象,锁定的都是类。即类被锁定了,只要一个线程中的对象或类访问了一个synchronized static方法,其他线程里的不管是类还是对象都不能再访问synchronized static方法了。

注:以上的对象即指一个类的实例。

再通过例子说明一下以上几点:
假设P1、P2是同一个类的不同对象,这个类中定义了以下几种情况的同步块或同步方法,P1、P2就都能够调用他们。
1.  把synchronized当作函数修饰符时,示例代码如下:

Java代码 复制代码 收藏代码
  1. 1.  Public synchronized void method(){      
  2. 2.  //….      
  3. 3.  }    


这也就是同步方法,那这时synchronized锁定的是哪个对象呢?他锁定的是调用这个同步方法对象。也就是说,当一个对象P1在不同的线程中 执行这个同步方法时,他们之间会形成互斥,达到同步的效果。但是这个对象所属的Class所产生的另一对象P2却能够任意调用这个被加了 synchronized关键字的方法。
上边的示例代码等同于如下代码:

Java代码 复制代码 收藏代码
  1. 1.  public void method()      
  2. 2.  {      
  3. 3.  synchronized (this)      //  (1)      
  4. 4.  {      
  5. 5.         //…..      
  6. 6.  }      
  7. 7.  }     


(1)处的this指的是什么呢?他指的就是调用这个方法的对象,如P1。可见同步方法实质是将synchronized作用于object reference。――那个拿到了P1对象锁的线程,才能够调用P1的同步方法,而对P2而言,P1这个锁和他毫不相干,程式也可能在这种情形下摆脱同 步机制的控制,造成数据混乱。
2.同步块,示例代码如下:

Java代码 复制代码 收藏代码
  1. 1.  public void method(SomeObject so) {      
  2. 2.  synchronized(so)      
  3. 3.  {      
  4. 4.         //…..      
  5. 5.  }      
  6. 6.  }     

这时,锁就是so这个对象,谁拿到这个锁谁就能够运行他所控制的那段代码。当有一个明确的对象作为锁时,就能够这样写程式,但当没有明确的对象作为锁,只是想让一段代码同步时,能够创建一个特别的instance变量(他得是个对象)来充当锁:

Java代码 复制代码 收藏代码
  1. 1.  class Foo implements Runnable      
  2. 2.  {      
  3. 3.         private byte[] lock = new byte[0];  // 特别的instance变量      
  4. 4.      Public void method()      
  5. 5.  {      
  6. 6.         synchronized(lock) { //… }      
  7. 7.  }      
  8. 8.  //…..      
  9. 9.  }   

 
注:零长度的byte数组对象创建起来将比任何对象都经济――查看编译后的字节码:生成零长度的byte[]对象只需3条操作码,而Object lock = new Object()则需要7行操作码。
3.将synchronized作用于static 函数,示例代码如下:

Java代码 复制代码 收藏代码
  1. 1.  Class Foo      
  2. 2.  {      
  3. 3.  public synchronized static void method1()   // 同步的static 函数      
  4. 4.  {      
  5. 5.  //….      
  6. 6.  }      
  7. 7.  public void method2()      
  8. 8.  {      
  9. 9.         synchronized(Foo.class)   //  class literal(类名称字面常量)      
  10. 10. }      
  11. 11.        }     

 



代码中的method2()方法是把class literal作为锁的情况,他和同步的static函数产生的效果是相同的,取得的锁很特别,是当前调用这个方法的对象所属的类(Class,而不再是由这个Class产生的某个具体对象了)。
记得在《Effective Java》一书中看到过将 Foo.class和 P1.getClass()用于作同步锁还不相同,不能用P1.getClass()来达到锁这个Class的目的。P1指的是由Foo类产生的对象。
能 够推断:假如一个类中定义了一个synchronized的static函数A,也定义了一个synchronized 的instance函数B,那么这个类的同一对象Obj在多线程中分别访问A和B两个方法时,不会构成同步,因为他们的锁都不相同。A方法的锁是Obj所 属的那个Class,而B的锁是Obj所属的这个对象。
搞清楚synchronized锁定的是哪个对象,就能帮助我们设计更安全的多线程程序。

1、synchronized实例方法

Java代码 复制代码 收藏代码
  1. synchronized void method(){   
  2.   ...   
  3. }  



在功能上,它相当于

Java代码 复制代码 收藏代码
  1. void method() {   
  2.      synchronized (this) {   
  3.     ...   
  4.     }   
  5. }  



2、synchronized类方法

Java代码 复制代码 收藏代码
  1. class Something {   
  2.    static synchronized void method(){   
  3.      ...   
  4.    }   
  5. }  

 

 



在功能上,它相当于

Java代码 复制代码 收藏代码
  1. class Something {   
  2.     static void method() {   
  3.       synchronized (Something.class){   
  4.        ...   
  5.     }   
  6.   }   
  7. }  



3、如果我们也可以自定义方法来实现相当于synchronized类似的功能,代码如下:

Java代码 复制代码 收藏代码
  1. void method() {   
  2.     lock();   
  3.     try{   
  4.        ...   
  5.     }finally{   
  6.        unlock();   
  7.     }   
  8. }  


以上的代码就是一个Before/After Pattern的一种实现方式
举例:
下面中Mutex类这种用来进行共享互斥的机制,一般称为mutex。

Java代码 复制代码 收藏代码
  1. public class Gate {   
  2.     private int counter = 0;   
  3.     private String name = "Nobody";   
  4.     private String address = "Nowhere";   
  5.     private final Mutex mutex = new Mutex();   
  6.     public void pass(String name, String address) { // 并非synchronized   
  7.         mutex.lock();   
  8.         try {   
  9.             this.counter++;   
  10.             this.name = name;   
  11.             this.address = address;   
  12.             check();   
  13.         } finally {   
  14.             mutex.unlock();   
  15.         }   
  16.     }   
  17.     public String toString() { // 并非synchronized   
  18.         String s = null;   
  19.         mutex.lock();   
  20.         try {   
  21.             s = "No." + counter + ": " + name + ", " + address;   
  22.         } finally {   
  23.             mutex.unlock();   
  24.         }   
  25.         return s;   
  26.     }   
  27.     private void check() {   
  28.         if (name.charAt(0) != address.charAt(0)) {   
  29.             System.out.println("***** BROKEN ***** " + toString());   
  30.         }   
  31.     }   
  32. }  



设计简单的Mutex类1:

Java代码 复制代码 收藏代码
  1. public final class Mutex {   
  2.     private boolean busy = false;   
  3.     public synchronized void lock() {   
  4.         while (busy) {   
  5.             try {   
  6.                 wait();   
  7.             } catch (InterruptedException e) {   
  8.             }   
  9.         }   
  10.         busy = true;   
  11.     }   
  12.     public synchronized void unlock() {   
  13.         busy = false;   
  14.         notifyAll();   
  15.     }   
  16. }  



设计完善的Mutex类2:

Java代码 复制代码 收藏代码
  1. public final class Mutex {   
  2.     private long locks = 0;   
  3.     private Thread owner = null;   
  4.     public synchronized void lock() {   
  5.         Thread me = Thread.currentThread();   
  6.         while (locks > 0 && owner != me) {   
  7.             try {   
  8.                 wait();   
  9.             } catch (InterruptedException e) {   
  10.             }   
  11.         }   
  12.         // locks == 0 || owner == me   
  13.         owner = me;   
  14.         locks++;   
  15.     }   
  16.     public synchronized void unlock() {   
  17.         Thread me = Thread.currentThread();   
  18.         if (locks == 0 || owner != me) {   
  19.             return;   
  20.         }   
  21.         // locks > 0 && owner == me   
  22.         locks--;   
  23.         if (locks == 0) {   
  24.             owner = null;   
  25.             notifyAll();   
  26.         }   
  27.     }   
  28. }  




4、我们看到synchronized块的时候,要先思考"synchronized是在保护什么",然后进一步思考"获取谁的锁定来保护的呢"
   要调用synchronized实例方法的纯种,一定会获取thid的锁定。一个实例的锁定,同一个时间内,只能有个线程可以得到。
   如果实例不同,那锁定也不同了。使用synchronized块的时候,特别需要考虑"获取谁的锁定来保护的呢"这种情况。因为
   synchronized需要明确地指明要获取的是哪个对象的锁定。例如:
 

Java代码 复制代码 收藏代码
  1. synchronized (obj) {   
  2.     ...   
  3.   }  
 


  这样的程序代码中,obj就是我们所要获取锁定的对象。请小心这个对象不可心写错,获取错误对象的锁定,就好想要保护自己
  自己的家,却把别人家的门给锁定了。

 

1.区别ThreadLocal 与 synchronized

  • ThreadLocal是一个线程隔离(或者说是线程安全)的变量存储的管理实体(注意:不是存储用的),它以Java类方式表现;
  • synchronized是Java的一个保留字,只是一个代码标识符,它依靠JVM的锁机制来实现临界区的函数、变量在CPU运行访问中的原子性。

两者的性质、表现及设计初衷不同,因此没有可比较性。

2.理解ThreadLocal中提到的变量副本
    事实上,我们向ThreadLocal中set的变量不是由ThreadLocal来存储的,而是Thread线程对象自身保存。当用户调用ThreadLocal对象的set(Object o)时,该方法则通过Thread.currentThread()获取当前线程,将变量存入Thread中的一个Map内,而Map的Key就是当前的ThreadLocal实例。请看源码,这是最主要的两个函数,能看出ThreadLocal与Thread的调用关系:

Java代码 复制代码 收藏代码
  1. public void set(T value) {   
  2.         Thread t = Thread.currentThread();   
  3.         ThreadLocalMap map = getMap(t);   
  4.         if (map != null)   
  5.             map.set(this, value);   
  6.         else  
  7.             createMap(t, value);   
  8. }   
  9.   
  10. ThreadLocalMap getMap(Thread t) {   
  11.         return t.threadLocals;   
  12. }  
public void set(T value) {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null)
            map.set(this, value);
        else
            createMap(t, value);
}

ThreadLocalMap getMap(Thread t) {
        return t.threadLocals;
}


(有兴趣的朋友可以阅读Java的ThreadLocal源码)因此,我们可以知道,所谓的变量副本,即是对Object Reference(对象引用)的拷贝。

3.理解Thread和 ThreadLocal对变量的引用关系
    实际上Thread和ThreadLocal对变量引用关系就像是坐标系中的X轴和Y轴,是从两个维度上来组织对变量的引用的。

  • 首先说Thread。
  •     我们知道一个ThreadOne的执行会贯穿多个方法MethodA、MethodB、MethodC这些方法可能分布于不同的类实例。假设,这些方法分别使用了ThreadLocalA、ThreadLocalB、ThreadLocalC来保存线程本地变量,那么这些变量都存于ThreadOne的Map中,并使用各自的ThreadLocal实例作为key。 因此,可以认为,借助ThreanLocal的set方法,在X轴上,Thread横向关联同一线程上下文中来自多个Method的变量引用副本。



  • 接着说ThreadLocal。
  •     一个MethodA中的X变量将被多个线程ThreadOne、ThreadTwo、ThreadThree所访问。假设MethodA使用ThreadLocal存储X,通过set方法,以ThreadLocal作为key值,将不同线程来访时的不同的变量值引用保存于ThreadOne、ThreadTwo、ThreadThree的各自线程上下文中,确保每个线程有自己的一个变量值。因此,可以认为,ThreadLocal是以Method为Y轴,纵向关联了处于同一方法中的不同线程上的变量。


 

 

1、对于多线程资源共享的问题,同步机制采用了“以时间换空间”的方式,而ThreadLocal采用了“以空间换时间”的方式。前者仅提供一份变量,让不同的线程排队访问,而后者为每一个线程都提供了一份变量,因此可以同时访问而互不影响。


2、ThreadLocal使用场合主要解决多线程中数据因并发产生不一致问题。ThreadLocal为每个线程的中并发访问的数据提供一个副本,通过访问副本来运行业务,这样的结果是耗费了内存,单大大减少了线程同步所带来性能消耗,也减少了线程并发控制的复杂度。

 

3、ThreadLocal不能使用基本数据类型,只能使用Object类型。ThreadLocal的使用比synchronized要简单得多。

ThreadLocal和Synchonized都用于解决多线程并发访问。但是ThreadLocal与synchronized有本质的区别。
synchronized是利用锁的机制,使变量或代码块在某一时该只能被一个线程访问。
而ThreadLocal为每一个线程都提供了变量的副本,使得每个线程在某一时间访问到的并不是同一个对象,这样就隔离了多个线程对数据的数据共享。而Synchronized却正好相反,它用于在多个线程间通信时能够获得数据共享。

 

4、Synchronized用于线程间的数据共享,而ThreadLocal则用于线程间的数据隔离。

 

示例1:

Java代码 复制代码 收藏代码
  1. //线程局部变量   
  2.     protected static ThreadLocal<Connection>  threadLocalCon = new ThreadLocal<Connection>();   
  3.        
  4.     /*  
  5.      * 获取数据库连接  
  6.      */  
  7.     public static  Connection getCon() {   
  8.   
  9.         Connection con = threadLocalCon.get();   
  10.         System.out.println("当前线程名称="+Thread.currentThread().getName()+",con="+con);   
  11.         try {   
  12.             if (con == null || con.isClosed()) {   
  13.                 Class.forName(driver);   
  14.                 con = DriverManager.getConnection(url, username, password);   
  15.                 threadLocalCon.set(con);   
  16.                    
  17.                 System.out.println("当前线程="+Thread.currentThread().getName()+",新建连接"+con);   
  18.                 System.out.println("当前线程="+Thread.currentThread().getName()+",threadLocalCon.get()="+threadLocalCon.get());   
  19.                 try {   
  20.                     System.out.println("当前线程="+Thread.currentThread().getName()+",开始休眠");   
  21.                     Thread.currentThread().sleep(10000);   
  22.                     System.out.println("当前线程="+Thread.currentThread().getName()+",结束休眠");   
  23.                 } catch (InterruptedException e) {   
  24.                     // TODO Auto-generated catch block   
  25.                     e.printStackTrace();   
  26.                 }   
  27.             }   
  28.         } catch (ClassNotFoundException e) {   
  29.             e.printStackTrace();   
  30.         } catch (SQLException e) {   
  31.             e.printStackTrace();   
  32.         }   
  33.         return con;   
  34.     }  
//线程局部变量
	protected static ThreadLocal<Connection>  threadLocalCon = new ThreadLocal<Connection>();
	
	/*
	 * 获取数据库连接
	 */
	public static  Connection getCon() {

		Connection con = threadLocalCon.get();
		System.out.println("当前线程名称="+Thread.currentThread().getName()+",con="+con);
		try {
			if (con == null || con.isClosed()) {
				Class.forName(driver);
				con = DriverManager.getConnection(url, username, password);
				threadLocalCon.set(con);
				
				System.out.println("当前线程="+Thread.currentThread().getName()+",新建连接"+con);
				System.out.println("当前线程="+Thread.currentThread().getName()+",threadLocalCon.get()="+threadLocalCon.get());
				try {
					System.out.println("当前线程="+Thread.currentThread().getName()+",开始休眠");
					Thread.currentThread().sleep(10000);
					System.out.println("当前线程="+Thread.currentThread().getName()+",结束休眠");
				} catch (InterruptedException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
			}
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		} catch (SQLException e) {
			e.printStackTrace();
		}
		return con;
	}

 

 

启动2个线程先后发出请求:



 

小结:

1、线程局部变量threadLocalCon为每一个线程分别提供一个该变量的副本,所以,每个线程第一次访问该变量时,通过get()取得的值都是null。

2、客户端的每一次请求,服务端都会新建一个线程为其服务,一次请求结束,相对应的线程也随之随之结束。

 

示例2、

在getCon()方法中加上synchronized关键字,再次执行(两次请求之间发起时间间隔短于10秒),看效果:

 

 

小结:
 1、synchronized是利用锁的机制,使变量或代码块在某一时该只能被一个线程访问,第一个线程在getCon()执行完成之前,第二个线程只能等待。

 

文章来自于:

http://www.iteye.com/topic/179040

http://huangqiqing123.iteye.com/blog/1428071

http://javabeezer.iteye.com/blog/644561

http://lighter.iteye.com/blog/133136

 

分享到:
评论

相关推荐

    Synchronized与ThreadLocal

    ### Synchronized与ThreadLocal #### 一、Synchronized机制详解 **Synchronized** 是 Java 中一个非常重要的关键字,主要用于实现线程同步。它通过在对象上加锁来确保多个线程能够安全地访问共享资源。 - **作用...

    ThreadLocal

    ThreadLocal通常被用来解决线程共享数据时可能出现的并发问题,避免了使用synchronized关键字进行同步控制的复杂性。 在Java中,ThreadLocal的工作原理是为每个线程创建一个单独的存储空间,每个线程可以独立地读写...

    Java中ThreadLocal的设计与使用

    ThreadLocal的生命周期与创建它的线程紧密相关。一旦线程结束,其ThreadLocal变量及其存储的值也会自动被清理。但是,如果线程持续存在且不调用`remove()`,ThreadLocal变量可能会导致内存泄漏,因为它们占用的内存...

    谈谈Java中的ThreadLocal

     ThreadLocal一般称为线程本地变量,它是一种特殊的线程绑定机制,将变量与线程绑定在一起,为每一个线程维护一个独立的变量副本。通过ThreadLocal可以将对象的可见范围限制在同一个线程内。  跳出误区  需要...

    Java多线程 之 临界区、ThreadLocal.docx

    总结来说,Java中的临界区通过`synchronized`关键字或Lock对象来保证多线程环境下的数据一致性,而ThreadLocal则通过为每个线程提供变量的独立副本,避免了共享资源的并发问题。在实际编程中,开发者需要根据具体...

    ThreadLocal源码分析和使用

    private Map valueMap=Collections.synchronizedMap(new HashMap()); public void set(Object newValue){ valueMap.put(Thread.currentThread(),newValue);//键为线程对象,值为本线程的变量副本 } public ...

    8个案例详解教会你ThreadLocal.docx

    - 在面试中,可能会遇到关于 `ThreadLocal` 生命周期管理、内存泄漏、与 `synchronized` 的比较以及在实际应用中的场景分析等问题。 通过以上介绍,我们可以了解到 `ThreadLocal` 在处理多线程环境中提供了独特的...

    ThreadLocal详解

    与传统的使用`synchronized`关键字或`Lock`接口来实现线程同步不同,ThreadLocal提供了另一种解决线程安全问题的思路——即为每个线程创建独立的变量副本,避免了线程间的变量共享所带来的同步问题,从而提高了程序...

    深入理解 Java 之 ThreadLocal 工作原理1

    当我们创建一个ThreadLocal实例并调用其set方法时,实际上是将值与当前线程关联起来。每个线程都有一个ThreadLocalMap,这是ThreadLocal内部的一个静态内部类,用来存储ThreadLocal对象和它们对应的值。 在...

    2、导致JVM内存泄露的ThreadLocal详解

    `ThreadLocal`类内部维护了一个`ThreadLocalMap`结构,该结构存储了线程与线程局部变量之间的映射关系。每当一个新的线程创建并首次访问某个`ThreadLocal`实例时,都会在该线程的`ThreadLocalMap`中添加一个新的键值...

    彻底理解ThreadLocal 1

    传统的线程同步,如使用`synchronized`关键字,会使得多个线程在访问共享资源时需要排队执行,确保了线程安全,但牺牲了并发性能。而ThreadLocal则通过为每个线程创建单独的变量副本,消除了并发冲突,提升了程序的...

    java面试题

    Java面试中,ThreadLocal和Synchronized是经常被讨论的话题,它们是Java并发编程中的关键概念。ThreadLocal,顾名思义,线程局部变量,它为每个线程提供了一个独立的变量副本,使得每个线程都可以独立地改变自己的...

    入研究java.lang.ThreadLocal类.docx

    - **生命周期**:线程局部变量的生命周期与线程相同,当线程结束时,其对应的变量副本也会被垃圾回收。 - **实现原理**:通过在 ThreadLocal 类中维护一个 Map 结构来存储每个线程的变量副本。 **应用场景**: - 在...

    18 线程作用域内共享变量—深入解析ThreadLocal.pdf

    传统的解决方案包括使用`Atomic`类、`volatile`关键字以及`synchronized`关键字来保证多线程环境下的数据一致性。然而,这些同步机制并不总是最优解,特别是在需要线程内共享变量且避免线程间干扰的情况下。此时,`...

    java ThreadLocal使用案例详解

    Java ThreadLocal使用案例详解 Java ThreadLocal是Java语言中的一种机制,用于为每个线程提供一个独立的变量副本,以解决多线程环境下共享变量的线程安全问题。在本文中,我们将详细介绍Java ThreadLocal的使用案例...

    java 中ThreadLocal 的正确用法

    在 Java 中,ThreadLocal 实例通常是 private static 字段,位于希望与线程关联状态的类中。例如,在 SerialNum 类中,我们定义了一个私有静态的 ThreadLocal 实例(serialNum),用于维护每个线程的序列号。 ```...

Global site tag (gtag.js) - Google Analytics