进阶Java编程(3)线程的同步与死锁
线程的同步与死锁
1,同步问题引出
在多线程的处理之中,可以利用Runnable描述多个线程操作的资源,而Thread描述每一个线程对象,对于当多个线程访问统一资源的时候如果处理不当就会产生数据的错误操作。
①同步问题的引出
下面编写一个买票程序,将创建若干个线程的对象实现卖票处理操作。
·范例:实现卖票操作
class MyThread implements Runnable{
private int ticket=10;//总票数为10张
@Override
public void run() {
while (true){
if(this.ticket>0){
try{
Thread.sleep(100);
}catch(InterruptedException e){
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"卖票,ticket="+this.ticket--);
}else {
System.out.println("****** 票已经卖光了 *********");
break;
}
}
}
}
public class Main {
public static void main(String[] args) {
MyThread myThread=new MyThread();
new Thread(myThread,"票贩子A").start();
new Thread(myThread,"票贩子B").start();
new Thread(myThread,"票贩子C").start();
new Thread(myThread,"票贩子D").start();
new Thread(myThread,"票贩子E").start();
}
}
票贩子A卖票,ticket=0
****** 票已经卖光了 *********
票贩子B卖票,ticket=1
****** 票已经卖光了 *********
票贩子C卖票,ticket=-2
****** 票已经卖光了 *********
票贩子D卖票,ticket=-3
****** 票已经卖光了 *********
****** 票已经卖光了 *********
此时的程序将创建三个线程对象,并且这三个线程对象将进行5章票的出售。此时的程序在进行卖票处理的时候并没有任何的问题(这是假象),下面可以模拟一下卖票过程中的延迟操作。
2,线程同步处理
经过分析之后已经可以确定同步问题所产生的主要原因了,那么下面就需要进行进行同步问题的关键是锁,指的是当某一个线程执行操作的时候,其他线程外面等待;
如果想要在程序之中实现这把锁的功能,就可以使用synchronized关键字来实现,利用此关键字可以定义同步方法或同步代码块,在同步代码块的操作里面的代码只允许一个线程执行。
①利用同步代码块进行处理:
synchronized (同步操作){
同步代码操作;
}
·范例:利用同步代码块解决数据同步访问问题一般要进行同步对象处理的时候可以采用当前对象this进行同步。
class MyThread implements Runnable{
private int ticket=10;//总票数为10张
@Override
public void run() {
while (true){
synchronized(this){//每一次只允许一个线程进行执行
if(this.ticket>0){
try{
Thread.sleep(100);
}catch(InterruptedException e){
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"卖票,ticket="+this.ticket--);
}else {
System.out.println("* 票已经卖光了 *");
break;
}
}
}
}
}
public class Main {
public static void main(String[] args) {
MyThread myThread=new MyThread();
new Thread(myThread,"票贩子A").start();
new Thread(myThread,"票贩子B").start();
new Thread(myThread,"票贩子C").start();
}
}
②利用同步方法解决:只需要在方法定义上使用synchronized关键字即可加入同步处理之后,程序的整体的执行的性能下降了。同步实际上会造成性能的降低。
class MyThread implements Runnable{
private int ticket=10;//总票数为10张
public synchronized boolean sale(){
if(this.ticket>0){
try{
Thread.sleep(100);
}catch(InterruptedException e){
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"卖票,ticket="+this.ticket--);
return true;
}else {
System.out.println("****** 票已经卖光了 *********");
return false;
}
}
@Override
public void run() {
while (this.sale()){
;//空语句
}
}
}
public class Main {
public static void main(String[] args) {
MyThread myThread=new MyThread();
new Thread(myThread,"票贩子A").start();
new Thread(myThread,"票贩子B").start();
new Thread(myThread,"票贩子C").start();
}
}
在日后学习Java类库的时候会发现,系统中使用的类上使用的许多同步处理采用的都是同步方法。
注意:同步会造成性能的下降,并发才会高性能。
3,线程死锁
死锁是在进行多线程同步的处理之中有可能产生的一种问题,所谓的死锁指的是若干个线程彼此互相等待的状态。
·范例:通过一个简单的代码观察死锁
public class DeadLock implements Runnable{
private Jian jj =new Jian();
private XiaoQiang xq=new XiaoQiang(); @Override
public void run() {
jj.say(xq);
}
public DeadLock(){//构造方法
new Thread(this).start();
xq.say(jj);
}
public static void main(String[] args) {
new DeadLock();
}
} class Jian{
public synchronized void say(XiaoQiang xq){
System.out.println("阿建说:此路是我开留下买路财");
xq.get();
}
public synchronized void get(){
System.out.println("阿建说:获得过路费买饭吃");
}
}
class XiaoQiang{
public synchronized void say(Jian jj){
System.out.println("小强说:先过路在给钱");
jj.get();
}
public synchronized void get(){
System.out.println("小强说:通过道路去上班");
}
}
若干线程访问同一资源时一定要进行同步处理,而过多的同步会造成死锁。现在死锁造成的主要原因是因为彼此都在互相等待着,等待着对方先让出资源。死锁是开发中的一种不确定的状态,有的时候代码处理不当则会不定期出现死锁,这是属于正常开发中的调试问题。
进阶Java编程(3)线程的同步与死锁的更多相关文章
- 进阶Java编程(2)线程常用操作方法
线程常用操作方法 多线程的主要操作方法都在Thread类中定义的. 1,线程的命名和取得 多线程的运行状态是不确定的,那么在程序的开发之中为了可以获取到一些需要使用到的线程就只能依靠线程的名字来进行操 ...
- Java并发编程:线程的同步
Java并发编程:线程的同步 */--> code {color: #FF0000} pre.src {background-color: #002b36; color: #839496;} J ...
- Java多线程之线程的同步
Java多线程之线程的同步 实际开发中我们也经常提到说线程安全问题,那么什么是线程安全问题呢? 线程不安全就是说在多线程编程中出现了错误情况,由于系统的线程调度具有一定的随机性,当使用多个线程来访问同 ...
- 菜鸡的Java笔记 - java 线程的同步与死锁 (同步 synchronization,死锁 deadlock)
线程的同步与死锁 (同步 synchronization,死锁 deadlock) 多线程的操作方法 1.线程同步的产生与解决 2.死锁的问题 ...
- java多线程之线程的同步与锁定(转)
一.同步问题提出 线程的同步是为了防止多个线程访问一个数据对象时,对数据造成的破坏. 例如:两个线程ThreadA.ThreadB都操作同一个对象Foo对象,并修改Foo对象上的数据. publicc ...
- 进阶Java编程(1)多线程编程
Java多线程编程 1,进程与线程 在Java语言里面最大的特点是支持多线程的开发(也是为数不多支持多线程的编程语言Golang.Clojure方言.Elixir),所以在整个的Java技术学习里面, ...
- 进阶Java编程(13)反射与Annotation
1,反射取得Annotation信息 从JDK1.5之后Java提供了Annotation技术支持,这种技术为项目的编写带来了新的模型,而后经过了十年的发展,Annotation的技术得到了非常广泛的 ...
- 进阶Java编程(10)反射与简单Java类
1,传统属性自动赋值弊端 简单Java类主要由属性构成,并且提供有setter与getter类,同时简单Java类最大的特征就是通过对象保存相应的类属性的内容.但是如果使用传统的简单Java类开发,那 ...
- 进阶Java编程(5)基础类库
Java基础类库 1,StringBuffer类 String类是在所有项目开发之中一定会使用到的一个功能类,并且这个类拥有如下的特点: ①每一个字符串的常量都属于一个String类的匿名对象,并且不 ...
随机推荐
- Qt学习之如何启动和终止一个线程
先来给出每个文件的相关代码然后再加以分析 //*************dialog.h**************// #ifndef DIALOG_H #define DIALOG_H #incl ...
- 解决 无法启动此程序,因为计算机中丢失opencv_world341.dll。请尝试重新安装改程序已解决此问题
在运行OpenCV程序时报错:“无法启动此程序,因为计算机中丢失opencv_world341.dll.请尝试重新安装改程序已解决此问题”. 解决方法 我的bin目录是 D:\opencv\build ...
- Android 关于 CountDownTimer onTick() 倒计时不准确问题源码分析
一.问题 CountDownTimer 使用比较简单,设置 5 秒的倒计时,间隔为 1 秒. final String TAG = "CountDownTimer"; * , ) ...
- Nginx优化之日志优化,URL访问控制,防盗链,及站点文件目录优化
Nginx日志相关优化与安全 日志切割脚本如下: #!/bin #日志切割脚本 Date=`date +%Y%m%d` Bdir="/usr/local/nginx" Nginxl ...
- C# 批处理制作静默安装程序包
使用批处理+WinRAR制作静默安装程序包 @echo 安装完窗口会自动关闭!!! @echo off start /wait Lync.exe /Install /Silent start /wai ...
- 小D课堂 - 新版本微服务springcloud+Docker教程_3-06 服务注册和发现之Eureka Client搭建商品服务实战
笔记 6.服务注册和发现之Eureka Client搭建商品服务实战 简介:搭建用商品服务,并将服务注册到注册中心 1.创建一个SpirngBoot应用,增加服务注册和发现依赖 2.模 ...
- 关于XMind软件文件格式的一些思考
1,安装XMind软件,看见可以导入Markdown文件 2,因此新建了测试.md文件,代码格式为左图,显示效果为右图. 3,导入到XMind中显示为下图 4,也就是XMind中子标题对应着Markd ...
- shell脚本:DNS自检脚本
host.txt中信息,已配置DNS正反解. bigdata-hive-tidb005.bjthq.vivo.lan 10.20.94.5 bigdata-hive-tidb004.bjthq.viv ...
- Scrapy+redis实现分布式爬虫
概述 什么是分布式爬虫 需要搭建一个由n台电脑组成的机群,然后在每一台电脑中执行同一组程序,让其对同一网络资源进行联合且分布的数据爬取. 原生Scrapy无法实现分布式的原因 原生Scrapy中调度器 ...
- 深入解析d3弦图
记得上次看d3应该是1年前的事情了,当时还一边看一边写了d3(v5.7)的一个学习笔记:https://www.cnblogs.com/eco-just/tag/d3/ 后来转战three.js就没继 ...