AQS源码分析

 

 

 

ReentrantLock lock = new ReentrantLock();
        lock.lock();
        lock.unlock();

lock();如果获取成功,则设置独占线程。获取结束。

final void lock() {
            //通过cas获取锁成功
            if (compareAndSetState(0, 1))
                setExclusiveOwnerThread(Thread.currentThread());
            else
            //获取失败
                acquire(1);
        }

acquire(); 如果获取失败,则再次尝试获取。

public final void acquire(int arg) {
        if (!tryAcquire(arg) && //再次尝试获取锁
            acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
            //如果获取锁失败,并且加入阻塞队列成功。线程挂起。
            selfInterrupt();
    }

 

tryAcquire()--->nonfairTryAcquire(); 如果再次获取成功,则结束。

final boolean nonfairTryAcquire(int acquires) {
            final Thread current = Thread.currentThread();
            //获取锁的状态
            int c = getState();
            //如果==0,说明没有线程持有锁,将再次尝试获取锁
            if (c == 0) {
                //再次尝试
                if (compareAndSetState(0, acquires)) {
                //获取成功,设置独有线程
                    setExclusiveOwnerThread(current);
                    return true;
                }
            }
            //如果是重入锁,设置state.
            else if (current == getExclusiveOwnerThread()) {
                int nextc = c + acquires;
                if (nextc < 0) // overflow
                    throw new Error("Maximum lock count exceeded");
                setState(nextc);
                return true;
            }
            //尝试获取锁失败。
            return false;
        }

 

addWaiter(); 再次获取失败,则加入等待队列。并挂起。

private Node addWaiter(Node mode) {
        Node node = new Node(Thread.currentThread(), mode);
        // Try the fast path of enq; backup to full enq on failure
        Node pred = tail;
        //第一次,tail为 null 跳过
        //第二次,tail不为null,将node加入到链表中。(阻塞队列)
        if (pred != null) {
            //node的上一个指向tail
            node.prev = pred;
            if (compareAndSetTail(pred, node)) {
                //tail的下一个指向node。
                pred.next = node;
                return node;
            }
        }
        //初始化head和tail,并将node节点加入到链表(阻塞队列)中。
        enq(node);
        return node;
    }


private Node enq(final Node node) {
        for (;;) {
            Node t = tail;
            //自旋,第一次t为null
            if (t == null) { // Must initialize
                //设置head节点
                if (compareAndSetHead(new Node()))
                    //head和tail为同一节点。
                    tail = head;
            } else {
                //第二次。node的上一个指向tail。
                node.prev = t;
                //将node设置为tail
                if (compareAndSetTail(t, node)) {
                    //t的next指向node。即node变成了尾节点。
                    t.next = node;
                    return t;
                }
            }
        }
    }

 

acquireQueued(Node node);返回的是tail节点,处理tail节点。

final boolean acquireQueued(final Node node, int arg) {
        boolean failed = true;
        try {
            boolean interrupted = false;
            for (;;) {
                //倒叙遍历阻塞队列(链表)
                final Node p = node.predecessor();
                //如果node的上一个节点就是head节点,则再次尝试获取锁。
                if (p == head && tryAcquire(arg)) {
                    //如果获取成功,则将head设置为node
                    setHead(node);
                    //下一个指向null。即tail为null。
                    p.next = null; // help GC
                    failed = false;
                    return interrupted;
                }
                //获取失败,则挂起在这里。
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    interrupted = true;
            }
        } finally {
            if (failed)
                cancelAcquire(node);
        }
    }

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java中的AQS(AbstractQueuedSynchronizer)是实现锁和同步器的一种重要工具。在AQS中,一个节点表示一个线程,依次排列在一个双向队列中,同时使用CAS原子操作来保证线程安全。当多个线程对于同一资竞争时,一个节点会被放置在队列的尾部,其他线程则在其之前等待,直到该资可以被锁定。 当一个线程调用lock()方法进行锁定时,它会首先调用tryAcquire()方法尝试获取锁。如果当前资尚未被锁定,则该线程成功获取锁,tryAcquire()返回true。如果当前资已被锁定,则线程无法获取锁,tryAcquire()返回false。此时该线程就会被加入到等待队列中,同时被加入到前一个节点的后置节点中,即成为它的后继。然后该线程会在park()方法处等待,直到前一个节点释放了锁,再重新尝试获取锁。 在AQS中,当一个节点即将释放锁时,它会调用tryRelease()方法来释放锁,并唤醒后置节点以重试获取锁。如果当前节点没有后置节点,则不会发生任何操作。当一个线程在队列头部成功获取锁和资时,该线程需要使用release()方法释放锁和资,并唤醒等待队列中的后置节点。 总之,AQS中的锁机制是通过双向等待队列实现的,其中节点表示线程,使用CAS原子操作保证线程安全,并在tryAcquire()和tryRelease()方法中进行锁定和释放。该机制保证了多线程环境下资的正确访问和线程的安全执行。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值