首页 > 编程语言 >Java 多线程同步问题的探究(二、给我一把锁,我能创造一个规矩)

Java 多线程同步问题的探究(二、给我一把锁,我能创造一个规矩)

时间:2023-06-15 12:03:59浏览次数:44  
标签:同步 Java String 对象 lock 我能 threadNo 线程 多线程

在上一篇中,我们讲到了多线程是如何处理共享资源的,以及保证他们对资源进行互斥访问所依赖的重要机制:对象锁。




本篇中,我们来看一看传统的同步实现方式以及这背后的原理。



很多人都知道,在Java多线程编程中,有一个重要的关键字,synchronized。但是很多人看到这个东西会感到困惑:“都说同步机制是通过对象锁 来实现的,但是这么一个关键字,我也看不出来Java程序锁住了哪个对象阿?“



没错,我一开始也是对这个问题感到困惑和不解。不过还好,我们有下面的这个例程:

1  public   class  ThreadTest  extends  Thread {

 2 

 3       private   int  threadNo;

 4 

 5       public  ThreadTest( int  threadNo) {

 6           this .threadNo  =  threadNo;

 7      }

 8 

 9       public   static   void  main(String[] args)  throws  Exception {

10           for  ( int  i  =   1 ; i  <   10 ; i ++ ) {

11               new  ThreadTest(i).start();

12              Thread.sleep( 1 );

13          }

14      }

15 

16      @Override

17       public   synchronized   void  run() {

18           for  ( int  i  =   1 ; i  <   10000 ; i ++ ) {

19              System.out.println( " No. "   +  threadNo  +   " : "   +  i);

20          }

21      }

22  }


这个程序其实就是让10个线程在控制台上数数,从1数到9999。理想情况下,我们希望看到一个线程数完,然后才是另一个线程开始数数。但是这个程序的执 行过程告诉我们,这些线程还是乱糟糟的在那里抢着报数,丝毫没有任何规矩可言。





但是细心的读者注意到:run方法还是加了一个synchronized关键字的,按道理说,这些线程应该可以一个接一个的执行这个run方法才对阿。





但是通过上一篇中,

我们提到的,对于一个成员方法加synchronized关键字,这实际上是以这个成员方法所在的对象本身作为对象锁。在本例中,就是 以ThreadTest类的一个具体对象,也就是该线程自身作为对象锁的。一共十个线程,每个线程持有自己 线程对象的那个对象锁。这必然不能产生同步的效果。换句话说, 如果要对这些线程进行同步,那么这些线程所持有的对象锁应当是共享且唯一 的!


我们来看下面的例程:

 1  public   class  ThreadTest2  extends  Thread {

 2 

 3       private   int  threadNo;

 4       private  String lock;

 5 

 6       public  ThreadTest2( int  threadNo, String lock) {

 7           this .threadNo  =  threadNo;

 8           this .lock  =  lock;

 9      }

10 

11       public   static   void  main(String[] args)  throws  Exception {

12          String lock  =   new  String( " lock " );

13           for  ( int  i  =   1 ; i  <   10 ; i ++ ) {

14               new  ThreadTest2(i, lock).start();

15              Thread.sleep( 1 );

16          }

17      }

18 

19       public   void  run() {

20           synchronized  (lock) {

21               for  ( int  i  =   1 ; i  <   10000 ; i ++ ) {

22                  System.out.println( " No. "   +  threadNo  +   " : "   +  i);

23              }

24          }

25      }

26  }



我们注意到,该程序通过在main方法启动10个线程之前,创建了一个String类型的对象。并通过ThreadTest2的构造函数,将这个对象赋值 给每一个ThreadTest2线程对象中的私有变量lock。根据Java方法的传值特点,

我们知道,这些线程的lock变量实际上指向的是堆内存中的 同一个区域,即存放main函数中的lock变量的区域。

程序将原来run方法前的synchronized关键字去掉,换用了run方法中的一个synchronized块来实现。这个同步块的对象锁,就是 main方法中创建的那个String对象。换句话说,他们指向的是同一个String类型的对象,对象锁是共享且唯一的!


于是,我们看到了预期的效果:10个线程不再是争先恐后的报数了,而是一个接一个的报数。


再来看下面的例程:

 1  public   class  ThreadTest3  extends  Thread {

 2 

 3       private   int  threadNo;

 4       private  String lock;

 5 

 6       public  ThreadTest3( int  threadNo) {

 7           this .threadNo  =  threadNo;

 8      }

 9 

10       public   static   void  main(String[] args)  throws  Exception {

11           // String lock = new String("lock");

12           for  ( int  i  =   1 ; i  <   20 ; i ++ ) {

13               new  ThreadTest3(i).start();

14              Thread.sleep( 1 );

15          }

16      }

17 

18       public   static   synchronized   void  abc( int  threadNo) {

19           for  ( int  i  =   1 ; i  <   10000 ; i ++ ) {

20             

21                  System.out.println( " No. "   +  threadNo  +   " : "   +  i);

22                 

23             

24                 

25             

26          }

27      }

28 

29       public   void  run() {

30          abc(threadNo);

31      }

32  }



细心的读者发现了:这段代码没有使用main方法中创建的String对象作为这10个线程的线程锁。而是通过在run方法中调用本线程中一个静态的同步 方法abc而实现了线程的同步。我想看到这里,你们应该很困惑:这里synchronized静态方法是用什么来做对象锁的呢?



我们知道,对于同步静态方法,对象锁就是该静态放发所在的类的Class实例,由于在JVM中,所有被加载的类都有唯一的类对象,具体到本例,就是唯一的 ThreadTest3.class对象。不管我们创建了该类的多少实例,但是它的类实例仍然是一个!


这样我们就知道了:



1、对于同步的方法或者代码块来说,必须获得对象锁才能够进入同步方法或者代码块进行操作;


2、如果采用method级别的同步,则对象锁即为method所在的对象,如果是静态方法,对象锁即指method所在的


Class对象(唯一);

3、对于代码块,对象锁即指synchronized(abc)中的abc;




4、因为第一种情况,对象锁即为每一个线程对象,因此有多个,所以同步失效,第二种共用同一个对象锁lock,因此同步生效,第三个因为是


static因此对象锁为ThreadTest3的class 对象,因此同步生效。


如上述正确,则同步有两种方式,同步块和同步方法(为什么没有wait和notify?这个我会在补充章节中做出阐述)



如果是同步代码块,则对象锁需要编程人员自己指定,一般有些代码为synchronized(this)只有在单态模式才生效;


(本类的实例有且只有一个)



如果是同步方法,

则分静态和非静态两种



静态方法则一定会同步,非静态方法需在单例模式才生效,推荐用静态方法(不用担心是否单例)。


所以说,在Java多线程编程中,最常见的synchronized关键字实际上是依靠对象锁的机制来实现线程同步的。


我们似乎可以听到synchronized在向我们说:“给我

一把锁,我能创造一个规矩”。

下一篇中,我们将看到JDK 5提供的新的同步机制,也就是大名鼎鼎的Doug Lee提供的Java Concurrency框架。

标签:同步,Java,String,对象,lock,我能,threadNo,线程,多线程
From: https://blog.51cto.com/u_16065168/6486141

相关文章

  • Java中的WeakHashMap与类示例
    在本文中,我们将WeakHashMap 通过示例从java.util包中学习  类。我们将学到什么?WeakHashMap 课程概述WeakHashMap 类构造方法摘要WeakHashMap 类构造方法WeakHashMap 类示例1.WeakHashMap类概述WeakHashMap 是一个基于Hash表的Map接口实现的弱键。当其密钥不再正常使用......
  • Java正则表达式详解
    如果你曾经用过Perl或任何其他内建正则表达式支持的语言,你一定知道用正则表达式处理文本和匹配模式是多么简单。如果你不熟悉这个术语,那么“正则表达式”(RegularExpression)就是一个字符构成的串,它定义了一个用来搜索匹配字符串的模式。许多语言,包括Perl、PHP、Python、JavaScript......
  • JavaScript内存限制
    JavaScriptmemorylimitJavaScript应用程序可以存储的最大数据量是多少?我猜这是由浏览器处理的,每个浏览器都有其局限性吗?如果没有限制,将创建页面文件吗?如果是这样,那不安全吗? 相关讨论  有一些限制,尽管这些取决于浏览器。例如,Firefox对堆栈空间以及过多的CPU消......
  • Java_Dom4j_解析xml
    via:http://blog.163.com/kewangwu@126/blog/static/8672847120126261033594/ 1、DOM4J简介DOM4J是dom4j.org出品的一个开源XML解析包。DOM4J应用于Java平台,采用了Java集合框架并完全支持DOM,SAX和JAXP。Dom:把整个文档作为一个对象。DOM4J最大的特色是使用大量的接口......
  • ajax + java 实现类似网易邮箱邮件地址自动完成功能
    ajax+java实现类似网易邮箱邮件地址自动完成功能2008-04-0218:30********************************************************************源代码下载链接:http://www.javaeye.com/topic/150778***************************************************************......
  • 您必须知道的重要Java关键字
    什么是Java中的关键字?Java关键字是一个保留字,具有与之关联的特殊含义。为便于识别,它们通常以Java格式突出显示。在50个关键字中,有48个正在使用,而有两个不在。让我们更详细地研究一些重要的Java关键字。重要的Java关键字列表摘要: 它用于完成 抽象。它是一种与类和方法相关的非访......
  • java服务器更换jdk版本后报错:javax.net.ssl.SSLHandshakeException: No appropriate p
    java,服务器更换jdk版本后报错:Causedby:javax.net.ssl.SSLHandshakeException:Noappropriateprotocol(protocolisdisabledorciphersuitesareinappropriate)然后数据库出现:###Errorqueryingdatabase.Cause:java.lang.reflect.UndeclaredThrowableExc......
  • javascript现代编程系列教程之七——字符集(七)
    Unicode:Unicode是一个字符集(Charset),包含了世界上所有的字符。每个字符在Unicode中都有其唯一对应的数字编号,这就是我们常说的Unicode码。UTF-8:UTF-8是Unicode的实现方式之一。UTF-8使用一至四个字节为每个字符编码,英文字符通常使用一个字节,西欧其他语言的部分字符使用......
  • 用JavaScript绘制树状图(具有分支合并功能)的一种方法(其一)
    需求分析在很多模拟经营游戏中,科技树是一项重要的内容,其为玩家提供了各项技术与其前后置科技间的拓扑关系。这些科技树在表现形式上和普通树状图很相似,但由于其频繁的分支合并,为科技树的绘制带来困难。因此,我们需要一种简单的方法来绘制科技树。比如,当用户输入:<!--为了降低用户......
  • java file I/O流
    一.File的简介:(java.io包) 生活中的文件: (1)文件的作用:持久化(瞬时状态的对立面状态)(1)文件的定义:一堆数据的集合(2)文件存储的位置:磁盘,硬盘,软盘,U盘等等 计算机中的文件File (1)file的定义:java.io中的File类 (2)创建File:newFile();创建文件实例 (3)File属性:文件的位置,文件的名称......