美文网首页
内部锁和同步

内部锁和同步

作者: 原创迷恋者 | 来源:发表于2019-10-14 17:31 被阅读0次

同步是在被称为内部锁或者管锁的内部实体上建立起来的。内部锁在同步的两个方面都扮演了重要的角色:加强对一个对象状态互斥的进入,以及建立对可见性很重要的happens-before关系。

每个对象都有一个与其关联的内部锁。习惯地说,如果一个线程需要对某个对象的域进行互斥、持续的访问,那么该线程需要在访问之前,取得这个对象的内部锁。线程持有锁的时间,是指该线程获得锁,到它释放锁之间的时间。只要一个线程持有一个内部锁,其他线程就无法获取相同的锁。当其他线程试图获取时,它们会被阻塞。

当一个线程释放一个内部锁时,一个happens-before关系就在释放的动作和之后对相同锁的获取动作之间建立。

同步方法中的锁

当一个线程调用一个同步方法时,它自动获取了该对象的内部锁,然后在方法返回时自动释放。即使返回是异常退出,锁仍然会释放。

同步代码块中的锁

另一个创建同步代码的方式是使用同步代码块。不像是同步方法,同步代码块必须指定提供内部锁的对象:

public void addName(String name) {
    synchronized(this) {
        lastName = name;
        nameCount++;
    }
    nameList.add(name);
}

在这个例子中,addName方法需要对lastName和nameCount进行变化同步,但同时也需要避免同步调用该对象的其他方法(在同步代码块中调用其他方法,会导致活锁,这会在之后的章节讲到)。没有同步语句,那只会有一个分离的、非同步方法,只为了调用nameList.add。

同步语句使用更细致的同步颗粒,对提高并发也很有帮助。比如,有一个MsLLunch类,它有两个变量,c1和c2,它们从来不会同时使用。所有对这两个域进行修改的操作都必须是同步的,但c1和c2各自的更新操作却没有理由被对方打扰——如果这样做,会降低并发,因为创造了不必要的阻塞。因此此处不用同步方法,或是与this相关的锁,我们创建两个独立的对象,来关联锁。

public class MsLunch {
    private long c1 = 0;
    private long c2 = 0;
    private Object lock1 = new Object();
    private Object lock2 = new Object();

    public void inc1() {
        synchronized(lock1) {
            c1++;
        }
    }

    public void inc2() {
        synchronized(lock2) {
            c2++;
        }
    }
}

使用这个用法需要非常小心。你需要要完成确定对域交错访问是真的安全。

可重入同步

一个线程无法获取被另一个线程持有的锁,但一个线程可以获取它自己已经拥有的锁。允许一个锁来超过一次、反复获取相同的锁,就被称为允许可重入同步。这描述了以下的情况:同步的代码直接或间接地调用了一个方法,其中也包含着同步的代码,并且这两个代码集使用相同的锁。如果没有可重入同步,那么被同步的代码必须要有更多的预先检查,来防止一个线程自己导致自己阻塞。

相关文章

  • 内部锁和同步

    同步是在被称为内部锁或者管锁的内部实体上建立起来的。内部锁在同步的两个方面都扮演了重要的角色:加强对一个对象状态互...

  • 并发编程反模式

    不连贯的同步性: 为了同步某个对象或者对象本身的某个域的访问,使用同步锁(内部锁或者显式锁,例如,对象本身的内部锁...

  • 从ReentrantLock简单理解线程同步中的一些概念

    1,锁的内部实现AbstractQueuedSynchronizer(AQSync) AQSync是实现同步锁的核...

  • 探究synchronized底层原理(基于JAVA8源码分析)

    JVM支持方法级和方法内部一段指令序列的同步,都用同步锁(monitor)实现 synchronized可以保证方...

  • 源码|并发一枝花之ReentrantLock与AQS(1):lo

    显示锁ReentrantLock的内部同步依赖于AQS(AbstractQueuedSynchronizer),因...

  • 多线程"锁重入"概念

    Java多线程锁重入是指: 在已经获得锁的同步方法或同步代码块内部可以调用锁定对象的其他同步方法, 不需要重新获取...

  • synchronized实现原理

    Java虚拟机支持方法级的同步和方法内部一段指令序列的同步,两种同步结构都使用管程(Monitor,或直接称为锁)...

  • [并发编程]快速解读AQS实现

    前言 如果说Lock是用户和锁的交互接口,那么AQS就是锁的实现。它将所有竞争锁的线程排队,内部维护了同步状态管理...

  • 锁分类

    是否锁住资源分为悲观锁(互斥同步锁)和乐观锁 悲观锁 典型的就是synchronize锁和Lock锁互斥同步锁, ...

  • Condition的实现

    ConditionObject是同步器AQS的内部类,因为Condition的操作需要获取相关联的锁,所以作为同步...

网友评论

      本文标题:内部锁和同步

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