在多线程编程中,要执行synchronized块,必须首先获得指定对象的锁。

1.Java的线程锁是可重入的锁

    public void add(int m){
synchronized (lock){
this.value += m;
}
}

什么叫可重入的锁?

对同一个对象,同一个线程,可以多次获取它的锁,即同一把锁可以嵌套。

在add方法中,用synchronized获取了一个lock对象的锁,在synchronized代码块内部,调用another方法,可以再次获取到lock对象的锁。因为这是同一个线程对lock对象获取锁,所以它是一种可重入的锁。

    public void add(int m){
synchronized (lock){
this.value += m;
another(m);
}
}
public void addAnother(int m){
synchronized (lock){
this.another += m;
}
}
//上述代码可以合并为
public void add(int m){
synchronized (lock){
this.value += m;
synchronized (lock){
this.another += m;
}
}
}

2. Java的线程还可以获取多个不同对象的锁

先用synchronized(lockA)获得lockA的锁,在用synchronized(lockB)获得lockB的锁。在释放锁的时候,会依次释放lockB的锁,再释放lockA的锁。

    public void add(int m){
synchronized (lockA){//获取lockA的锁
this.value += m;
synchronized (LockB){//获取lockB的锁
this.another += m;
}//释放lockB的锁
}//释放lockA的锁
}

3.死锁

不同的线程获取多个不同对象的锁可能导致死锁

线程1和线程2分别执行。

线程1获取lockA的锁,线程2获得lockB的锁。

线程1执行语句,线程2执行语句。

线程1等待lockB的锁,线程2等待lockA的锁。这个时候,线程1和线程2会永远等待下去,谁也无法继续执行,就形成了死锁。



死锁形成的条件:

  • 两个线程各自持有不同的锁。线程1持有A锁,线程2持有B锁。
  • 两个线程各自试图获取对方的锁。线程1等待B锁,线程2等待A锁。
  • 双方无限等待下去,导致死锁

死锁发生后:

  • 没有任何机制能解除死锁
  • 只能强制结束JVM进程

所以,我们在编写多线程程序时,必须要避免出现死锁。

如何避免死锁:线程获取锁的顺序要一致

4.示例

4.1 死锁

class ShareObject{
final Object lockA = new Object();
final Object lockB = new Object();
int accountA = 1000;
int accountB = 2000;
public void a2b(int balance){ //a2b: acountA减, accountB加
synchronized (lockA){
accountA -= balance;
synchronized (lockB){
accountB += balance;
}
}
}
public void b2a(int balance){ //b2a:acountA加, accountB减
synchronized (lockB){
accountB -= balance;
synchronized (lockA){
accountA += balance;
}
}
}
}
class AThread extends Thread{
public void run(){
for(int i=0;i<Main.LOOP;i++){
Main.shared.a2b(1);
if(i%100==0){
System.out.println(".");
}
}
}
}
class BThread extends Thread{
public void run(){
for(int i=0;i<Main.LOOP;i++){
Main.shared.b2a(1);
if(i%100==0){
System.out.println(".");
}
}
}
}
public class Main{
final static int LOOP = 1000;
public static ShareObject shared = new ShareObject();
public static void main(String[] args) throws Exception{
Thread t1 = new AThread();
Thread t2 = new BThread();
t1.start();
t2.start();
t1.join();
t2.join();
System.out.println("end");
}
}


### 4.2 避免死锁
修改b2a()
```#java
public void b2a(int balance){
synchronized (lockA){
accountA += balance;
synchronized (lockB){
accountB -= balance;
}
}
}
}
```

4总结

  • 死锁产生的条件:

    * 多线程各自持有不同的锁,并互相试图获取对方已持有的锁
  • 如何避免死锁?

    * 多线程获得锁的顺序要一致

廖雪峰Java11多线程编程-2线程同步-3死锁的更多相关文章

  1. 廖雪峰Java11多线程编程-2线程同步-2synchronized方法

    1.Java使用synchronized对一个方法进行加锁 class Counter{ int count = 0; public synchronized void add(int n){ cou ...

  2. 廖雪峰Java11多线程编程-2线程同步-1同步代码块

    1.线程安全问题 多个线程同时运行,线程调度由操作系统决定,程序本身无法决定 如果多个线程同时读写共享变量,就可能出现问题 class AddThread extends Thread{ public ...

  3. 廖雪峰Java11多线程编程-2线程同步-4wait和notify

    wait和notify synchronized解决了多线程竞争的问题 我们可以在synchronized块中安全的对一个变量进行修改,但是它没有解决多线程协调的问题. 例如设计一个TaskQueue ...

  4. 廖雪峰Java11多线程编程-1线程的概念-1多线程简介

    多任务 现代操作系统(windows,MacOS,Linux)都可以执行多任务: 多任务就是同时运行多个任务,例如同时开启钉钉.百度网盘.火狐.谷歌.ps等 操作系统执行多任务就是让多个任务交替执行, ...

  5. 廖雪峰Java11多线程编程-4线程工具类-1ThreadLocal

    多线程是Java实现多任务的基础: Thread ExecutorService ScheduledThreadPool Fork/Join Thread对象代表一个线程:调用Tread.curren ...

  6. 廖雪峰Java11多线程编程-1线程的概念-2创建新线程

    Java语言内置多线程支持: 一个Java程序实际上是一个JVM进程 JVM用一个主线程来执行main()方法 在main()方法中又可以启动多个线程 1.创建新线程 1.1 方法一:使用Thread ...

  7. 廖雪峰Java11多线程编程-1线程的概念-5中断线程

    1.中断线程: 如果线程需要执行一个长时间任务,就可能需要中断线程.场景:从网络上下载一个100M的文件,用户在下载过程中中断下载任务的执行. 中断线程就是其他线程给该线程发一个信号,该线程收到信号后 ...

  8. 廖雪峰Java11多线程编程-1线程的概念-3线程的状态

    1线程的状态 线程终止的的原因: run()或call()方法执行完成,线程正常结束 线程抛出一个未捕获的Exception或Error 直接调用该线程的stop()方法来结束该线程--该方法容易导致 ...

  9. 廖雪峰Java11多线程编程-3高级concurrent包-4Concurrent集合

    Concurrent 用ReentrantLock+Condition实现Blocking Queue. Blocking Queue:当一个线程调用getTask()时,该方法内部可能让给线程进入等 ...

随机推荐

  1. Java-Class-C:org.springframework.web.client.RestTemplate

    ylbtech-Java-Class-C:org.springframework.web.client.RestTemplate 1.返回顶部 1. org.springframework.web.c ...

  2. JVM内核-原理、诊断与优化学习笔记(八):JAVA堆分析

    文章目录 内存溢出(OOM)的原因 在JVM中,有哪些内存区间? 堆溢出 永久区 Java栈溢出 直接内存溢出 小问题? MAT使用基础 柱状图显示 支配树 显示线程信息 显示堆总体信息,比如消耗最大 ...

  3. 29-Ubuntu-远程管理命令-03-SSH工作方式简介

    在Linux中SSH是非常重要的工具,通过SSH客户端可以连接到运行了SSH服务器的远程机器上. 1.SSH客户端是一种使用Secure Shell(SSH)协议连接到远程计算机的软件程序. 2.SS ...

  4. 最接近神的人_NOI导刊2010提高(02)

    题目描述 破解了符文之语,小FF开启了通往地下的道路.当他走到最底层时,发现正前方有一扇巨石门,门上雕刻着一幅古代人进行某种活动的图案.而石门上方用古代文写着“神的殿堂”.小FF猜想里面应该就有王室的 ...

  5. oracle中的decode函数的使用

    含义解释:decode(条件,值1,返回值1,值2,返回值2,...值n,返回值n,缺省值) 该函数的含义如下:IF 条件=值1 THEN RETURN(翻译值1)ELSIF 条件=值2 THEN R ...

  6. JDBC_数据库连接池工具类

    //定义一个类JDBCUtils public class JDBCUtils { //1.定义成员方法 private static DataSource ds; //2.提供静态代码块 stati ...

  7. ie8以下不兼容h5新标签的解决方法

    HTML5新添了一些语义化标签,他们能让代码语义化更直观易懂,有利于SEO优化.但是此HTML5新标签在IE6/IE7/IE8上并不能识别,需要进行JavaScript处理. 解决思路就是用js创建h ...

  8. Hadoop的局限与不足

  9. Flask框架图

  10. Windows where

    WHERE [/R dir] [/Q] [/F] [/T] pattern... 描述:    显示符合搜索模式的文件位置.在默认情况下,搜索是在当前目录和 PATH    环境变量指定的路径中执行的 ...