Java多线程的同步与死锁

2024-08-28 16:08
文章标签 java 多线程 同步 死锁

本文主要是介绍Java多线程的同步与死锁,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 知识要点:

了解线程同步的作用

了解同步代码块以及同步方法的使用

了解死锁的产生

在多线程开发中,同步与死锁是至关重要的需要掌握以下几点:

1、哪里需要同步

2、如何实现同步

3、以及实现同步之后会有哪些副作用。

问题的引出

以卖火车票为例,不管在哪个地方买火车票,最终一趟列车的车票数量是固定的,如果将多个售票点理解为线程的话,则所有线程应该共同拥有同一份票数。
class MyThread implements Runnable{private int ticket = 5 ;	// 假设一共有5张票public void run(){for(int i=0;i<100;i++){if(ticket>0){	// 还有票try{Thread.sleep(300) ;	// 加入延迟}catch(InterruptedException e){e.printStackTrace() ;}System.out.println("卖票:ticket = " + ticket-- );}}}
};
public class SyncDemo01{public static void main(String args[]){MyThread mt = new MyThread() ;	// 定义线程对象Thread t1 = new Thread(mt) ;	// 定义Thread对象Thread t2 = new Thread(mt) ;	// 定义Thread对象Thread t3 = new Thread(mt) ;	// 定义Thread对象t1.start() ;t2.start() ;t3.start() ;}
};

从程序的执行结果看,卖出的票数成了负数,程序代码出了问题。
程序的问题:
从运行的结果可以发现,程序中加入了延迟操作,所以在运行的最后出现了负数的情况,那么为什么会出现这种情况呢?
从上面的操作代码可以发现对于票数的操作步骤如下:
1、判断票数是否大于0,大于0则表示还有票可以卖。
2、如果票数大于0,则卖票出去。
但是,在上面的操作代码中,第1步和第2步之间加入了延迟操作,那么一个线程就有可能在还没有对票数进行减操作之前其他线程就已经将票数减少了,这样一来就会出现票数为负的情况。
问题的解决:
如果想解决这样的问题就必须使用同步,所谓的同步就是指多个操作在同一个时间段内只能有一个线程执行,其他线程要等待此线程完成之后才能继续执行。

使用同步问题

想解决资源共享的同步操作问题,可以使用同步代码块和同步方法两种方式完成。
已知代码块分为四种:
普通代码块:是直接定义在方法之中的。
构造块:是直接定义在类中的,优先于构造方法执行,重复调用。
静态块:是使用static关键字声明的,优先于构造块执行,只执行一次。
同步代码块:使用synchronized关键字声明的代码块,称为同步代码块。

同步代码块:
在代码块上加上"synchronized"关键字的话,则此代码块就称为同步代码块。
同步代码块格式:
synchronized(同步对象){
     需要同步的代码
}

同步的时候必须指明同步的对象,一般情况下会将当前对象作为同步对象,使用this表示。
class MyThread implements Runnable{private int ticket = 5 ;	// 假设一共有5张票public void run(){for(int i=0;i<100;i++){synchronized(this){	// 要对当前对象进行同步if(ticket>0){	// 还有票try{Thread.sleep(300) ;	// 加入延迟}catch(InterruptedException e){e.printStackTrace() ;}System.out.println("卖票:ticket = " + ticket-- );}}}}
};
public class SyncDemo02{public static void main(String args[]){MyThread mt = new MyThread() ;	// 定义线程对象Thread t1 = new Thread(mt) ;	// 定义Thread对象Thread t2 = new Thread(mt) ;	// 定义Thread对象Thread t3 = new Thread(mt) ;	// 定义Thread对象t1.start() ;t2.start() ;t3.start() ;}
};


从运行的结果可以发现,加入了同步操作,所以不会产生负数的情况,但是程序的执行效率明显降低很多。

同步方法:
除了可以将需要的代码设置成同步代码块之外,也可以使用synchronized关键字将一个方法声明成同步方法。
同步方法定义格式:
synchronized 方法返回值 方法名称(参数列表){   }
具体代码如下:
class MyThread implements Runnable{private int ticket = 5 ;	// 假设一共有5张票public void run(){for(int i=0;i<100;i++){this.sale() ;	// 调用同步方法}}public synchronized void sale(){	// 声明同步方法if(ticket>0){	// 还有票try{Thread.sleep(300) ;	// 加入延迟}catch(InterruptedException e){e.printStackTrace() ;}System.out.println("卖票:ticket = " + ticket-- );}}
};
public class SyncDemo03{public static void main(String args[]){MyThread mt = new MyThread() ;	// 定义线程对象Thread t1 = new Thread(mt) ;	// 定义Thread对象Thread t2 = new Thread(mt) ;	// 定义Thread对象Thread t3 = new Thread(mt) ;	// 定义Thread对象t1.start() ;t2.start() ;t3.start() ;}
};

  所谓的同步就是同样的操作在同一时间段内只能由一个线程进行访问。同步通常用于多个线程同时共享同一资源时。 
过多的同步会导致线程死锁。
具体代码如下所示:
class Zhangsan{	// 定义张三类public void say(){System.out.println("张三对李四说:“你给我画,我就把书给你。”") ;}public void get(){System.out.println("张三得到画了。") ;}
};
class Lisi{	// 定义李四类public void say(){System.out.println("李四对张三说:“你给我书,我就把画给你”") ;}public void get(){System.out.println("李四得到书了。") ;}
};
public class ThreadDeadLock implements Runnable{private static Zhangsan zs = new Zhangsan() ;		// 实例化static型对象private static Lisi ls = new Lisi() ;		// 实例化static型对象private boolean flag = false ;	// 声明标志位,判断那个先说话public void run(){	// 覆写run()方法if(flag){synchronized(zs){	// 同步张三zs.say() ;try{Thread.sleep(500) ;}catch(InterruptedException e){e.printStackTrace() ;}synchronized(ls){zs.get() ;}}}else{synchronized(ls){ls.say() ;try{Thread.sleep(500) ;}catch(InterruptedException e){e.printStackTrace() ;}synchronized(zs){ls.get() ;}}}}public static void main(String args[]){ThreadDeadLock t1 = new ThreadDeadLock() ;		// 控制张三ThreadDeadLock t2 = new ThreadDeadLock() ;		// 控制李四t1.flag = true ;t2.flag = false ;Thread thA = new Thread(t1) ;Thread thB = new Thread(t2) ;thA.start() ;thB.start() ;}
};

可以发现线程会因为相互等待而造成死锁。

这篇关于Java多线程的同步与死锁的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1115243

相关文章

使用Java将各种数据写入Excel表格的操作示例

《使用Java将各种数据写入Excel表格的操作示例》在数据处理与管理领域,Excel凭借其强大的功能和广泛的应用,成为了数据存储与展示的重要工具,在Java开发过程中,常常需要将不同类型的数据,本文... 目录前言安装免费Java库1. 写入文本、或数值到 Excel单元格2. 写入数组到 Excel表格

Java并发编程之如何优雅关闭钩子Shutdown Hook

《Java并发编程之如何优雅关闭钩子ShutdownHook》这篇文章主要为大家详细介绍了Java如何实现优雅关闭钩子ShutdownHook,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 目录关闭钩子简介关闭钩子应用场景数据库连接实战演示使用关闭钩子的注意事项开源框架中的关闭钩子机制1.

Maven中引入 springboot 相关依赖的方式(最新推荐)

《Maven中引入springboot相关依赖的方式(最新推荐)》:本文主要介绍Maven中引入springboot相关依赖的方式(最新推荐),本文给大家介绍的非常详细,对大家的学习或工作具有... 目录Maven中引入 springboot 相关依赖的方式1. 不使用版本管理(不推荐)2、使用版本管理(推

Java 中的 @SneakyThrows 注解使用方法(简化异常处理的利与弊)

《Java中的@SneakyThrows注解使用方法(简化异常处理的利与弊)》为了简化异常处理,Lombok提供了一个强大的注解@SneakyThrows,本文将详细介绍@SneakyThro... 目录1. @SneakyThrows 简介 1.1 什么是 Lombok?2. @SneakyThrows

在 Spring Boot 中实现异常处理最佳实践

《在SpringBoot中实现异常处理最佳实践》本文介绍如何在SpringBoot中实现异常处理,涵盖核心概念、实现方法、与先前查询的集成、性能分析、常见问题和最佳实践,感兴趣的朋友一起看看吧... 目录一、Spring Boot 异常处理的背景与核心概念1.1 为什么需要异常处理?1.2 Spring B

如何在 Spring Boot 中实现 FreeMarker 模板

《如何在SpringBoot中实现FreeMarker模板》FreeMarker是一种功能强大、轻量级的模板引擎,用于在Java应用中生成动态文本输出(如HTML、XML、邮件内容等),本文... 目录什么是 FreeMarker 模板?在 Spring Boot 中实现 FreeMarker 模板1. 环

SpringMVC 通过ajax 前后端数据交互的实现方法

《SpringMVC通过ajax前后端数据交互的实现方法》:本文主要介绍SpringMVC通过ajax前后端数据交互的实现方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价... 在前端的开发过程中,经常在html页面通过AJAX进行前后端数据的交互,SpringMVC的controll

Java中的工具类命名方法

《Java中的工具类命名方法》:本文主要介绍Java中的工具类究竟如何命名,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录Java中的工具类究竟如何命名?先来几个例子几种命名方式的比较到底如何命名 ?总结Java中的工具类究竟如何命名?先来几个例子JD

Java Stream流使用案例深入详解

《JavaStream流使用案例深入详解》:本文主要介绍JavaStream流使用案例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录前言1. Lambda1.1 语法1.2 没参数只有一条语句或者多条语句1.3 一个参数只有一条语句或者多

Spring Security自定义身份认证的实现方法

《SpringSecurity自定义身份认证的实现方法》:本文主要介绍SpringSecurity自定义身份认证的实现方法,下面对SpringSecurity的这三种自定义身份认证进行详细讲解,... 目录1.内存身份认证(1)创建配置类(2)验证内存身份认证2.JDBC身份认证(1)数据准备 (2)配置依