关于Synchronized的小结

news/2024/11/29 20:53:14/

目录

一.特性

1.既是乐观锁又是悲观锁

2.是轻量级锁,也是重量级锁 

 3.不是读写锁,是互斥锁

4.是可重入锁

5.非公平锁

6.加锁之后,运行完毕自动解锁释放资源

 二:Synchronized使用

第一种:定义一个额外的变量来控制加锁和解锁(类似于吉祥物)

第一种:直接给类/方法上锁

三.synchronized的优化

运行机制上的优化

 编译阶段进行的优化手段

锁消除

程序员代码上进行优化

锁粗化


一.特性

1.既是乐观锁又是悲观锁

乐观锁:就是在操作数据时非常乐观,认为别的线程不会同时修改数据,所以不会上锁,但是在更新的时候会判断在此期间别的线程有没有更新过这个数据。

悲观锁:就在操作数据时比较悲观,每次去拿数据的时候认为别的线程也会同时修改数据,所以每次在拿数据的时候都会上锁,这样别的线程想拿到这个数据就会阻塞直到它拿到锁。

2.是轻量级锁,也是重量级锁 

偏向锁:单线程情况下,第一个使用锁,就升级为偏向锁

轻量级锁:多线程下,第二个线程来同一个线程竞争同一个资源,这个时候偏向锁升级为轻量级锁(竞争很小)

重量级锁:超级多线程同时来竞争一个资源(竞争非常激烈),这个时候由轻量级锁升级为重量级锁

 3.不是读写锁,是互斥锁

读写锁:多线程下,同一时刻只能有一个线程读取这份资源,其他线程不能读取处于阻塞

互斥锁:多线程下,同一时刻只能有一个线程使用这份资源,其他线程不能使用处于阻塞

4.是可重入锁

可重入锁:在两个带锁的线程中,当前线程1的锁没被解锁的情况下,能够去执行带有锁的线程2

不可重入锁:在两个带锁的线程中,当前线程1的锁没被解锁的情况下,不可以去执行带有锁的线程2

5.非公平锁

公平锁:多线程下,当锁被线程1释放后,线程2,3,4依据自己谁先请求资源的时间先后获取该资源,进行上锁操作

非公平锁:多线程下,当锁被线程1释放后,不管时间请求的顺序先后,线程2,3,4各凭本事抢占该资源,进行上锁操作

6.加锁之后,运行完毕自动解锁释放资源

自动释放:Synchronized的代码执行完毕之后,不用调用解锁的代码,就能自动解锁释放资源

非自动释放:与Synchronized当对应的有个叫做ReentrantLock的锁,ReentrantLock就是调用之后,必须要程序员手动输入"解锁指令"来关闭

 二:Synchronized使用

第一种:定义一个额外的变量来控制加锁和解锁(类似于吉祥物)

public class demo2 {
//定义一个Object类,名字叫做lock的变量作为锁static  Object lock = new Object();public static void main(String[] args) {synchronized (lock){}}
}

第一种:直接给类/方法上锁

//给类上锁
public class demo2 {synchronized public static void main(String[] args) {}
}//给方法上锁
public class demo2 {synchronized static void  func(){System.out.println("hello");}public static void main(String[] args) {}
}

三.synchronized的优化

运行机制上的优化

无->偏向锁->轻量级锁->重量级锁

注意:在轻量级锁这里synchronized,采用[自适应自旋锁]

自旋锁:一直不停反复查看当前锁是否被释放,一旦释放,自己就立刻占为己有,为期上锁

自适应自旋锁:重复一定次数/一定时间后,停止查看

 编译阶段进行的优化手段

锁消除

锁消除:编译器+JVM会检测当前代码是否是多线程执行,是否有必要加锁,如果没有必要,在编写的时候又把锁给写上了,就会在编译过程中自动把锁去掉。

列如StringBuffer,他每次apped操作都是需要加锁和解锁,如果只有简单的append操作的话

 那么编译器就直接进行一次加锁和解锁,而不是多次加锁和解锁

程序员代码上进行优化

锁粗化

锁的粒度:synchronized代码块,包含代码的多少(代码越多,粒度越大 ;代码越少,粒度越细)。 

一般写代码的时候,多数情况下,是希望锁的粒度更小一点(串行执行的代码少,并发执行的代码就多)。串行代码越少越少,程序执行就越快。

 举一个例子

假如说你现在需要向领导汇报三个工作

你可以选择,分三次汇报三个不同的工作

或者选择,一次汇报三个工作

显然,我们是更倾向于一次汇报三个工作的,因为他更高效


http://www.ppmy.cn/news/531910.html

相关文章

L2-014 列车调度 (25 分)

L2-014 列车调度 (25 分) 火车站的列车调度铁轨的结构如下图所示。 两端分别是一条入口(Entrance)轨道和一条出口(Exit)轨道,它们之间有N条平行的轨道。每趟列车从入口可以选择任意一条轨道进入,最后从出口…

427-二叉树(617.合并二叉树、700.二叉搜索树中的搜索、98. 验证二叉搜索树、530.二叉搜索树的最小绝对差)

617.合并二叉树 class Solution { public:TreeNode* mergeTrees(TreeNode* root1, TreeNode* root2) {if (root1 nullptr) return root2;if (root2 nullptr) return root1;root1->val root2->val;root1->left mergeTrees(root1->left, root2->left);r…

【卡特兰数】HJ77.火车进站

题目&#xff1a; 描述 给定一个正整数N代表火车数量&#xff0c;0<N<10&#xff0c;接下来输入火车入站的序列&#xff0c;一共N辆火车&#xff0c;每辆火车以数字1-9编号&#xff0c;火车站只有一个方向进出&#xff0c;同时停靠在火车站的列车中&#xff0c;只有后进站…

210920-车站问题

mainByHuber.cpp 文件&#xff1a; #include <iostream> #include <string.h> #include<stdlib.h> struct station_info{char station_name[100]; }; struct station_node{station_info s;station_node * next; };int findStation(station_node * head,cha…

12306一直显示服务器忙,网购春运火车票首日 12306网站又现服务器忙

①青岛火车站售票大厅内&#xff0c;购票回家的旅客已明显增多。 ②来自广西的杨先生和同伴们准备拎着行李提前回家。据了解&#xff0c;他们在船厂工作&#xff0c;由于天冷活少&#xff0c;所以提前放假了。 ③"小候鸟"在妈妈的背后等候进站上车。 本版图/记者 孙传…

L2-014 列车调度 - JAVA

L2-014 列车调度 题目描述&#xff1a; 火车站的列车调度铁轨的结构如下图所示。 两端分别是一条入口&#xff08; Entrance &#xff09;轨道和一条出口&#xff08; Exit &#xff09;轨道&#xff0c;它们之间有 N 条平行的轨道。每趟列车从入口可以选择任意一条轨道进入&…

7-2 列车调度(25 分)

火车站的列车调度铁轨的结构如下图所示。 两端分别是一条入口&#xff08;Entrance&#xff09;轨道和一条出口&#xff08;Exit&#xff09;轨道&#xff0c;它们之间有N条平行的轨道。每趟列车从入口可以选择任意一条轨道进入&#xff0c;最后从出口离开。在图中有9趟列车…

什么蓝牙耳机通话质量好?高清通话蓝牙耳机排行

由于无线蓝牙耳机方便佩戴易于操作&#xff0c;如今已经成为手机用户欣赏音乐的主流外设&#xff0c;蓝牙耳机的产量每年都是在递增的状态&#xff0c;使用蓝牙耳机的场景有很多&#xff0c;听歌难免会有电话的进入&#xff0c;下面分享几款高清通话的蓝牙耳机。 TOP1、南卡小…