【多线程核心技术】---synchronized同步代码块

synchronized声明方法的缺点

    A线程调用同步方法执行一个长时间的任务,那么B线程则必须等待较长的时间。

synchronized同步代码块

  当一个线程访问Object的一个synchronized同步代码块时,另一个线程仍然可以访问该Object对象中的非synchronized(this)同步代码块。

一半同步,一半异步

    不在synchronized同步代码块中的就是异步执行,在synchronized同步代码块中的就是同步执行。

synchronized代码块间(多个代码块)的同步性

    当一个线程访问Object的一个synchronized(this)代码块时,其他线程同时对一个Object中的其他synchronized(this)同步代码块的访问将被阻塞。这说明synchronized使用的“对象监视器”是一个。

同步synchronized(this)代码块是锁定当前对象的

    和synchronized方法一样,synchronized(this)代码块是锁定当前对象的

将任意对象作为对象监视器

    多个线程调用同一个对象中的不同名称的 synchronized同步方法或synchronized(this)同步代码块时,调用的效果是按顺序执行,也就是同步的,阻塞的。

    (1)synchronized同步方法

    1)对其他synchronized同步方法或synchronized(this)同步代码块调用呈阻塞状态。

    2)同一时间只有一个线程可以执行synchronized同步方法中的代码。

    (2)synchronized(this)同步代码块    

    1)对其他synchronized同步方法或synchronized(this)同步代码块调用呈阻塞状态。

    2)同一时间只有一个线程可以执行ynchronized(this)同步代码块中的代码。

    Java还支持对“任意对象”作为“对象监视器”来实现同步的功能。这个“任意对象”大多数是实例变量及方法的参数,使用格式为synchronized(非this对象)

   同步代码块synchronized(非this对象)

    1)在多个线程持有“对象监视器”为同一个对象的前提下,同一时间只有一个线程可以执行synchronized(非this对象X)同步代码块中的代码。

    2)当持有“对象监视器”为同一个对象的前提下,同一时间只有一个线程可以执行synchronized(非this对象X)同步代码块中的代码。

    锁非this对象有一定的优点:如果在一个类中有很多synnchronized方法,这时虽然能实现同步,但会受到阻塞,影响运行效率;但如果使用同步代码块锁非this对象,则synchronized(非this)代码块中的程序与同步方法是异步的,不与其他锁this同步方法抢this锁,则可以提高运行效率。

    使用“synchronized(非this对象X)同步代码块”格式进行同步操作时,对象监视器必须是同一个对象(锁)。如果不是同一个对象监视器,运行的结果就是异步调用。

    两个线程操作同一个对象的同一个synchronized(非this对象X)同步代码块;

    当(this对象X   锁的对象)为成员变量时--》同步效果

    当(非this对象X   锁的对象)为局部变量时--》异步效果    

synchronized(非this对象)与同步synchronized方法是异步调用的 

    由于对象监视器不同,运行结果是异步的。

    同步代码块放在非同步synchronized方法中进行声明,并不能保证调用方法的线程的执行同步/顺序性。虽然能在同步块中的执行的顺序是同步的,这极易出现“脏读”问题。

    使用“synchronized(非this对象X)同步代码块”格式可以解决“脏读”问题。

    两个线程访问同一个对象的两个synchronized方法,并操作其中一个synchronized方法,结果同步块中是同步打印的,但线程A和线程B是异步执行的。这就会出现“脏读”。线程的执行方法的顺序不确定,当A和B两个线程执行带有分支判断的方法时,就会出现逻辑上的错误,有可能出现“脏读”。

细化验证:

    “synchronized(非this对象X)”格式的写法是将X对象本身作为”对象监视器”,这就得出以下三个结论:

    1)当多个线程同时执行synchronized(X){ }同步代码块时呈同步效果。

       创建(new)同一个对象--》 同一个”对象监视器”同步,   

       创建(new)两个对象--》 两个对象”对象监视器”异步,

    2)当其他线程执行X对象中synchronized同步方法时呈同步效果。

    3)当其他线程执行X对象方法里面的synchronized(this)代码块时也呈现同步效果。

    注意:如果其他线程调用不加synshronized关键字的方法时,还是异步调用。

静态同步方法synchronized方法与synchronized(class)代码块

    在synchronized修饰的方法上+Static,表示当前文件对应的Class类进行持锁。持同步效果。

   1) synchronized关键字+Static静态方法上表示跟类加锁。

   2) synchronized关键字+非Static静态方法上表示跟对象加锁。

   两线程相互调用Static(class锁)呈同步, 相互调用(class锁---对象锁)呈异步效果。

    class锁可以对类的所有对象起实例作用。 及对所有你所创建当前类(new Xxx)的不同对象加锁    ---》同步

    同步synchronized(class)代码块的作用和synchronized   static的方法一样。

数据类型String的常量池特性

    synchronized(String)同步块与String联合使用时,当String值相等时,就会认为两个线程持有相同的锁,另一个线程不能执行,造成死循环。这就是String常量池的特性。一般改为 new object()实例化一个Object对象。

     String  b=“a”---》String b=new String (“a”)

同步Synchronized方法无限等待与解决

   1) 同步代码块

   2)new Object()

多线程的死锁

    线程都在等待得不到被释放的锁,重而导致所有的线程无法继续执行。在设计时避免双方互相持有对方的锁。

内置类与静态内置类

    内部类的两个同步方法,两个线程调用一个对象(new 内部类),结果呈异步显示,他们使用的是不同的锁。

    同步代码块synchronized(class)对class加锁后,其他线程只能以同步的方式调用class的静态同步方法。

锁对象的改变


    再将任何数据类型作为同步锁时,如否有多个线程同时持有锁对象,如果同时持有相同的锁对象,则这些线程之间就是同步的;如果分别持有锁对象,则这些线程之间是异步的。

    对象不变,即使对象的属性被改变,运行的结果还是同步的(既不会因为对象的属性的改变而改变)。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值