【性能提升源码】【直播采集 易语言源码】【代号降临源码怎么获得】airtest源码分析

2025-01-24 14:35:43 来源:as源码目录 分类:焦点

1.面试突击46:公平锁和非公平锁有什么区别?

airtest源码分析

面试突击46:公平锁和非公平锁有什么区别?

       从公平的码分角度来说,Java 中的码分锁总共可分为两类:公平锁和非公平锁。但公平锁和非公平锁有哪些区别?孰优孰劣呢?在 Java 中的码分应用场景又有哪些呢?接下来我们一起来看。

正文

       公平锁:每个线程获取锁的码分顺序是按照线程访问锁的先后顺序获取的,最前面的码分线程总是最先获取到锁。 非公平锁:每个线程获取锁的码分性能提升源码顺序是随机的,并不会遵循先来先得的码分规则,所有线程会竞争获取锁。码分 举个例子,码分公平锁就像开车经过收费站一样,码分所有的码分车都会排队等待通过,先来的码分车先通过,如下图所示:

       通过收费站的码分顺序也是先来先到,分别是码分张三、李四、码分王五,这种情况就是公平锁。 而非公平锁相当于,来了一个强行加塞的老司机,它不会准守排队规则,来了之后就会试图强行加塞,直播采集 易语言源码如果加塞成功就顺利通过,当然也有可能加塞失败,如果失败就乖乖去后面排队,这种情况就是非公平锁。

应用场景

       在 Java 语言中,锁 synchronized 和 ReentrantLock 默认都是非公平锁,当然我们在创建 ReentrantLock 时,可以手动指定其为公平锁,但 synchronized 只能为非公平锁。 ReentrantLock 默认为非公平锁可以在它的代号降临源码怎么获得源码实现中得到验证,如下源码所示:当使用 new ReentrantLock(true) 时,可以创建公平锁,如下源码所示:

公平和非公平锁代码演示

       接下来我们使用 ReentrantLock 来演示一下公平锁和非公平锁的执行差异,首先定义一个公平锁,开启 3 个线程,每个线程执行两次加锁和释放锁并打印线程名的操作,如下代码所示:

import?java.util.concurrent.locks.Lock;import?java.util.concurrent.locks.ReentrantLock;public?class?ReentrantLockFairTest?{ static?Lock?lock?=?new?ReentrantLock(true);public?static?void?main(String[]?args)?throws?InterruptedException?{ for?(int?i?=?0;?i?<?3;?i++)?{ new?Thread(()?->?{ for?(int?j?=?0;?j?<?2;?j++)?{ lock.lock();System.out.println("当前线程:"?+?Thread.currentThread().getName());lock.unlock();}}).start();}}}

       以上程序的执行结果如下图所示:接下来我们使用非公平锁来执行上面的代码,具体实现如下:

import?java.util.concurrent.locks.Lock;import?java.util.concurrent.locks.ReentrantLock;public?class?ReentrantLockFairTest?{ static?Lock?lock?=?new?ReentrantLock();public?static?void?main(String[]?args)?throws?InterruptedException?{ for?(int?i?=?0;?i?<?3;?i++)?{ new?Thread(()?->?{ for?(int?j?=?0;?j?<?2;?j++)?{ lock.lock();System.out.println("当前线程:"?+?Thread.currentThread().getName());lock.unlock();}}).start();}}}

       以上程序的执行结果如下图所示:从上述结果可以看出,使用公平锁线程获取锁的顺序是:A -> B -> C -> A -> B -> C,也就是主力机构潜伏指标源码按顺序获取锁。而非公平锁,获取锁的顺序是 A -> A -> B -> B -> C -> C,原因是所有线程都争抢锁时,因为当前执行线程处于活跃状态,其他线程属于等待状态(还需要被唤醒),所以当前线程总是会先获取到锁,所以最终获取锁的顺序是:A -> A -> B -> B -> C -> C。

执行流程分析公平锁执行流程

       获取锁时,先将线程自己添加到等待队列的队尾并休眠,当某线程用完锁之后,爆量指标公式源码会去唤醒等待队列中队首的线程尝试去获取锁,锁的使用顺序也就是队列中的先后顺序,在整个过程中,线程会从运行状态切换到休眠状态,再从休眠状态恢复成运行状态,但线程每次休眠和恢复都需要从用户态转换成内核态,而这个状态的转换是比较慢的,所以公平锁的执行速度会比较慢。

非公平锁执行流程

       当线程获取锁时,会先通过 CAS 尝试获取锁,如果获取成功就直接拥有锁,如果获取锁失败才会进入等待队列,等待下次尝试获取锁。这样做的好处是,获取锁不用遵循先到先得的规则,从而避免了线程休眠和恢复的操作,这样就加速了程序的执行效率。 公平锁和非公平锁的性能测试结果如下,以下测试数据来自于《Java并发编程实战》:

        从上述结果可以看出,使用非公平锁的吞吐率(单位时间内成功获取锁的平均速率)要比公平锁高很多。

优缺点分析

       公平锁的优点是按序平均分配锁资源,不会出现线程饿死的情况,它的缺点是按序唤醒线程的开销大,执行性能不高。 非公平锁的优点是执行效率高,谁先获取到锁,锁就属于谁,不会“按资排辈”以及顺序唤醒,但缺点是资源分配随机性强,可能会出现线程饿死的情况。

总结

       在 Java 语言中,锁的默认实现都是非公平锁,原因是非公平锁的效率更高,使用 ReentrantLock 可以手动指定其为公平锁。非公平锁注重的是性能,而公平锁注重的是锁资源的平均分配,所以我们要选择合适的场景来应用二者。

       是非审之于己,毁誉听之于人,得失安之于数。

       公众号:Java面试真题解析

       面试合集:/post/

更多资讯请点击:焦点

推荐资讯

​辽宁盘锦:食品安全“六大体系”显成效

中国消费者报沈阳讯记者王文郁)12月4日,记者从辽宁省盘锦市市场监管局了解到,该市通过建立健全食品安全“六大体系”,持续开展保障食品安全相关专项行动,食品安全监管成效全面凸显,人民群众对食品安全的满意

裴洛西離台前推特發文 「美國對台灣人民承諾堅定不移」

美國眾議院議長裴洛西Nancy Pelosi)訪台行程今3)日傍晚結束,於下午5點53分左右離台。離開前裴洛西在推特上發文表示,美國對台灣人民的承諾始終堅定不移,不管是現在或未來幾十年都一樣。裴洛西表

最低稅負過關 誰受影響最大?|天下雜誌

立法院在十二月九日迅速三讀,讓備受矚目,朝野合力推動的最低稅負制所得基本稅額條例)通過,成為三十六年來唯一對所得稅加稅的法案。