手机企业网站源码深圳龙岗有什么好玩的地方
news/
2025/9/24 0:41:43/
文章来源:
手机企业网站源码,深圳龙岗有什么好玩的地方,江门加盟网站建设,uiapp界面设计模板Condition 源码解析 文章目录 Condition 源码解析一、Condition二、Condition 源码解读2.1. lock.newCondition() 获取 Condition 对象2.2. condition.await() 阻塞过程2.3. condition.signal() 唤醒过程2.4. condition.await() 被唤醒后 三、总结 一、Condition
在并发情况下…Condition 源码解析 文章目录 Condition 源码解析一、Condition二、Condition 源码解读2.1. lock.newCondition() 获取 Condition 对象2.2. condition.await() 阻塞过程2.3. condition.signal() 唤醒过程2.4. condition.await() 被唤醒后 三、总结 一、Condition
在并发情况下进行线程间的协调如果是使用的 synchronized 锁我们可以使用 wait()/notify() 进行唤醒如果是使用的 Lock 锁的方式则可以使用 Condition 进行针对性的阻塞和唤醒相较于 wait()/notify() 使用起来更灵活。那么 Condition 是如何实现线程的等待和唤醒的呢本文通过解析Condition 的源码进行理解。
在进行源码分析前先通过一个案例看下 Condition 是如何使用的
public class Test {public synchronized static void main(String[] args) throws InterruptedException {Lock lock new ReentrantLock();Condition condition lock.newCondition();new Thread(() - {lock.lock();System.out.println(线程1开始等待);try {condition.await();} catch (InterruptedException e) {e.printStackTrace();}System.out.println(线程1被唤醒继续执行结束);lock.unlock();}, 1).start();new Thread(() - {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}lock.lock();System.out.println(开始唤醒线程);condition.signal();try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(线程2执行结束);lock.unlock();}, 2).start();}
}运行之后可以看到下面日志
由于在第一个线程中使用的 condition.await() 因此当前线程会被阻塞挂起而第二个线程在 1s 后进行了 condition.signal() 操作因此第一个线程会被唤醒继续执行。这里可以发现第一个线程阻塞时锁并没有释放而第二个线程在1s后也成功拿到锁了所以表明在 condition.await() 时会自动释放当前锁这点和 wait() 相同在第二个线程进行了 condition.signal() 操作第一个线程并没有继续向下执行而是等待第二个线程处理完才会继续执行由此可以表明被唤醒的线程会重新获取锁成功获取锁后继续执行。
下面通过源码看下 Condition 是如何实现的等待唤醒。
二、Condition 源码解读
2.1. lock.newCondition() 获取 Condition 对象
首先看下在使用 lock.newCondition() 获取一个Condition 对象时具体做了什么这里以 ReentrantLock 为例进入到 ReentrantLock 的 newCondition() 方法中又执行了 Sync 的 newCondition() 方法再进去就会发现其实是 new 了一个 ConditionObject 类对象
下面点到这个类中可以看到其实是 AQS 下的一个子类
2.2. condition.await() 阻塞过程
了解到 Condition 的对象后可以看到是 AQS 下的一个子类那下面其他的方法也肯定依赖于 AQS 下面看下 condition.await() 方法点到 await() 方法中 其中 addConditionWaiter() 则是将自己加入到 AQS的队列中并获取到当前线程所在的 Node 这里注意下 Node 的状态是 Node.CONDITION 也就是 -2后面会依赖于该状态。 再回到 await() 方法继续向下看接着使用了 fullyRelease() 方法传入了当前的 Node 这里的 fullyRelease() 方法主要做了释放当前线程锁的操作可以看到又调用了 AQS 的 release() 进行释放资源也就是释放了当前所持有的锁。
回到 await() 方法中当释放锁后下面进入到了 while 循环中通过查看 isOnSyncQueue() 方法可以看到是符合while的条件也就可以进入到循环中
在循环中可以明显的看到 LockSupport.park(this) 将当前线程进行了阻塞。
2.3. condition.signal() 唤醒过程
上面已经看到线程被阻塞了如果需要被唤醒则需要通过condition.signal()这个方法是如何唤醒的呢
下面来到 AbstractQueuedSynchronizer 类的 signal() 方法中
主要执行了 doSignal() 方法再点到 doSignal() 中可以看到这里开启了一个循环对链表的每一个元素都进行了 transferForSignal() 操作这里也比较好理解就是要唤醒等待中的线程。 下面点到 transferForSignal() 中看下对每个 Node 都做了什么操作。点进去之后也比较好理解如果状态是 Node.CONDITION 也就是 -2刚才在解读 await() 方法时就提到这个状态了这里正好形成了呼应下面有个非常显眼的操作 LockSupport.unpark(node.thread) 直接唤醒了目标线程。也就是唤醒了 2.2 中的最后一步操作。 2.4. condition.await() 被唤醒后
当 await() 方法中的 LockSupport.park(this) 被唤醒后继续向下执行下面会判断下当前线程有没有被打断如果没被打断则 break 终止循环继续执行。
下面会使用 AQS 的 acquireQueued() 方法将先进入队列的线程进行抢占锁资源如果成功获取锁后就会继续执行如果抢占失败则继续被挂起阻塞。 三、总结
通过上面的源码分析应该对 Condition 有了新的理解和掌握在源码中好多地方都使用了 CAS 因此当竞争资源非常激烈时 Lock 的性能要远远优于 synchronized。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/914387.shtml
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!