Java多线程学习——synchronized锁机制
Java在多线程中使用同步锁机制时,一定要注意锁对对象,下面的例子就是没锁对对象(每个线程使用一个被锁住的对象时,得先看该对象的被锁住部分是否有人在使用)
例子:两个人操作同一个银行账户,丈夫在ATM机上操作,妻子在银行柜台操作
账户类:账户里面有100万
public class Acount {
public int money=100;
}
ATM机类:里面存在一个Acount对象和要取的钱数,在takeMoney方法中加了synchronized 机制
public class ATM implements Runnable{
private Acount acount;
private int withdrawMoney;
public ATM(Acount acount, int withdrawMoney) {
this.acount = acount;
this.withdrawMoney = withdrawMoney;
}
@Override
public void run() {
takeMoney();
}
public synchronized void takeMoney(){ //取钱
if(acount.money<withdrawMoney){
return;
}
try {
Thread.sleep(200); //线程1进入后休眠,线程2仍可以进来,这样可能造成赤字
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"取出"+withdrawMoney+"万元");
acount.money-=withdrawMoney;
System.out.println("账户余额:"+acount.money);
}
}
客户类:创建一个唯一账户,两台ATM机(其中一台模拟柜台),分别给两个人使用
public class Customer {
public static void main(String[] args) {
Acount acount=new Acount();
ATM atm1=new ATM(acount,70); //丈夫在ATM机上操作账户,妻子在柜台操作账户
ATM atm2=new ATM(acount,80);
Thread husband=new Thread(atm1,"丈夫");
Thread wife=new Thread(atm2,"妻子");
husband.start();
wife.start();
}
}
运行结果:

可以看到加了synchronized 后仍然出现线程不安全。
分析:synchronized 机制一般用在被数据操作的对象中,而takeMoney方法是属于ATM机的方法,在此例子中,一共存在一个账户类,两个ATM机类,两个ATM机类去操作账户类,所以应该把账户类锁住。
修正:使用同步块机制锁住acount对象
public void takeMoney(){ //取钱
synchronized (acount){
if(acount.money<withdrawMoney){
return;
}
try {
Thread.sleep(200); //线程1进入后休眠,线程2仍可以进来,这样可能造成赤字
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"取出"+withdrawMoney+"万元");
acount.money-=withdrawMoney;
System.out.println("账户余额:"+acount.money);
}
}
例子2 Tickert类是线程类
public class Ticket implements Runnable{
private int ticker=100;
private boolean flag=true;
@Override
public void run() {
while(flag){
robTicket();
}
}
/**
* 抢票
*/
public void robTicket(){
if (ticker <= 0) {
flag = false;
return;
}
synchronized (this){ //多重验证机制,在这里检测这个对象的这部分代码是否被使用,如果有线程正在使用该对象的该部分代码,就等待
if (ticker <= 0) {
flag = false;
return;
}
try { //模拟网络延时
Thread.sleep(200);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"-->"+ticker--);
}
}
}
主类
public class RobTicket {
public static void main(String[] args) {
Runnable ticket=new Ticket();
//3个线程同时对一个ticket对象进行操作,准确来说是对ticket对象里面的ticket变量操作
Thread feizhu=new Thread(ticket,"飞猪");
Thread zhixing=new Thread(ticket,"智行");
Thread xiecheng=new Thread(ticket,"携程");
zhixing.start();
feizhu.start();
xiecheng.start();
}
}
在这个例子里面运用了多重的验证机制,保证了抢票重复和抢票出现负数的情况,如果不加synchronized 里面的if判断语句,仍然会出现线程不安全,因为其他线程可能并发的排队在同步块外面等候了,此时如果还剩一张票的话,当前线程抢完这最后一张票后其他线程仍然有机会抢票,这是不合理的,当然可以为整个方法上锁,但是性能会下降。
Java多线程学习——synchronized锁机制的更多相关文章
- java 多线程8 : synchronized锁机制 之 方法锁
脏读 一个常见的概念.在多线程中,难免会出现在多个线程中对同一个对象的实例变量或者全局静态变量进行并发访问的情况,如果不做正确的同步处理,那么产生的后果就是"脏读",也就是取到的数 ...
- java 多线程9 : synchronized锁机制 之 代码块锁
synchronized同步代码块 用关键字synchronized声明方法在某些情况下是有弊端的,比如A线程调用同步方法执行一个较长时间的任务,那么B线程必须等待比较长的时间.这种情况下可以尝试使用 ...
- java 多线程并发 synchronized 同步机制及方式
2. 锁机制 3. 并发 Excutor框架 4. 并发性与多线程介绍 1. synchronized 参考1. synchronized 分两种方式进行线程的同步:同步块.同步方法 1. 方法同步 ...
- Java多线程,对锁机制的进一步分析
1 可重入锁 可重入锁,也叫递归锁.它有两层含义,第一,当一个线程在外层函数得到可重入锁后,能直接递归地调用该函数,第二,同一线程在外层函数获得可重入锁后,内层函数可以直接获取该锁对应其它代码的控制权 ...
- Java多线程5:Synchronized锁机制
一.前言 在多线程中,有时会出现多个线程对同一个对象的变量进行并发访问的情形,如果不做正确的同步处理,那么产生的后果就是“脏读”,也就是获取到的数据其实是被修改过的. 二.引入Synchronized ...
- Java多线程学习(六)Lock锁的使用
系列文章传送门: Java多线程学习(二)synchronized关键字(1) Java多线程学习(二)synchronized关键字(2) Java多线程学习(三)volatile关键字 Java多 ...
- Java多线程学习(四)等待/通知(wait/notify)机制
转载请备注地址:https://blog.csdn.net/qq_34337272/article/details/79690279 系列文章传送门: Java多线程学习(一)Java多线程入门 Ja ...
- Java多线程学习(二)synchronized关键字(1)
转载请备注地址: https://blog.csdn.net/qq_34337272/article/details/79655194 Java多线程学习(二)将分为两篇文章介绍synchronize ...
- Java多线程学习(二)synchronized关键字(2)
转载请备注地址:https://blog.csdn.net/qq_34337272/article/details/79670775 系列文章传送门: Java多线程学习(一)Java多线程入门 Ja ...
随机推荐
- 【NOIP2012模拟10.25】单元格
题目 在一个R行C列的表格里,我们要选出3个不同的单元格.但要满足如下的两个条件: (1)选中的任意两个单元格都不在同一行. (2)选中的任意两个单元格都不在同一列. 假设我们选中的单元格分别是:A, ...
- hibernate中save()、get()、load()、update()、saveorupdate()、merge()等方法
1.save()方法 直接传个user对象 session.save(user); 2.get()方法和load()方法 get(): 传id session.get(UserInfo. ...
- [算法]概率与期望DP
前言 前两节主要针对题目分析,没时间的珂以跳过. 初步 首先举一道简单.经典的好题: [Lightoj1038]Race to 1 Again 懒得单独写,安利一下DennyQi同学的博客:https ...
- 目标检测Object Detection概述(Tensorflow&Pytorch实现)
1999:SIFT 2001:Cascades 2003:Bag of Words 2005:HOG 2006:SPM/SURF/Region Covariance 2007:PASCAL VOC 2 ...
- 小程序的image图片显示
最近接触到了一点小程序的东西,跟我们平常的HTML还真不太一样,这里我先大概讲一下图片的显示问题, 小程序的图片用的是<image><image/>标签,他默认的大小是宽300 ...
- logstash之Input插件
1:stdin标准输入和stdout标准输出 首先执行命令: bin/logstash -e 'input { stdin { } } output { stdout { codec => ...
- sqli-labs(24)
0x1 前言 SQL注入一般分为两类:一阶SQL注入(普通SQL注入),二阶SQL注入 .二次注入不是注入两次的意思,请不要混淆 0x2 什么是二阶注入 二阶注入是指客通过构造数据的形式,在浏览器或者 ...
- 初识REST
RESTful介绍: REST与技术无关,代表的是一种软件架构风格,REST是Representational State Transfer的简称,中文翻译为“表征状态转移”或“表现层状态转化”. 1 ...
- Base table or view not found
项目 代码分细致 改为Logic, Model, Controller ,View 四个模块 $model=D("Index",'Logic'); $res=$model-> ...
- 在aspx页面的checkbox取值验证
在做项目的时候遇到了一个选择性的问题,之前都可以用$("#id").checked,但是不知道为什么现在不可以了,只能if($(this).is(":checked&qu ...