Java 多线程同步问题的探究(二、给我一把锁,我能创造一个规矩)_第1页
Java 多线程同步问题的探究(二、给我一把锁,我能创造一个规矩)_第2页
Java 多线程同步问题的探究(二、给我一把锁,我能创造一个规矩)_第3页
Java 多线程同步问题的探究(二、给我一把锁,我能创造一个规矩)_第4页
Java 多线程同步问题的探究(二、给我一把锁,我能创造一个规矩)_第5页
已阅读5页,还剩2页未读 继续免费阅读

付费下载

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、在上一篇中,我们讲到了多线程是如何处理共享资源的,以及保证他们对资源进行互斥访问所依赖的重要机制:对象锁。本篇中,我们来看一看传统的同步实现方式以及这背后的原理。很多人都知道,在Java多线程编程中,有一个重要的关键字,synchronized。但是很多人看到这个东西会感到困惑:“都说同步机制是通过对象锁来实现的,但是这么一个关键字,我也看不出来Java程序锁住了哪个对象阿?“没错,我一开始也是对这个问题感到困惑和不解。不过还好,我们有下面的这个例程: 1 public class ThreadTest extends Thread 2 3 private int threadNo; 4 5

2、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 1516 Override17 public synchronized void run() 18 for (int i = 1; i < 10000; i+) 19 Syste

3、m.out.println("No." + threadNo + ":" + i);20 21 22 这个程序其实就是让10个线程在控制台上数数,从1数到9999。理想情况下,我们希望看到一个线程数完,然后才是另一个线程开始数数。但是这个程序的执行过程告诉我们,这些线程还是乱糟糟的在那里抢着报数,丝毫没有任何规矩可言。但是细心的读者注意到:run方法还是加了一个synchronized关键字的,按道理说,这些线程应该可以一个接一个的执行这个run方法才对阿。但是通过上一篇中,我们提到的,对于一个成员方法加synchronized关键字,这实际上是以这个成

4、员方法所在的对象本身作为对象锁。在本例中,就是以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.t

5、hreadNo = threadNo; 8 this.lock = lock; 9 1011 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 1819 public void run() 20 synchronized (lock) 21 for (int

6、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方法前的synchroni

7、zed关键字去掉,换用了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.th

8、readNo = threadNo; 8 910 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 1718 public static synchronized void abc(int threadNo) 19 for (int i = 1; i < 1000

9、0; i+) 20 21 System.out.println("No." + threadNo + ":" + i);22 23 24 25 26 27 2829 public void run() 30 abc(threadNo);31 32 细心的读者发现了:这段代码没有使用main方法中创建的String对象作为这10个线程的线程锁。而是通过在run方法中调用本线程中一个静态的同步方法abc而实现了线程的同步。我想看到这里,你们应该很困惑:这里synchronized静态方法是用什么来做对象锁的呢? 我们知道,对于同步静态方法,对象锁就是该静态放

10、发所在的类的Class实例,由于在JVM中,所有被加载的类都有唯一的类对象,具体到本例,就是唯一的 ThreadTest3.class对象。不管我们创建了该类的多少实例,但是它的类实例仍然是一个! 这样我们就知道了:1、对于同步的方法或者代码块来说,必须获得对象锁才能够进入同步方法或者代码块进行操作; 2、如果采用method级别的同步,则对象锁即为method所在的对象,如果是静态方法,对象锁即指method所在的Class对象(唯一); 3、对于代码块,对象锁即指synchronized(abc)中的abc;4、因为第一种情况,对象锁即为每一个线程对象,因此有多个,所以同步失效,第二种共用同一个对象锁lock,因此同步生效,第三个因为是static因此对象锁为ThreadTest3的class 对象,因此同步生效。 如上述正确,则同步有两种方式,同步块和同步方法(为什么没有wait和notify?这个我会在补充章节中做出阐述)如果是同步代码块,则对象锁需要编程人员自己指定,一般有些代码为synchronized(this)只有在单态模式才生效;(本类的实例有且只有一个)如果是同步方法,则分静态和非静态两种。静态方法则一定会同步,非静态方法需在单例模式才生效,推荐用静态方法(不用

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

最新文档

评论

0/150

提交评论