美文网首页
多线程中的synchronized锁机制

多线程中的synchronized锁机制

作者: superpf | 来源:发表于2019-05-24 10:38 被阅读0次

1、synchronized关键字的机制是对象锁(把这个对象锁住【这个对象的带synchronized关键字的方法锁住】)

如   class A

{

    public synchronized void a()

    {

    }

}

在方法a()上加上synchronized关键字,在多个线程中无法同时调用A类的某个对象的a()方法。

关键字synchronized取得的锁都是对象锁,而不是把一段代码或方法(函数)当作锁,哪个线程先执行带synchronized关键字的方法,哪个线程就持有该方法所属对象的锁,其他线程都只能呈等待状态。但是这有个前提:既然锁叫做对象锁,那么势必和对象相关,所以多个线程访问的必须是同一个对象

如果多个线程访问的是多个对象,那么Java虚拟机就会创建多个锁,既然两个线程持有的是不同的锁,自然不会受到"等待释放锁"这一行为的制约,可以分别运行方法中的代码。

在一个类中的两个方法一个加了synchronized,一个未加synchronized,两个线程是可以同时分别执行两个方法的,但是两个方法都加synchronized,则必须顺序执行

两个结论:

1、A线程持有Object对象的Lock锁,B线程可以以异步方式调用Object对象中的非synchronized类型的方法

2、A线程持有Object对象的Lock锁,B线程如果在这时调用Object对象中的synchronized类型的方法则需要等待,也就是同步

换言之,一个线程调用一个带了synchronized关键字的方法,则这个线程就获得了该对象的锁,其他线程想执行该对象的的所有带synchronized的方法,都得等这个线程执行完才可以去竞争。但是可以直接执行非同步的方法。

2、synchronized锁重入

关键字synchronized拥有锁重入的功能。所谓锁重入的意思就是:当一个线程得到一个对象锁后,再次请求此对象锁时时可以再次得到该对象的锁的

在一个有synchronized关键的方法中调用另一个有synchronized关键的方法,该线程将重新获取该对象的锁

3、异常自动释放锁

当一个线程执行的代码出现异常时,其所持有的锁会自动释放

4、synchronized同步代码块

当执行一个同步方法时间较长的时候,其他等待线程将会长时间等待,这个情况可以将方法中需要同步的部分设置为synchronized同步代码块,这部分代码就只有一个县城能同时执行,而其余代码可以多个线程同时执行。

synchronized(this)

        {

。。。。。。。。。。。。。。

        }

结论:

1、当A线程访问对象的synchronized代码块的时候,B线程依然可以访问对象方法中其余非synchronized块的部分

2、当A线程进入对象的synchronized代码块的时候,B线程如果要访问这段synchronized块,那么访问将会被阻塞

5、两个synchronized块之间具有互斥性

这个与1比较相似,同一个对象的两个方法中都有synchronized块,则两个线程不能同时分别执行这两个带有synchronized块的方法,必须顺序执行。

结论:synchronized块获得的是一个对象锁,换句话说,synchronized块锁定的是整个对象

6、synchronized块和synchronized方法

这两个也是互斥的,执行一个,另一个不能执行,必须顺序执行,总之带synchronized关键字就是锁住对象。

总结一下前面的内容:

1、synchronized同步方法

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

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

2、synchronized同步代码块

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

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

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

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

多个线程持有"对象监视器"为同一个对象的前提下,同一时间只能有一个线程可以执行synchronized(非this对象x)代码块中的代码。(在一个类中,每个synchronized(非this对象x)中的x指示的是同一个变量或参数的时候,它的作用与synchronized(this)相同)

(1)当一个对象的两个方法的synchronized(x)中的x不同时,他们是不互斥的,两个线程可以同时调用这两个方法(synchronized(this)与同步方法效果一致,一个同步方法和一个x不为this的方法块也可以同时执行)。

(2)当多个线程同时执行synchronized(x){}同步代码块时呈同步效果(只有一个线程能执行)

(3)当其他线程执行x对象中的synchronized同步方法时呈同步效果(比如:有一个同步代码块synchronized(A),而A对象中存在是同步方法(同步代码块,必须是synchronized(this),必须是this,这样才能表示这个对象被锁定),并且有其他线程正在执行此方法,也就是A对象被锁定,这时synchronized(A)这个代码块也无法执行

(4)当其他线程执行x对象方法中的synchronized(this)代码块时也呈同步效果

其他线程对一个对象的被锁(X)锁住(其他线程正在执行这段代码,这个对象的这段代码就被锁住了)的代码是无法执行的。

8、同步静态方法

拥有 static 与 synchronized两个关键字的方法

1、如果线程A调用了某个类的静态同步方法(直接通过类名调用或通过对象调用),则该类被上锁(类锁,和对象锁不同),则该类中的静态同步方法都无法被调用(直接通过类名调用或通过对象都无法调用),但是非静态同步方法可以被调用。

例子:假如一个类中有一个静态同步方法A,new出了两个类的实例B和实例C,线程D持有实例B,线程E持有实例C,只要线程D调用了A方法,那么线程E调用A方法必须等待线程D执行完A方法,尽管两个线程持有的是不同的对象。

9、volatile关键字

java有一块主内存,不同线程有不同的工作内存,所有变量在主内存中有一份,当线程需要用到时就去主内存中取,用完后,把新值返还给主内存。

在启动一个线程A后,这个A线程将取走它需要的变量a,如果这个变量a被其他的线程B的更改,线程A是无法感知的,也就是线程A每次使用这个变量a时不会再从主内存中获取,就无法保证它取到的是最新的值。

解决办法:将这个变量前加上volatile关键字。

10、原子类也无法保证线程安全

线程安全不是绝对的,在有逻辑的情况下输出结果也具有随机性

结果安全,顺序会混乱。

相关文章

网友评论

      本文标题:多线程中的synchronized锁机制

      本文链接:https://www.haomeiwen.com/subject/mdvwzqtx.html