一、synchronized同步语句块:
在介绍同步语句块之前,先做一个实验,验证多个线程调用同一个同步方法的顺序是随机的。
例子一:
1)MyList.java
2)测试:
可见,在同步方法中的代码是同步打印的,但线程A和线程B是异步执行的,这可能会出现脏读的情况。看下面的例子:
例子二:创建一个只能放一个元素的集合工具类,在多线程的情况下,出现“脏读”,集合不止有一个元素。
1) 在上面例子的基础上,创建业务类MyListService.java:
2)测试与结果:
分析:首先list初始为空,业务类MyListService中的addData()方法原意是要保证list只能放一个元素的,但这里出现“脏读”。第一个线程判断 ( list.getSize() < 1 )为真时,就进入休眠2s的状态,此时第二线程判断
( list.getSize()<1 )也为真,于是也休眠2s。这样两个线程休眠完后,都会添加数据的。我们需要把if{...}里面的操作作为整体,这样才不会出现脏读。这时可以使用同步语句块来解决问题:
3)用"synchronized(非this对象)同步代码块"来解决“脏读”问题:
重复上述的测试,结果如下:
可见,只有一个线程执行add()方法,实现了线程之间同步。
小结:
1、无论是synchronized()方法还是synchronized(对象){}代码块,其作用无非就是把一系列语句(花括号括起来的语句)作为一个整体操作,同一时间内只有一个线程能执行这个整体操作。
2、synchronized关键字与对象锁有关,执行同步方法或者同步语句块,必须先获得相应的对象锁;如果相应的对象锁
已经被某个线程获得了,其它线程要执行同步方法或同步代码块的话,必须等相应的对象锁被释放。
二、静态同步synchronized方法与synchronized(类名.class)语句块
1、静态同步synchronized方法
关键字synchronized加在static静态方法上,是给当前*.java文件对应的Class类上锁;而加在非static静态方法上,是给对象上锁。
1)类文件Service.java代码如下:
2)三个线程类ThreadA.java、ThreadB.java、ThreadC.java
3)测试一的代码与结果:
以上例子中,两个线程类访问同一个Service实例,但出现了“异步”执行的结果,原因是持有不同的锁,一个是Class类锁,另一个是对象锁。
4)测试二的代码与结果:
上述测试中,两个线程访问的是同一个类的不同实例,但是调用的都是类的静态同步方法,此时Class锁起作用,两个线程是同步的。
2、synchronized(类名.class)语句块
同步synchronized(类名.calss)代码块的作用与synchronized static 方法的作用是一样的,把synchronized static改为synchronized(Srevice.class)做相关的实验,会得到一样的结果,这里不再演示。