多线程锁

Synchronized锁的八种情况

以一个手机类为例,包含两个synchronized方法和一个普通方法。

public class Phone {
public synchronized void sendSMS() throws InterruptedException {
System.out.println("-----send SMS-----");
} public synchronized void sendEmail() {
System.out.println("-----send email-----");
} public void getHello() {
System.out.println("-----get hello-----");
}
}
  • 1 标准访问
    public static void main(String[] args) {
Phone phone = new Phone();
new Thread(() -> {
phone.sendSMS();
}).start(); new Thread(() -> {
phone.sendEmail();
}).start();
}
-----send SMS-----
-----send email-----
  • 2 短信方法设置4s停止
public class Phone {
public synchronized void sendSMS() {
try {
TimeUnit.SECONDS.sleep(4);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("-----send SMS-----");
} public synchronized void sendEmail() {
System.out.println("-----send email-----");
} public void getHello() {
System.out.println("-----get hello-----");
}
}

-----send SMS-----

-----send email-----

会先等待四秒,短信发送结束才会发送邮件

  • 3 短信停止四秒,同时运行getHello
public class Lock_8 {
public static void main(String[] args) {
Phone phone = new Phone();
new Thread(() -> {
phone.sendSMS();
}).start(); new Thread(() -> {
phone.getHello();
}).start();
}
}

-----get hello-----

4s。。。

-----send SMS-----

  • 4 两部手机,一部邮件另一部短信
public static void main(String[] args) {
Phone phone = new Phone();
Phone phone2 = new Phone();
new Thread(() -> {
phone.sendSMS();
}).start(); new Thread(() -> {
phone2.sendEmail();
}).start();
}

-----send email-----

4s...

-----send SMS-----

  • 5 两个静态方法,一部手机,先打印邮件还是短信(短信停4s,然后邮件)
  • 6 两个静态方法,两部手机,先打印邮件还是短信(短信停4s,然后邮件)
  • 7 一个静态短信方法,一个邮件方法,1部手机(先邮件,停4s短信)
  • 8 一个静态短信方法,一个邮件方法,2部手机(先邮件,停4s短信)

总结

  • synchronized加在普通方法上,是对this进行加锁,作用范围为当前实例对象
  • synchronized加在静态方法上,是对类的类加载的Class对象加锁,因此作用范围是全部的实例对象

公平锁和非公平锁

当锁被一个线程释放的时候,其他线程抢占锁的机会是否是公平的,分为公平锁和非公平锁。如之前卖票的例子,导致ABC卖票全被A卖了:

  • 公平锁可以防止线程饥饿,但是效率低
A sale 1 ticket, there are 29 tickets left
A sale 1 ticket, there are 28 tickets left
A sale 1 ticket, there are 27 tickets left
A sale 1 ticket, there are 26 tickets left
A sale 1 ticket, there are 25 tickets left
A sale 1 ticket, there are 24 tickets left
A sale 1 ticket, there are 23 tickets left
A sale 1 ticket, there are 22 tickets left
... Process finished with exit code 0

通过ReetranLock构造器设置公平锁

lock = new ReentrantLock(true);
A sale 1 ticket, there are 29 tickets left
B sale 1 ticket, there are 28 tickets left
C sale 1 ticket, there are 27 tickets left
A sale 1 ticket, there are 26 tickets left
B sale 1 ticket, there are 25 tickets left
C sale 1 ticket, there are 24 tickets left
A sale 1 ticket, there are 23 tickets left
B sale 1 ticket, there are 22 tickets left
C sale 1 ticket, there are 21 tickets left

可重入锁

可重入锁:可以重复使用的锁,即某个线程已经获得了某个锁,再次获取锁而不会出现死锁,可重入锁有:

  • synchronized
  • ReentrantLock:ReentrantLock 和 synchronized 不一样,需要手动释放锁,所以使用 ReentrantLock的时候一定要手动释放锁,并且加锁次数和释放次数要一样
synchronized
public static void main(String[] args) {
var obj = new Object();
new Thread(() -> {
synchronized (obj) {
System.out.println("外层");
synchronized (obj) {
System.out.println("中层");
synchronized (obj) {
System.out.println("内层");
}
}
}
}).start();
}

可重入锁又称为递归锁,能够实现同步方法的递归调用而不产生死锁

    static synchronized void add() {
add();
}
public static void main(String[] args) {
new Thread(ThreadDemo4::add).start();
}
Lock
class ShareSource {
private final Lock lock = new ReentrantLock();
public void lockTest() {
lock.lock();
System.out.println("waiceng");
lock.lock();
System.out.println("zhongceng");
lock.lock();
System.out.println("neiceng");
lock.unlock();
lock.unlock();
}
} public class ThreadDemo4 { public static void main(String[] args) {
ShareSource shareSource = new ShareSource();
new Thread(shareSource::lockTest).start();
new Thread(shareSource::lockTest).start();
}
}

Lock不会自动释放锁,上锁和解锁不对应的话会导致死锁的发生,如这里的第二个线程就一直执行不了

死锁

def:两个或两个以上的线程在执行过程中,由于竞争资源而造成的相互等待的现象,如果没有外力作用,则会一直等待下去。

死锁产生的原因

  1. 竞争资源
  2. 进程间推进顺序非法
  3. 资源分配不当
public static void main(String[] args) {
var a = new Object();
var b = new Object(); new Thread(() -> {
synchronized (a) {
System.out.println("已获取锁A,尝试获取锁B");
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (b) {
System.out.println("已获取锁B");
}
}
}).start(); new Thread(() -> {
synchronized (b) {
System.out.println("已获取锁B,尝试获取锁A");
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (a) {
System.out.println("已获取锁A");
}
}
}).start();
}
检查死锁
  1. jsp:查看系统中运行的java进程
  2. jstack:追踪java进程的堆栈情况
"Thread-0":
at com.hikaru.juc.lock.DeadLockDemo.lambda$main$0(DeadLockDemo.java:19)
- waiting to lock <0x0000000711818d28> (a java.lang.Object)
- locked <0x0000000711818d18> (a java.lang.Object)
at com.hikaru.juc.lock.DeadLockDemo$$Lambda$14/0x00000008010031f0.run(Unknown Source)
at java.lang.Thread.run(java.base@19.0.2/Thread.java:1589)
"Thread-1":
at com.hikaru.juc.lock.DeadLockDemo.lambda$main$1(DeadLockDemo.java:33)
- waiting to lock <0x0000000711818d18> (a java.lang.Object)
- locked <0x0000000711818d28> (a java.lang.Object)
at com.hikaru.juc.lock.DeadLockDemo$$Lambda$15/0x0000000801003400.run(Unknown Source)
at java.lang.Thread.run(java.base@19.0.2/Thread.java:1589) Found 1 deadlock.

JUC(四)多线程锁的更多相关文章

  1. JUC之多线程锁问题

    多线程锁 8种问题锁状态: 该部分全部围绕的是以下内容并结合相应的例子:synchronized实现同步的基础:Java中每个对象都可以作为锁. 具体表现为以下三种形式:(之前只是简单的了解) 对于普 ...

  2. Java并发编程(3) JUC中的锁

    一 前言 前面已经说到JUC中的锁主要是基于AQS实现,而AQS(AQS的内部结构 .AQS的设计与实现)在前面已经简单介绍过了.今天记录下JUC包下的锁是怎么基于AQS上实现的 二 同步锁 同步锁不 ...

  3. Python多线程锁

    [Python之旅]第六篇(四):Python多线程锁   python lock 多线程 多线程使用方法 多线程锁 摘要:   在多线程程序执行过程中,为什么需要给一些线程加锁以及如何加锁,下面就来 ...

  4. java 并发多线程 锁的分类概念介绍 多线程下篇(二)

    接下来对锁的概念再次进行深入的介绍 之前反复的提到锁,通常的理解就是,锁---互斥---同步---阻塞 其实这是常用的独占锁(排它锁)的概念,也是一种简单粗暴的解决方案 抗战电影中,经常出现为了阻止日 ...

  5. Java多线程--锁的优化

    Java多线程--锁的优化 提高锁的性能 减少锁的持有时间 一个线程如果持有锁太长时间,其他线程就必须等待相应的时间,如果有多个线程都在等待该资源,整体性能必然下降.所有有必要减少单个线程持有锁的时间 ...

  6. synchronized与static synchronized 的差别、synchronized在JVM底层的实现原理及Java多线程锁理解

    本Blog分为例如以下部分: 第一部分:synchronized与static synchronized 的差别 第二部分:JVM底层又是怎样实现synchronized的 第三部分:Java多线程锁 ...

  7. Java高并发与多线程(四)-----锁

    今天,我们开始Java高并发与多线程的第四篇,锁. 之前的三篇,基本上都是在讲一些概念性和基础性的东西,东西有点零碎,但是像文科科目一样,记住就好了. 但是本篇是高并发里面真正的基石,需要大量的理解和 ...

  8. java并发编程(四)----(JUC)Lock锁初探

    首先我们来回忆一下上一节讲过的synchronized关键字,该关键字用于给代码段或方法加锁,使得某一时刻它修饰的方法或代码段只能被一个线程访问.那么试想,当我们遇到这样的情况:当synchroniz ...

  9. 细说.NET中的多线程 (四 使用锁进行同步)

    通过锁来实现同步 排它锁主要用来保证,在一段时间内,只有一个线程可以访问某一段代码.两种主要类型的排它锁是lock和Mutex.Lock和Mutex相比构造起来更方便,运行的也更快.但是Mutex可以 ...

  10. Java - "JUC" ReentrantLock获取锁

    [Java并发编程实战]-----“J.U.C”:ReentrantLock之一简介 ReentrantLock介绍 ReentrantLock是一个可重入的互斥锁,又被称为“独占锁”. 顾名思义,R ...

随机推荐

  1. 【基础知识】C++算法基础(快速排序)

    快速排序: 1.执行流程(一趟快排): 2.一趟快排的结果:获得一个枢纽,在此左边皆小于此数,在此右边皆大于此数,因此可以继续使用递归获得最终的序列.

  2. 如何利用 C# + Echarts 绘制 Bar Simple

    背景 Echarts 是百度推出的一个使用 JavaScript 实现的开源可视化库. 该库提供了常规的折线图.柱状图.散点图.饼图.K线图,用于统计的盒形图,用于地理数据可视化的地图.热力图.线图, ...

  3. 让 rtb 不显示 横纵 滚动条的方法

    让 rtb 不显示 横纵 滚动条的方法: a.设置属性: tb.ScrollBars=None; b.设置属性:rtb.WordWarp=False; c. 添加事件代码: rtb.ContentsR ...

  4. Glinux 1395 build ENV setup

    1.安装ubuntu 14.04.05 LTS 64bit 2.初始化root 密码 sudo passwd  Password: <--- 输入安装时那个用户的密码  Enter new UN ...

  5. C语言中字符数组的赋值和复制

    /*C中,字符串,即字符数组的赋值与字符变量.常量.变量的赋值是不同的.初学者总会犯错误. 常见错误如下: 1.定义的时候直接用字符串赋值 char a[10]; char a[10]="h ...

  6. 如何通过dump文件定位崩溃,解决问题

    1.dump文件是附加堆栈信息的存储文件的简称,文件扩展名".dmp",通过dump文件我们可以得到程序运行某一时刻的堆栈数据.2.当程序意外崩溃后,通常程序会立即中断运行,拿到这 ...

  7. 解决idea xml文件中的中文注释乱码

    今天用idea编译xml文件的时候报错了,打开编译好的xml文件发现是中文乱码问题,按照百度上靠前的办法设置了一下,并没有解决乱码问题,在当前项目中直接设置就是不生效,最后删掉编译好的 target/ ...

  8. shell多进程并发数控制

    在批量执行任务时,单进程执行速度太慢,使用&不加数量控制,又担心资源占用过多,导致宕机等问题,因此我们需要控制并发进程的数量,保证效率的同时,保证资源占用不会太高. 其中一个解决思路是利用简单 ...

  9. Strus框架

    一.框架概述 1.框架的意义与作用: 所谓框架,就是把一些繁琐的重复性代码封装起来,使程序员在编码中把更多的经历放到业务需求的分析和理解上面. 特点:封装了很多细节,程序员在使用的时候会非常简单. 2 ...

  10. Matlab - 在Figure中调整X轴到x=0,y=0处

    原图 调整 选中图像后打开属性编辑器,将X轴位置修改为原点. 结果