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锁机制的更多相关文章

  1. java 多线程8 : synchronized锁机制 之 方法锁

    脏读 一个常见的概念.在多线程中,难免会出现在多个线程中对同一个对象的实例变量或者全局静态变量进行并发访问的情况,如果不做正确的同步处理,那么产生的后果就是"脏读",也就是取到的数 ...

  2. java 多线程9 : synchronized锁机制 之 代码块锁

    synchronized同步代码块 用关键字synchronized声明方法在某些情况下是有弊端的,比如A线程调用同步方法执行一个较长时间的任务,那么B线程必须等待比较长的时间.这种情况下可以尝试使用 ...

  3. java 多线程并发 synchronized 同步机制及方式

    2. 锁机制 3. 并发 Excutor框架 4. 并发性与多线程介绍 1. synchronized  参考1. synchronized 分两种方式进行线程的同步:同步块.同步方法 1. 方法同步 ...

  4. Java多线程,对锁机制的进一步分析

    1 可重入锁 可重入锁,也叫递归锁.它有两层含义,第一,当一个线程在外层函数得到可重入锁后,能直接递归地调用该函数,第二,同一线程在外层函数获得可重入锁后,内层函数可以直接获取该锁对应其它代码的控制权 ...

  5. Java多线程5:Synchronized锁机制

    一.前言 在多线程中,有时会出现多个线程对同一个对象的变量进行并发访问的情形,如果不做正确的同步处理,那么产生的后果就是“脏读”,也就是获取到的数据其实是被修改过的. 二.引入Synchronized ...

  6. Java多线程学习(六)Lock锁的使用

    系列文章传送门: Java多线程学习(二)synchronized关键字(1) Java多线程学习(二)synchronized关键字(2) Java多线程学习(三)volatile关键字 Java多 ...

  7. Java多线程学习(四)等待/通知(wait/notify)机制

    转载请备注地址:https://blog.csdn.net/qq_34337272/article/details/79690279 系列文章传送门: Java多线程学习(一)Java多线程入门 Ja ...

  8. Java多线程学习(二)synchronized关键字(1)

    转载请备注地址: https://blog.csdn.net/qq_34337272/article/details/79655194 Java多线程学习(二)将分为两篇文章介绍synchronize ...

  9. Java多线程学习(二)synchronized关键字(2)

    转载请备注地址:https://blog.csdn.net/qq_34337272/article/details/79670775 系列文章传送门: Java多线程学习(一)Java多线程入门 Ja ...

随机推荐

  1. Java基本的程序结构设计 大数操作

    大数操作 BigInteger 不可变的任意精度的整数.所有操作中,都以二进制补码形式表示 BigInteger(如 Java 的基本整数类型).BigInteger 提供所有 Java 的基本整数操 ...

  2. cve-2019-1609,Harbor任意管理员注册漏洞复现

    一.Harbor介绍 以Docker为代表的容器技术的出现,改变了传统的交付方式.通过把业务及其依赖的环境打包进Docker镜像,解决了开发环境和生产环境的差异问题,提升了业务交付的效率.如何高效地管 ...

  3. Python分布式+云计算

    参考: http://wiki.python.org/moin/ParallelProcessing http://wiki.python.org/moin/DistributedProgrammin ...

  4. 集合综合练习<二>

    package com.JiHeTotal; import java.util.Map; public class Student { int id; String name; Map<Stri ...

  5. CodeForces-598D(BFS,染色)

    链接: https://vjudge.net/problem/CodeForces-598D 题意: Igor is in the museum and he wants to see as many ...

  6. 《SaltStack技术入门与实践》—— Grains

    Grains 本章节参考<SaltStack技术入门与实践>,感谢该书作者: 刘继伟.沈灿.赵舜东 前几章我们已经了解SaltStack各个组件以及通过一个案例去熟悉它的各种应用,从这章开 ...

  7. 【leetcode】1217. Play with Chips

    题目如下: There are some chips, and the i-th chip is at position chips[i]. You can perform any of the tw ...

  8. Set数据结构

    1.生成Set数据结构 const s = new Set(); const set = new Set([1, 2, 3, 4, 4]); 以上如果打印set值: 2.特性 它类似于数组,但是成员的 ...

  9. 举例子说明ubuntu中remove,autoremove,purge区别

    转自:慎用 apt-get autoremove !   apt-get 提供了一个用于下载和安装软件包的简易命令行界面.卸载软件包主要有这3个命令 remove – 卸载软件包autoremove ...

  10. Java数据结构之排序---插入排序

    插入排序的基本介绍: 插入排序是对想要排序的序列以插入的方式寻找该元素的适当的位置,从而达到排序的目的. 插入排序的基本思想: 把n个待排序的元素看成一个有序表和一个无序表,开始时,有序表只有一个元素 ...