Java多线程同步锁的理解
java主要通过synchronized的关键字来实现的。让我们从一个买票程序说起吧。
package com.day04; /**
*
* @author Administrator 问题描述:使用多线程的方式来模拟多个窗口买票
*
*/
public class SaleWindow implements Runnable { // 初始化票数10
private int ticket = 10; @Override
public void run() {
// 获取线程的名称,比如Thread-0,并将它截掉Thread-取0这个数字标识,为了构造下面卖票窗口名称
int threadNum = Integer.parseInt(Thread.currentThread().getName().substring(7));
String saleWindowName = "销售窗口" + threadNum;
// 开始买票
while (true) {
if (ticket > 0) {
// 这里为了演示出线程不同步的问题,让线程睡眠一段时间,延时)
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(saleWindowName + " 卖 出 了 " + ticket-- + " 号 票 !");
} else {
break;
}
}
} public static void main(String[] args) {
// 创建了销售窗口对象
SaleWindow sw = new SaleWindow();
// 启动线程,让第一个窗口开始买票
new Thread(sw).start();
// 启动线程,让第二个窗口开始买票
new Thread(sw).start();
// 启动线程,让第三个窗口开始买票
new Thread(sw).start(); } }
运行结果如下所示:
销售窗口2 卖 出 了 10 号 票 !
销售窗口1 卖 出 了 8 号 票 !
销售窗口0 卖 出 了 9 号 票 !
销售窗口2 卖 出 了 7 号 票 !
销售窗口1 卖 出 了 6 号 票 !
销售窗口0 卖 出 了 5 号 票 !
销售窗口2 卖 出 了 4 号 票 !
销售窗口1 卖 出 了 3 号 票 !
销售窗口0 卖 出 了 2 号 票 !
销售窗口2 卖 出 了 1 号 票 !
销售窗口1 卖 出 了 0 号 票 !《-----
销售窗口0 卖 出 了 -1 号 票 !《------
可以看到我们的程序出来了问题,上面打红色箭头所示,竟然卖出了0号票和-1号票了。
让我们画个图来分析一下如下所示:
通过以上分析,不难得出,造成问题原因,是因为同步操作问题。
那我们如何确定哪些是同步操作(或者有同步问题)?
1.明确哪些代码是多线程成运行的代码(run方法中的代码)
2.明确那些是共享数据(ticket票数)
3.明确多线程运行代码中那些语句是操作共享数据(System.out.println(saleWindowName + " 卖 出 了 " + ticket-- + " 号 票 !");)
接下来我们就可以通过Java给我们提供的synchroized关键字使用同步锁来解决以上的问题
package com.day04; /**
* @author Administrator 问题描述:使用多线程的方式来模拟多个窗口买票
*/
public class SaleWindow implements Runnable { // 初始化票数10
private int ticket = 10;
//线程的锁
private Object lock; @Override
public void run() {
// 获取线程的名称,比如Thread-0,并将它截掉Thread-取0这个数字标识,为了构造下面卖票窗口名称
int threadNum = Integer.parseInt(Thread.currentThread().getName().substring(7));
String saleWindowName = "销售窗口" + threadNum;
// 开始买票
while (true) {
//加上synchronized,并加入对象锁,new一个任意对象即可,我们这里使用Object来解决同步问题,注意这里必须是公用同一个锁lock
synchronized (lock) { if (ticket > 0) {
// 这里为了演示出线程不同步的问题,让线程睡眠一段时间,延时)
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(saleWindowName + " 卖 出 了 " + ticket-- + " 号 票 !");
} else {
break;
}
}
}
} public static void main(String[] args) {
// 创建了销售窗口对象
SaleWindow sw = new SaleWindow();
// 启动线程,让第一个窗口开始买票
new Thread(sw).start();
// 启动线程,让第二个窗口开始买票
new Thread(sw).start();
// 启动线程,让第三个窗口开始买票
new Thread(sw).start(); } }
运行结果如下所示:
销售窗口1 卖 出 了 10 号 票 !
销售窗口2 卖 出 了 9 号 票 !
销售窗口0 卖 出 了 8 号 票 !
销售窗口0 卖 出 了 7 号 票 !
销售窗口2 卖 出 了 6 号 票 !
销售窗口1 卖 出 了 5 号 票 !
销售窗口2 卖 出 了 4 号 票 !
销售窗口0 卖 出 了 3 号 票 !
销售窗口0 卖 出 了 2 号 票 !
销售窗口2 卖 出 了 1 号 票 !
这样就有效的解决了同步的问题。
同样我们也可将上面的操作共享数据的同步操作抽取出来,单独封装成一个同步方法,只需要在方法上面的返回值前面加上synchronized关键字即可,这样可以更方面理解和阅读,优化后代码如下。
package com.day04; /**
*
* @author Administrator 问题描述:使用多线程的方式来模拟多个窗口买票
*
*/
public class SaleWindow implements Runnable { // 初始化票数10
private int ticket = 10; @Override
public void run() { // 开始买票
while (true) {
// 当没有票了结束
if (!saleSuccess()) {
break;
}
}
} public synchronized boolean saleSuccess() {
// 获取线程的名称,比如Thread-0,并将它截掉Thread-取0这个数字标识,为了构造下面卖票窗口名称
int threadNum = Integer.parseInt(Thread.currentThread().getName().substring(7));
String saleWindowName = "销售窗口" + threadNum;
if (ticket > 0) {
// 这里为了演示出线程不同步的问题,让线程睡眠一段时间,延时)
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(saleWindowName + " 卖 出 了 " + ticket-- + " 号 票 !");
return true;
} else {
return false;
}
} public static void main(String[] args) {
// 创建了销售窗口对象
SaleWindow sw = new SaleWindow();
// 启动线程,让第一个窗口开始买票
new Thread(sw).start();
// 启动线程,让第二个窗口开始买票
new Thread(sw).start();
// 启动线程,让第三个窗口开始买票
new Thread(sw).start(); } }
销售窗口1 卖 出 了 10 号 票 !销售窗口2 卖 出 了 9 号 票 !销售窗口0 卖 出 了 8 号 票 !销售窗口0 卖 出 了 7 号 票 !销售窗口2 卖 出 了 6 号 票 !销售窗口1 卖 出 了 5 号 票 !销售窗口2 卖 出 了 4 号 票 !销售窗口0 卖 出 了 3 号 票 !销售窗口0 卖 出 了 2 号 票 !销售窗口2 卖 出 了 1 号 票 !
现在又有一个问题出现了,public synchronized boolean saleSuccess()该同步函数用的是哪一个锁?
我们猜想可能用的是this这个对象锁,如果我们让线程一个执行带有sychronized的同步方法,一个执行带有this对象的sychronized同步代码块的方法,如果能够得到正确的结果,不出现同步问题,即论证正确,反之,如果还是出现同步问题即用的不是this这个对象锁。代码如下:
package com.day04; /**
*
* @author Administrator 问题描述:使用多线程的方式来模拟多个窗口买票
*
*/
public class SaleWindow implements Runnable { // 初始化票数10
private int ticket = 10; @Override
public void run() {
// 获取当前线程的序号从0开始
int threadNum = Integer.parseInt(Thread.currentThread().getName().substring(7));
// 偶数线程执行该方法
if ((threadNum + 1) % 2 == 0) {
while (true) {
synchronized (this) {
String saleWindowName = "奇数销售窗口" + threadNum;
if (ticket > 0) {
// 这里为了演示出线程不同步的问题,让线程睡眠一段时间,延时)
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(saleWindowName + " 卖 出 了 " + ticket-- + " 号 票 !");
} else {
break;
}
}
}
} else {
// 奇数线程执行该方法
// 开始买票
while (true) {
// 当没有票了结束
if (!saleSuccess()) {
break;
}
}
} } public synchronized boolean saleSuccess() {
// 获取线程的名称,比如Thread-0,并将它截掉Thread-取0这个数字标识,为了构造下面卖票窗口名称
int threadNum = Integer.parseInt(Thread.currentThread().getName().substring(7));
String saleWindowName = "偶数销售窗口" + threadNum;
if (ticket > 0) {
// 这里为了演示出线程不同步的问题,让线程睡眠一段时间,延时)
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(saleWindowName + " 卖 出 了 " + ticket-- + " 号 票 !");
return true;
} else {
return false;
}
} public static void main(String[] args) {
// 创建了销售窗口对象
SaleWindow sw = new SaleWindow();
// 启动线程,让第一个窗口开始买票
new Thread(sw).start();
// 启动线程,让第二个窗口开始买票
new Thread(sw).start();
// 启动线程,让第三个窗口开始买票
new Thread(sw).start(); } }
运行结果如下:
偶数销售窗口0 卖 出 了 10 号 票 !
偶数销售窗口0 卖 出 了 9 号 票 !
偶数销售窗口2 卖 出 了 8 号 票 !
奇数销售窗口1 卖 出 了 7 号 票 !
偶数销售窗口2 卖 出 了 6 号 票 !
偶数销售窗口2 卖 出 了 5 号 票 !
偶数销售窗口2 卖 出 了 4 号 票 !
偶数销售窗口2 卖 出 了 3 号 票 !
偶数销售窗口2 卖 出 了 2 号 票 !
偶数销售窗口0 卖 出 了 1 号 票 !
由上面的接口即可论证同步方法使用对的对象锁是this。
同样的加入我们的将我们的共享数据ticket改成静态的,并将同步方法也改成静态,它用的是那个对象锁?
我们猜想是本类的class对象这个锁即(SaleWindow.class)这个对象锁。同理如果我们让线程一个执行带有sychronized的静态同步方法,一个执行带有本类的class对象这个锁即(SaleWindow.class)的sychronized同步代码块的方法,如果能够得到正确的结果,不出现同步问题,即论证正确。反之,如果还是出现同步问题,即说明静态同步方法使用的锁不是本类的class对象这个锁即(SaleWindow.class)这个对象锁。代码如下:
package com.day04; /**
*
* @author Administrator 问题描述:使用多线程的方式来模拟多个窗口买票
*
*/
public class SaleWindow implements Runnable { // 初始化票数10
private static int ticket = 10; @Override
public void run() {
// 获取当前线程的序号从0开始
int threadNum = Integer.parseInt(Thread.currentThread().getName().substring(7));
// 偶数线程执行该方法
if ((threadNum + 1) % 2 == 0) {
while (true) {
synchronized (SaleWindow.class) {
String saleWindowName = "奇数销售窗口" + threadNum;
if (ticket > 0) {
// 这里为了演示出线程不同步的问题,让线程睡眠一段时间,延时)
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(saleWindowName + " 卖 出 了 " + ticket-- + " 号 票 !");
} else {
break;
}
}
}
} else {
// 奇数线程执行该方法
// 开始买票
while (true) {
// 当没有票了结束
if (!saleSuccess()) {
break;
}
}
} } public synchronized static boolean saleSuccess() {
// 获取线程的名称,比如Thread-0,并将它截掉Thread-取0这个数字标识,为了构造下面卖票窗口名称
int threadNum = Integer.parseInt(Thread.currentThread().getName().substring(7));
String saleWindowName = "偶数销售窗口" + threadNum;
if (ticket > 0) {
// 这里为了演示出线程不同步的问题,让线程睡眠一段时间,延时)
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(saleWindowName + " 卖 出 了 " + ticket-- + " 号 票 !");
return true;
} else {
return false;
}
} public static void main(String[] args) {
// 创建了销售窗口对象
SaleWindow sw = new SaleWindow();
// 启动线程,让第一个窗口开始买票
new Thread(sw).start();
// 启动线程,让第二个窗口开始买票
new Thread(sw).start();
// 启动线程,让第三个窗口开始买票
new Thread(sw).start(); } }
运行结果如下:
偶数销售窗口0 卖 出 了 10 号 票 !
偶数销售窗口0 卖 出 了 9 号 票 !
偶数销售窗口2 卖 出 了 8 号 票 !
奇数销售窗口1 卖 出 了 7 号 票 !
偶数销售窗口2 卖 出 了 6 号 票 !
偶数销售窗口2 卖 出 了 5 号 票 !
偶数销售窗口2 卖 出 了 4 号 票 !
偶数销售窗口2 卖 出 了 3 号 票 !
偶数销售窗口2 卖 出 了 2 号 票 !
偶数销售窗口0 卖 出 了 1 号 票 !
由上面的接口即可论证同步方法使用对的对象锁是本类的class对象这个锁即(SaleWindow.class)这个对象锁。
public synchronized boolean saleSuccess()
Java多线程同步锁的理解的更多相关文章
- Java 多线程:锁(二)
Java 多线程:锁(二) 作者:Grey 原文地址: 博客园:Java 多线程:锁(二) CSDN:Java 多线程:锁(二) AtomicLong VS LongAddr VS Synchroni ...
- Java多线程同步问题的探究
一.线程的先来后到——问题的提出:为什么要有多线程同步?Java多线程同步的机制是什么? http://www.blogjava.net/zhangwei217245/archive/2010/03/ ...
- 转:关于JAVA多线程同步
转:http://lanvis.blog.163.com/blog/static/26982162009798422547/ 因为需要,最近关注了一下JAVA多线程同步问题.JAVA多线程同步主要依赖 ...
- Java多线程-同步:synchronized 和线程通信:生产者消费者模式
大家伙周末愉快,小乐又来给大家献上技术大餐.上次是说到了Java多线程的创建和状态|乐字节,接下来,我们再来接着说Java多线程-同步:synchronized 和线程通信:生产者消费者模式. 一.同 ...
- 第十五章、Python多线程同步锁,死锁和递归锁
目录 第十五章.Python多线程同步锁,死锁和递归锁 1. 引子: 2.同步锁 3.死锁 引子: 4.递归锁RLock 原理: 不多说,放代码 总结: 5. 大总结 第十五章.Python多线程同步 ...
- JAVA多线程与锁机制
JAVA多线程与锁机制 1 关于Synchronized和lock synchronized是Java的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码 ...
- Java 多线程:锁(一)
Java 多线程:锁(一) 作者:Grey 原文地址: 博客园:Java 多线程:锁(一) CSDN:Java 多线程:锁(一) CAS 比较与交换的意思 举个例子,内存有个值是 3,如果用 Java ...
- Java 多线程:锁(三)
Java 多线程:锁(三) 作者:Grey 原文地址: 博客园:Java 多线程:锁(三) CSDN:Java 多线程:锁(三) StampedLock StampedLock其实是对读写锁的一种改进 ...
- java多线程同步
一篇好文:java多线程机制同步原则 概括起来说,Java 多线程同步机制主要包含如下几点:1:如果一个类包含一个或几个同步方法,那么由此类生成的每一个对象都配备一个队列用来容纳那些等待执行同步的线程 ...
随机推荐
- Uni2D 入门
原地址:http://blog.csdn.net/kakashi8841/article/details/17599505
- C语言字符串操作总结大全(超具体)
1)字符串操作 strcpy(p, p1) 复制字符串 strncpy(p, p1, n) 复制指定长度字符串 strcat(p, p1) 附加字符串 strncat(p, p1, n) 附加指定长度 ...
- 基于React实现的【绿色版电子书阅读器】,支持离线下载
代码地址如下:http://www.demodashi.com/demo/12052.html MyReader 绿色版电子书阅读器 在线地址:http://myreader.linxins.com ...
- <转>windows下编译lua源码
因为之前一直使用 lua for windows 来搭建lua的使用环境,但是最新的 lua for windows 还没有lua5.2,我又想用这个版本的lua,所以被逼无奈只能自己编一下lua源码 ...
- Codeforces 476C Dreamoon and Sums (水
题目链接:点击打开链接 题意: 给定a,b 对于一个数x.若x是nice number,则满足(x/b)/(x%b) == [1,a](即结果在1-a之间) 问: 输出一个数表示 全部nice num ...
- 微信支付和微信支付通知基于sdk的说明
前提是,有微信服务号(必须开通了支付功能,也就是说有了商户后台) (注意商户后台 安全目录 的设置,不然即使你写的没错误,也调用不成功) 公众号h5页面写法: (购物车提交--我们上一步已经生成了订 ...
- 经典图算法Java代码实践:BFS,DFS以及几种最短路径算法
public class City { String name; int id; static int idCounter = 0; public City(String name) { this.n ...
- Atitit.变量的定义 获取 储存 物理结构 基本类型简化 隐式转换 类型推导 与底层原理 attilaxDSL
Atitit.变量的定义 获取 储存 物理结构 基本类型简化 隐式转换 类型推导 与底层原理 attilaxDSL 1.1. $ 美元字符, php 黑头1 1.2. 默认变量的范围和声明:1 1.3 ...
- 详解TCP建立连接全过程
TCP是因特网中的传输层协议,使用三次握手协议建立连接,下面是TCP建立连接的全过程. 上图画出了TCP建立连接的过程.假定主机A是TCP客户端,B是服务端.最初两端的TCP进程都处于CLOSED状态 ...
- centos 7 下安装haproxy
1 haproxy 下载 从如下目录下载haproxy:http://www.haproxy.org/download/1.7/src/haproxy-1.7.1.tar.gz 2 haproxy 安 ...