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中StopWatch的使用示例详解

《Java中StopWatch的使用示例详解》stopWatch是org.springframework.util包下的一个工具类,使用它可直观的输出代码执行耗时,以及执行时间百分比,这篇文章主要介绍... 目录stopWatch 是org.springframework.util 包下的一个工具类,使用它

Java进行文件格式校验的方案详解

《Java进行文件格式校验的方案详解》这篇文章主要为大家详细介绍了Java中进行文件格式校验的相关方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、背景异常现象原因排查用户的无心之过二、解决方案Magandroidic Number判断主流检测库对比Tika的使用区分zip

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

Java使用Curator进行ZooKeeper操作的详细教程

《Java使用Curator进行ZooKeeper操作的详细教程》ApacheCurator是一个基于ZooKeeper的Java客户端库,它极大地简化了使用ZooKeeper的开发工作,在分布式系统... 目录1、简述2、核心功能2.1 CuratorFramework2.2 Recipes3、示例实践3

Springboot处理跨域的实现方式(附Demo)

《Springboot处理跨域的实现方式(附Demo)》:本文主要介绍Springboot处理跨域的实现方式(附Demo),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不... 目录Springboot处理跨域的方式1. 基本知识2. @CrossOrigin3. 全局跨域设置4.

springboot security使用jwt认证方式

《springbootsecurity使用jwt认证方式》:本文主要介绍springbootsecurity使用jwt认证方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录前言代码示例依赖定义mapper定义用户信息的实体beansecurity相关的类提供登录接口测试提供一

Spring Boot 3.4.3 基于 Spring WebFlux 实现 SSE 功能(代码示例)

《SpringBoot3.4.3基于SpringWebFlux实现SSE功能(代码示例)》SpringBoot3.4.3结合SpringWebFlux实现SSE功能,为实时数据推送提供... 目录1. SSE 简介1.1 什么是 SSE?1.2 SSE 的优点1.3 适用场景2. Spring WebFlu

基于SpringBoot实现文件秒传功能

《基于SpringBoot实现文件秒传功能》在开发Web应用时,文件上传是一个常见需求,然而,当用户需要上传大文件或相同文件多次时,会造成带宽浪费和服务器存储冗余,此时可以使用文件秒传技术通过识别重复... 目录前言文件秒传原理代码实现1. 创建项目基础结构2. 创建上传存储代码3. 创建Result类4.

Java利用JSONPath操作JSON数据的技术指南

《Java利用JSONPath操作JSON数据的技术指南》JSONPath是一种强大的工具,用于查询和操作JSON数据,类似于SQL的语法,它为处理复杂的JSON数据结构提供了简单且高效... 目录1、简述2、什么是 jsONPath?3、Java 示例3.1 基本查询3.2 过滤查询3.3 递归搜索3.4

Tomcat版本与Java版本的关系及说明

《Tomcat版本与Java版本的关系及说明》:本文主要介绍Tomcat版本与Java版本的关系及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Tomcat版本与Java版本的关系Tomcat历史版本对应的Java版本Tomcat支持哪些版本的pythonJ