Synchronized 和 Lock

1、原始构成

Synchronized 是关键字属于JVM层面 (代码中以蓝色字体呈现)
monitorenter 、monitorexit
Lock 是具体类(java.util.concurrent.locks.lock)是api层面的锁

2、使用方法

Synchronized 不需要用户手动释放锁,当synchronized代码执行完后系统会自动让线程释放对锁的占用
ReentrantLock 需要用户手动释放锁 如果没有释放,就会导致死锁
需要lock() 和 unlock()方法配合try/finally语句块来完成 (Lock需要在while中使用,不能在if里面使用)

3、等待是否可中断

Synchronized 不可中断,除非抛出异常 或 正常运行完成
ReentrantLock 可中断
1>设置超时方法 tryLock(long timeout, TimeUnit unit)
2>lockInterruptibly() 放代码块中,调用 interrup() 方法可中断

4、加锁是否公平

Synchronized 非公平锁
ReentrantLock 两者都可以,默认非公平锁 (可重入锁) 构造方法可以传入 boolean值,true公平锁 false非公平锁

5、锁绑定多个条件Condition

Synchronized 没有
ReentrantLock 用来实现分组唤醒需要唤醒的线程们,可以精确唤醒 而synchronized要么随机唤醒一个,要么全部唤醒

题目:多线程之间顺序调用,实现A->b->C三个线程启动,要求如下:

AA打印5次,BB打印10次,CC打印15次 重复10次

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; /**
* @Author: LeavesCai7
* @Date: 2021/01/22 11:29
* 题目:多线程之间顺序调用,实现A->b->C三个线程启动,要求如下:
* AA打印5次,BB打印10次,CC打印15次 重复10次
* ReentrantLock
*/ class shareResource{
private int num = 1; //A:1 B:2 C:3
private Lock lock = new ReentrantLock();
private Condition c1 = lock.newCondition();
private Condition c2 = lock.newCondition();
private Condition c3 = lock.newCondition(); public void print5(){
lock.lock();
try{ //判断
while (num != 1){
c1.await();
} //干活
for (int i =1;i<=5;i++){
System.out.println(Thread.currentThread().getName() + "\t" + i);
} //通知
num = 2;
c2.signal(); }catch (Exception e){
e.printStackTrace();
}finally {
lock.unlock();
}
}
public void print10(){
lock.lock();
try{ //判断
while (num != 2){
c1.await();
} //干活
for (int i =1;i<=10;i++){
System.out.println(Thread.currentThread().getName() + "\t" + i);
} //通知
num = 3;
c3.signal(); }catch (Exception e){
e.printStackTrace();
}finally {
lock.unlock();
}
} public void print15() {
lock.lock();
try { //判断
while (num != 3) {
c1.await();
} //干活
for (int i = 1; i <= 15; i++) {
System.out.println(Thread.currentThread().getName() + "\t" + i);
} //通知
num = 1;
c3.signal(); } catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
} public class SyncAndReentrantLockDemo {
public static void main(String[] args) {
shareResource shareResource = new shareResource(); new Thread(() -> {
for (int i = 0; i <= 10; i++) {
shareResource.print5();
}
},"A").start(); new Thread(() -> {
for (int i = 0; i <= 10; i++) {
shareResource.print10();
}
},"B").start(); new Thread(() -> {
for (int i = 0; i <= 10; i++) {
shareResource.print15();
}
},"C").start(); }
}

Synchronized和Lock有什么区别?用Lock有什么好处?的更多相关文章

  1. synchronized 和 Lock 有什么区别?(未完成)

    synchronized 和 Lock 有什么区别?(未完成)

  2. 用阻塞队列实现一个生产者消费者模型?synchronized和lock有什么区别?

    多线程当中的阻塞队列 主要实现类有 ArrayBlockingQueue是一个基于数组结构的有界阻塞队列,此队列按FIFO原则对元素进行排序 LinkedBlockingQueue是一个基于链表结构的 ...

  3. synchronized和lock有什么区别?

    一.原始构成 synchronized是关键字属于JVM层面,monitorenter(底层是通过monitor对象来完成,其实wait/notify等方法也依赖monitor对象只有在同步代码块和同 ...

  4. java面试-synchronized与lock有什么区别?

    1.原始构成: synchronized是关键字,属于JVM层面,底层是由一对monitorenter和monitorexit指令实现的. ReentrantLock是一个具体类,是API层面的锁. ...

  5. Synchronized和Lock, 以及自旋锁 Spin Lock, Ticket Spin Lock, MCS Spin Lock, CLH Spin Lock

    Synchronized和Lock synchronized是一个关键字, Lock是一个接口, 对应有多种实现. 使用synchronized进行同步和使用Lock进行同步的区别 使用synchro ...

  6. C#中Monitor对象与Lock关键字的区别分析

    这篇文章主要介绍了C#中Monitor对象与Lock关键字的区别,需要的朋友可以参考下 Monitor对象 1.Monitor.Enter(object)方法是获取 锁,Monitor.Exit(ob ...

  7. 转:C#中Monitor对象与Lock关键字的区别分析

    Monitor对象1.Monitor.Enter(object)方法是获取 锁,Monitor.Exit(object)方法是释放锁,这就是Monitor最常用的两个方法,当然在使用过程中为了避免获取 ...

  8. 【JUC】synchronizated和lock的区别&新lock的优势

    原始构成 synchronized是关键字,属于JVM层面 javap -c 的结果显示 synchronized是可重入锁 11:是正常退出 17:是异常退出[保证不产生死锁和底层故障] Lock是 ...

  9. ReentrantLock可重入锁lock,tryLock的区别

    void lock(); Acquires the lock. Acquires the lock if it is not held by another thread and returns im ...

  10. volatile和synchronized实现内存可见性的区别

    先看看synchronized实现内存可见性 加锁(synchronized同步)的功能不仅仅局限于互斥行为,同时还存在另外一个重要的方面:内存可见性.我们不仅希望防止某个线程正在使用对象状态而另一个 ...

随机推荐

  1. scrapy框架学习(六)日志设置和数据存储

    日志设置 CRITICAL : 严重错误 ERROR : 一般错误 WARNING : 警告 INFO : 一般的信息 DEBUG : 调试信息 默认的显示级别是DEBUG # 设置错误显示级别 LO ...

  2. qt vnc 远程左桌面

    设备端启动服务器 ./testqt -qws -display VNC:0 客服端启动监控端 VNC Viewer 4

  3. uniapp改变icon

    <!DOCTYPE html><html lang="zh-CN"> <head> <meta charset="utf-8&q ...

  4. Java8 提供CompletableFuture来简化高并发异步处理编程

    (摘录自狂乱的贵公子)所谓异步调用其实就是实现一个可无需等待被调用函数的返回值而让操作继续运行的方法.在 Java 语言中,简单的讲就是另启一个线程来完成调用中的部分计算,使调用继续运行或返回,而不需 ...

  5. Characterizing possible failure modes in physics-informed neural network

    NeurIPS 2021 表征PINN中可能的失败模式.本文的思路也比较简单,通过对PINN的优化域进行观察,发现导致PINN训练的原因并不是因为神经网络的表达力不足,而是由于PINN中引入了基于PD ...

  6. 吴恩达老师机器学习课程chapter03——过拟合与正则化

    吴恩达老师机器学习课程chapter03--过拟合与正则化 本文是非计算机专业新手的自学笔记,欢迎指正与其他任何合理交流. 本文仅作速查备忘之用,对应吴恩达(AndrewNg)老师的机器学期课程第七章 ...

  7. django日志分割的几种处理

    https://blog.csdn.net/qq_42819407/article/details/125041634

  8. javax.naming.NameNotFoundException: Name [aa.bb.cc/xx] is not bound in this Context. Unable to find [aa.bb.cc]

    今天尝试把一个SpringBoot项目由jdk11转移为jdk8,在IDEA里运行一切正常.打包,部署到tomcat启动后,在客户端调用WebSocket出现如下异常: java.lang.Illeg ...

  9. (二)REDIS-重要概念与原理

    1 Redis的概念: Redis是一种key-value类型的内存数据库,可以用于保存string,list,set,sorted set,hash等多种数据结构.由于整个数据库统统加载在内存中进行 ...

  10. Python学习笔记调试之取得反向跟踪的字符串

    随笔记录方便自己和同路人查阅. #------------------------------------------------我是可耻的分割线--------------------------- ...