关于那些你可能不知道的多线程与高并发
创始人
2025-05-28 11:39:46
0

关于那些你可能不知道的多线程与高并发

锁的大白话

锁的概念其实在多线程中简单,只要满足两方面的要求就可以构成一把"锁"。

  1. 该对象是唯一的
  2. 该对象是被共享的

就好比一家有好几口人,只且一个大门且该大门上只有一把锁。每个人手中都有一个钥匙,而同一时间内只能有一个人才可以完成“获取锁🔒---->插入钥匙🔑—>旋转钥匙🔐----->开锁🔓” ,这一连续的动作。

直到当前的人完成这一操作后,另一个人才可以获取当前的这把锁 ,再次完成进行上述动作。

可以很明显的感受到,其实获取锁,完成打开锁的这过程中,一直都是一位锁持有者在进行操作。

image-20230315094953392

其实程序的很多设计思想与理念都是从生活与现实中来的,然通过语言抽象到我们的计算机中,通过计算机去模拟我们的“生活”,所以我们的多线程原理也是与上面提到的案例是相似的。

你会选择什么样的锁作为对象?

无论是我们自己在实际开发环境,还是一些 Dome 中,当我们使用锁对象的时候是否考虑过关于锁对象的选择呢?

对于这个刚开始接触多线程的初学者(如笔者我一样),都会有一个直观的感受。

那当然是选择 :

  • 1️⃣ 锁对象是唯一的

  • 2️⃣ 锁对象是共享的

其实这样的选择并没有错,通过 synchronized 可以实现多线程下多个线程的阻塞资源锁资源争用。

但却忽略了一点细节,就是在 Java 对于中一些特殊对象规避。

对于 Java 我们都知道为了提高执行的效率,有字符缓冲常量池这样的一块内存区域存在的,比如我们的 String 对象,在创建以后,就会暂时的存储在常量池当中,而在某处再次调用相同的字符串的时候,就不会再进行创建直接会从常量池中取出。

image-20230315100558403

而对于我们常用的基本类型的包装类,其实在底层对于某些常用的范围也做了处理,比如 Integer

/**
返回表示 Integer 指定 int 值的实例。如果不需要新 Integer 实例,则此方法通常应优先于构造函数 Integer(int)使用,因为此方法可能会通过缓存频繁请求的值来产生明显更好的空间和时间性能。此方法将始终缓存 -128 到 127(含)范围内的值,并可能缓存此范围之外的其他值。
*/
public static Integer valueOf(int i) {if (i >= IntegerCache.low && i <= IntegerCache.high)return IntegerCache.cache[i + (-IntegerCache.low)];return new Integer(i);}

那么,说了这么多。即便是会有缓冲池会指向同一个对象,这样又会有什么问题吗?

我的目的就是为了让一个对象是共享的,并且该对象的是唯一的呀。正好常量池中的对象满足以上的两点要求呀!

对的,常量池是满足了以上两点的要求,但明显它共享范围是在是太大了,对与每一个类对象都是共享的,这样就导致了一个非常严重的问题。

假设我只是想让 A 段业务逻辑中的线程 1 与 线程 2 共享 String 对象 str_1 = “卡卡罗特” ,从而实现对线程 1 与 2的并发控制。

但是我在另一个业务 B 中的线程 1 与 2 创建了新的 String 对象 str_2 = “卡卡罗特” ,像实现线程 1 与 2 的并发控制,并且业务 A 与 业务 B 之间没有任何的关联。

但是在实际任务启动的执行过程中,却发现业务 B 中的线程 1 与线程 2都无法执行,都处于等待的状态。

这其实就是因为由于业务 A 中的线程 1 与线程 2 ,业务 B 中的线程 1 与线程 2 都是的锁资源其实都是指向了同一个常量池中的对象 “卡卡罗特”,最终四个线程争夺一把锁(“卡卡罗特”)

image-20230315103055929

通过一个代码进行演示

package org.peggy.synchronizeds;/*** 对于基本数据类型 String 类型的常量是不能做为锁对象使用的* 这样很可能对造成其他加载的线程出现被锁占用的情况* 因为 String 类型的变量会将对象存储在缓存池中* 而对于基本数据类型的包装类,一部分数据也做了缓存的处理* @author peggy* @date 2023-03-10 19:50*/
public class StringSynchronized {
//    String s1 = "Hello";
//    String s2 = "Hello";Integer s1 = 1;Integer s2 = 1;public void getValue1() {synchronized (s1) {System.out.println("当前的线程:"+Thread.currentThread().getName());try {Thread.sleep(3000);} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println("s1==>"+s1);}}public void getValue2(){synchronized(s2){System.out.println("当前的线程:"+Thread.currentThread().getName());System.out.println("s2==>"+s2);}}public static void main(String[] args) {StringSynchronized stringSynchronized = new StringSynchronized();new Thread(stringSynchronized::getValue1,"t1").start();new Thread(stringSynchronized::getValue2,"t2").start();}
}

执行的结果如果所示

image-20230315103216526

可以看到的是,尽管我们的加锁的对象是两个不同的 Integer ,但是对于 Java 底层对于 Integer 范围在 -127~128 进行了处理,所以导致了尽管我们两个线程加锁对象,表面上看是两个不同的两个对象,但是实际上指向的是常量池中的同一个对象,使得程序在运行的期间占用同一把锁。

所以我们在多线程的情况下不可以使用 String 以及基本类型的包装类型作为锁对象使用。

你有想过为什么 synchronized 是可以重入锁吗?

相比于 Lock 的细致绵柔我更喜欢 synchronized 的简单粗暴。如果要有非要有一个具体的原因,我会说。synchronized 两个字,方便,方便,简单,还TM 的是简单。。。

其实在我接触到 Lock 锁之后,那多姿多彩的 API 让我有点眼花缭乱,觉得 Lock 锁是正的强大,也渐渐开始“鄙视”synchronized ,但秉持自己了解的越多就越发现自己就越“无知”的执念,最终才慢慢的了解到了 synchronized 的真相。

image-20230315104330770

image-20230315104403874

想必大家都知道的一点就是 synchronized 是C/C++编写的,由于其本身的是有 Java 虚拟机去管理的,本身没有具体类也没有相应的接口,它的用法一般都是“简单、粗暴”。

而对于 Lock 底层是通过 Java 实现的一种 CAS 操作,所以提供了大量功能丰富的 API。

但 synchronized 看似简单的一个修饰,其背后往往也富有其巧妙的思想与意义。看下面的代码。。。

package org.peggy.synchronizeds;/*** 可重新入锁 synchronized 测试* 如果说父类中的一个方法加锁,而子类对父类的方法进行了重写* 而在调用的过程中的调用的是 supper 父类下的方法,如果不是可重入锁,那么在执行子类的时候就会进入死锁的状态** @author peggy* @date 2023-03-09 15:25*/public class ReentrantSynchronized extends FatherSynchronized {/*** 通过这种基础实现其实就已经证明了 synchronized 锁是可重入的锁.* 否则就会出现由于先调用了子类的方法而子类优先获得了锁,当在子类中调用父类的时候,* 由于父类也需要获得当前对象的锁,但由于子类的方法还没有执行完毕,* 因此子类方法是无法释放锁资源的,父类方法就无法获取当前锁资源,导致父类的方法无法执行,* 而子类方法始终无法执行完毕,处于一个僵持的死锁状态*/@Overridepublic synchronized void setAge() {super.setAge();System.out.println("获取年龄为:" + this.age);}public synchronized void t1() {System.out.println("方法t1开始执行");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}t2();System.out.println("方法t1执行完毕");}public synchronized void t2() {System.out.println("方法t2开始执行");}public static void main(String[] args) {ReentrantSynchronized r = new ReentrantSynchronized();r.t1();r.setAge();}
}
/*** 父类对象加锁* @author peggy* @date 2023-03-09 15:33*/
public class FatherSynchronized {Integer age = 10;public synchronized void setAge() {System.out.println("父类执行完毕。。。");age--;}
}

执行的结果:

image-20230315110105567

可以看到我们的在父类的方法上用 synchronize 进行修饰,而子类在重新父类的 setAge() 方法后,也同时用 synchronize 进行修饰,同时在子类的方法中调用 super.setAge(); 方法。

可以看到的结果是:

在执行子类的 setAge() 方法是,锁已经被子类的 setAge() 方法占用,此时子类中的 setAge() 方法还没有执行完毕锁应该是不会释放的,但是在执行 super.setAge() 方法调用父类的 setAge() 方法时,锁的持有者由变为了其的 setAge() 方法。不会出现我们所设想的死锁状态。

通过下图解释:

  • 假设的运行

    image-20230315111931912

  • 实际的运行

    image-20230315112617453

通过上面的案例其实就可以看出,synchronize 是一种可重入的锁。

🧩也通过反例有 synchronize 修饰的子类方法调用 synchronize 修饰的父类方法验证了这一点,如若不然,那就真的死锁赖。

相关内容

热门资讯

2025年《财富》世界500强... 7月29日,2025年《财富》世界500强发布。小米集团排名第297位,较上一年提高100位,创下2...
中关村解除235万元银行账户资... 7月29日,中关村(000931)发布公告,公司部分银行账户资金已解除冻结。 此前,由于公司与托普天...
A股上市银行分红季:工行、建行... 7月28日,成都银行发放现金红利,至此,持续数月的银行股“红包雨”收官。 据Wind数据,A股上市银...
大牌扎堆入驻,淘宝闪购新上线1... 数据显示,品牌加速入驻淘宝闪购,7月新入驻品牌数环比6月增长110%,新上线非餐品牌门店超过1200...
高丝中国:由守转攻,从未放弃中... 高丝中国,正在进行一场颠覆式的变革。 近两年来,“高丝撤出线下”、“高丝收缩中国市场”等论调不绝于耳...
专线物流被低价绞杀?开单果帮我... “第 3 次被客户用‘头部企业便宜 200’抢走订单时,我把报价单撕得粉碎,纸屑飘了满地。” 老郑蹲...
最新消息:茅台联营公司计划突按... 每经记者|熊嘉楠 每经编辑|彭水萍 从去年以来,“厂商一体”战略被各大酒企频频推至台前。 近日,有...
附起诉书┃索尼起诉腾讯涉嫌抄袭... 翻译:叶许乐 知产财经 知产财经从海外媒体reuters获悉,索尼互动娱乐公司已在加利福尼亚联邦法院...
泉州港抢抓海南自贸港新机遇——... 泉州海关关员现场监管首次自海南洋浦港进口转关货物。(林鼎昌 摄) 记者从泉州海关获悉:7月24日,经...
原创 特... 中国海关数据显示,中国对美国的三大能源进口,已经全部清零,这让美国人担心中美谈判会非常艰难。 中国...
央企名录更新中国长安汽车集团入... 新京报贝壳财经讯(记者王琳琳)7月29日,国务院国资委网站中央企业名录更新,中国长安汽车集团有限公司...
A股午评:三大指数分化,沪指跌... 格隆汇7月29日|A股主要指数涨跌不一,截至午间收盘,沪指跌0.08%报3595.19点,深成指跌0...
2025年基金二季报划重点!泓... 来源:新浪基金 2025年第二季度泓德睿享一年持有期混合A基金净值增长率为3.09%,同期业绩比较基...
美团发文:绝不自营,浣熊食堂只... 来源:猎云网 7月29日,美团官方公众号发文称,经过半年多的试运营,7月初正式推出“浣熊食堂”品牌以...
独角兽疫苗企业三冲港股IPO:... 疫苗行业独角兽三冲港股IPO了! 在国产疫苗行业持续升级转型的浪潮中,一家坚持技术创新的企业正蓄势待...
A股站上新台阶,看好科技非银接... |2025年7月28日 星期一| NO.1中信建投:A股站上新台阶,看好科技非银接力 中信建投研报表...
中国长安汽车集团挂牌成立,朱华... 7月29日上午,中国长安汽车集团有限公司(以下简称“中国长安”)在重庆挂牌成立。这既是国内第三家汽车...