Java中一个类里面有两个用synchronized修饰的非静态方法,不同的线程中的实例访问这两个方法时会发生什么?

首先这个问题涉及到的是Java的对象锁。

java的对象锁和类锁:java的对象锁和类锁在锁的概念上基本上和内置锁是一致的,但是,两个锁实际是有很大的区别的,对象锁是用于对象实例方法,或者一个对象实例上的,类锁是用于类的静态方法或者一个类的class对象上的。我们知道,类的对象实例可以有很多个,但是每个类只有一个class对象,所以不同对象实例的对象锁是互不干扰的,但是每个类只有一个类锁。但是有一点必须注意的是,其实类锁只是一个概念上的东西,并不是真实存在的,它只是用来帮助我们理解锁定实例方法和静态方法的区别的。

  1. public class TestSynchronized   
  2. {    
  3.     public void test1()   
  4.     {    
  5.          synchronized(this)   
  6.          {    
  7.               int i = 5;    
  8.               while( i-- > 0)   
  9.               {    
  10.                    System.out.println(Thread.currentThread().getName() + " : " + i);    
  11.                    try   
  12.                    {    
  13.                         Thread.sleep(500);    
  14.                    }   
  15.                    catch (InterruptedException ie)   
  16.                    {    
  17.                    }    
  18.               }    
  19.          }    
  20.     }    
  21.       
  22.     public synchronized void test2()   
  23.     {    
  24.          int i = 5;    
  25.          while( i-- > 0)   
  26.          {    
  27.               System.out.println(Thread.currentThread().getName() + " : " + i);    
  28.               try   
  29.               {    
  30.                    Thread.sleep(500);    
  31.               }   
  32.               catch (InterruptedException ie)   
  33.               {    
  34.               }    
  35.          }    
  36.     }    
  37.       
  38.     public static void main(String[] args)   
  39.     {    
  40.          final TestSynchronized myt2 = new TestSynchronized();    
  41.          Thread test1 = new Thread(  new Runnable() {  public void run() {  myt2.test1();  }  }, "test1"  );    
  42.          Thread test2 = new Thread(  new Runnable() {  public void run() { myt2.test2();   }  }, "test2"  );    
  43.          test1.start();;    
  44.          test2.start();    
  45. //         TestRunnable tr=new TestRunnable();  
  46. //         Thread test3=new Thread(tr);  
  47. //         test3.start();  
  48.     }   
  49.     
  50. }  
运行结果:

  1. test2 : 4  
  2. test2 : 3  
  3. test2 : 2  
  4. test2 : 1  
  5. test2 : 0  
  6. test1 : 4  
  7. test1 : 3  
  8. test1 : 2  
  9. test1 : 1  
  10. test1 : 0  
上述的代码,第一个方法时用了同步代码块的方式进行同步,传入的对象实例是this,表明是当前对象,当然,如果需要同步其他对象实例,也不可传入其他对象的实例;第二个方法是修饰方法的方式进行同步。因为第一个同步代码块传入的this,所以两个同步代码所需要获得的对象锁都是同一个对象锁,下面main方法时分别开启两个线程,分别调用test1和test2方法,那么两个线程都需要获得该对象锁,另一个线程必须等待。上面也给出了运行的结果可以看到:直到test2线程执行完毕,释放掉锁,test1线程才开始执行。

如果我们把test2方法的synchronized关键字去掉,执行结果会如何呢?

运行结果:

  1. test1 : 4  
  2. test2 : 4  
  3. test2 : 3  
  4. test1 : 3  
  5. test1 : 2  
  6. test2 : 2  
  7. test2 : 1  
  8. test1 : 1  
  9. test2 : 0  
  10. test1 : 0  
上面是执行结果,我们可以看到,结果输出是交替着进行输出的,这是因为,某个线程得到了对象锁,但是另一个线程还是可以访问没有进行同步的方法或者代码。进行了同步的方法(加锁方法)和没有进行同步的方法(普通方法)是互不影响的,一个线程进入了同步方法,得到了对象锁,其他线程还是可以访问那些没有同步的方法(普通方法)。

如果是synchronized修饰的静态方法呢?

Java代码:

  1. public class TestSynchronized   
  2. {    
  3.     public void test1()   
  4.     {    
  5.          synchronized(TestSynchronized.class)   
  6.          {    
  7.               int i = 5;    
  8.               while( i-- > 0)   
  9.               {    
  10.                    System.out.println(Thread.currentThread().getName() + " : " + i);    
  11.                    try   
  12.                    {    
  13.                         Thread.sleep(500);    
  14.                    }   
  15.                    catch (InterruptedException ie)   
  16.                    {    
  17.                    }    
  18.               }    
  19.          }    
  20.     }    
  21.       
  22.     public static synchronized void test2()   
  23.     {    
  24.          int i = 5;    
  25.          while( i-- > 0)   
  26.          {    
  27.               System.out.println(Thread.currentThread().getName() + " : " + i);    
  28.               try   
  29.               {    
  30.                    Thread.sleep(500);    
  31.               }   
  32.               catch (InterruptedException ie)   
  33.               {    
  34.               }    
  35.          }    
  36.     }    
  37.       
  38.     public static void main(String[] args)   
  39.     {    
  40.          final TestSynchronized myt2 = new TestSynchronized();    
  41.          Thread test1 = new Thread(  new Runnable() {  public void run() {  myt2.test1();  }  }, "test1"  );    
  42.          Thread test2 = new Thread(  new Runnable() {  public void run() { TestSynchronized.test2();   }  }, "test2"  );    
  43.          test1.start();    
  44.          test2.start();    
  45. //         TestRunnable tr=new TestRunnable();  
  46. //         Thread test3=new Thread(tr);  
  47. //         test3.start();  
  48.     }   
  49.     
  50. }  
运行结果:

  1. test1 : 4  
  2. test1 : 3  
  3. test1 : 2  
  4. test1 : 1  
  5. test1 : 0  
  6. test2 : 4  
  7. test2 : 3  
  8. test2 : 2  
  9. test2 : 1  
  10. test2 : 0  
其实,类锁修饰方法和代码块的效果和对象锁是一样的,因为类锁只是一个抽象出来的概念,只是为了区别静态方法的特点,因为静态方法是所有对象实例共用的,所以对应着synchronized修饰的静态方法的锁也是唯一的,所以抽象出来个类锁。其实这里的重点在下面这块代码,synchronized同时修饰静态和非静态方法。

Java代码:

  1. public class TestSynchronized   
  2. {    
  3.     public synchronized void test1()   
  4.     {    
  5.               int i = 5;    
  6.               while( i-- > 0)   
  7.               {    
  8.                    System.out.println(Thread.currentThread().getName() + " : " + i);    
  9.                    try   
  10.                    {    
  11.                         Thread.sleep(500);    
  12.                    }   
  13.                    catch (InterruptedException ie)   
  14.                    {    
  15.                    }    
  16.               }    
  17.     }    
  18.       
  19.     public static synchronized void test2()   
  20.     {    
  21.          int i = 5;    
  22.          while( i-- > 0)   
  23.          {    
  24.               System.out.println(Thread.currentThread().getName() + " : " + i);    
  25.               try   
  26.               {    
  27.                    Thread.sleep(500);    
  28.               }   
  29.               catch (InterruptedException ie)   
  30.               {    
  31.               }    
  32.          }    
  33.     }    
  34.       
  35.     public static void main(String[] args)   
  36.     {    
  37.          final TestSynchronized myt2 = new TestSynchronized();    
  38.          Thread test1 = new Thread(  new Runnable() {  public void run() {  myt2.test1();  }  }, "test1"  );    
  39.          Thread test2 = new Thread(  new Runnable() {  public void run() { TestSynchronized.test2();   }  }, "test2"  );    
  40.          test1.start();    
  41.          test2.start();    
  42. //         TestRunnable tr=new TestRunnable();  
  43. //         Thread test3=new Thread(tr);  
  44. //         test3.start();  
  45.     }   
  46.     
  47. }  
运行结果:

  1. test1 : 4  
  2. test2 : 4  
  3. test1 : 3  
  4. test2 : 3  
  5. test2 : 2  
  6. test1 : 2  
  7. test2 : 1  
  8. test1 : 1  
  9. test1 : 0  
  10. test2 : 0  
上面代码synchronized同时修饰静态方法和实例方法,但是运行结果是交替进行的,这证明了类锁和对象锁是两个不一样的锁,控制着不同的区域,它们是互不干扰的。同样,线程获得对象锁的同时,也可以获得该类锁,即同时获得两个锁,这是允许的。

到这里,对synchronized的用法已经有了一定的了解。

        打个比方:一个object就像一个大房子,大门永远打开。房子里有 很多房间(也就是方法)。这些房间有上锁的(synchronized方法), 和不上锁之分(普通方法)。房门口放着一把钥匙(key),这把钥匙可以打开所有上锁的房间。另外我把所有想调用该对象方法的线程比喻成想进入这房子某个 房间的人。所有的东西就这么多了,下面我们看看这些东西之间如何作用的。在此我们先来明确一下我们的前提条件。该对象至少有一synchronized方法,否则这个key还有啥意义。当然也就不会有我们的这个主题了。一个人想进入某间上了锁的房间,他来到房子门口,看见钥匙在那儿(说明暂时还没有其他人要使用上锁的 房间)。于是他走上去拿到了钥匙,并且按照自己 的计划使用那些房间。注意一点,他每次使用完一次上锁的房间后会马上把钥匙还回去。即使他要连续使用两间上锁的房间,中间他也要把钥匙还回去,再取回来。因此,普通情况下钥匙的使用原则是:“随用随借,用完即还。”这时其他人可以不受限制的使用那些不上锁的房间,一个人用一间可以,两个人用一间也可以,没限制。但是如果当某个人想要进入上锁的房间,他就要跑到大门口去看看了。有钥匙当然拿了就走,没有的话,就只能等了。要是很多人在等这把钥匙,等钥匙还回来以后,谁会优先得到钥匙?Not guaranteed。象前面例子里那个想连续使用两个上锁房间的家伙,他中间还钥匙的时候如果还有其他人在等钥匙,那么没有任何保证这家伙能再次拿到。 (JAVA规范在很多地方都明确说明不保证,像Thread.sleep()休息后多久会返回运行,相同优先权的线程那个首先被执行,当要访问对象的锁被 释放后处于等待池的多个线程哪个会优先得到,等等。我想最终的决定权是在JVM,之所以不保证,就是因为JVM在做出上述决定的时候,绝不是简简单单根据 一个条件来做出判断,而是根据很多条。而由于判断条件太多,如果说出来可能会影响JAVA的推广,也可能是因为知识产权保护的原因吧。SUN给了个不保证 就混过去了。无可厚非。但我相信这些不确定,并非完全不确定。因为计算机这东西本身就是按指令运行的。即使看起来很随机的现象,其实都是有规律可寻。学过 计算机的都知道,计算机里随机数的学名是伪随机数,是人运用一定的方法写出来的,看上去随机罢了。另外,或许是因为要想弄的确太费事,也没多大意义,所 以不确定就不确定了吧。)

再来看看同步代码块。和同步方法有小小的不同。

1.从尺寸上讲,同步代码块比同步方法小。你可以把同步代码块看成是没上锁房间里的一块用带锁的屏风隔开的空间。

2.同步代码块还可以人为的指定获得某个其它对象的key。就像是指定用哪一把钥匙才能开这个屏风的锁,你可以用本房的钥匙;你也可以指定用另一个房子的钥匙才能开,这样的话,你要跑到另一栋房子那儿把那个钥匙拿来,并用那个房子的钥匙来打开这个房子的带锁的屏风。

         记住你获得的那另一栋房子的钥匙,并不影响其他人进入那栋房子没有锁的房间。

         为什么要使用同步代码块呢?我想应该是这样的:首先对程序来讲同步的部分很影响运行效率,而一个方法通常是先创建一些局部变量,再对这些变量做一些 操作,如运算,显示等等;而同步所覆盖的代码越多,对效率的影响就越严重。因此我们通常尽量缩小其影响范围。

如何做?同步代码块。我们只把一个方法中该同 步的地方同步,比如运算。

         另外,同步代码块可以指定钥匙这一特点有个额外的好处,是可以在一定时期内霸占某个对象的key。还记得前面说过普通情况下钥匙的使用原则吗。现在不是普通情况了。你所取得的那把钥匙不是永远不还,而是在退出同步代码块时才还。

          还用前面那个想连续用两个上锁房间的家伙打比方。怎样才能在用完一间以后,继续使用另一间呢。用同步代码块吧。先创建另外一个线程,做一个同步代码 块,把那个代码块的锁指向这个房子的钥匙。然后启动那个线程。只要你能在进入那个代码块时抓到这房子的钥匙,你就可以一直保留到退出那个代码块。也就是说 你甚至可以对本房内所有上锁的房间遍历,甚至再sleep(10*60*1000),而房门口却还有1000个线程在等这把钥匙呢。很过瘾吧。


  • 1
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值