为什么要使用同步锁?

因为当使用多线程同时访问一个变量或对象时,如果这些线程中即有读又有写操作时,会造成导致变量或对象的状态出现混乱。例如:一个银行账户被A/B两个线程同时操作,A线程、B线程同时开始操作:A线程存款100,B线程取款100,此时就会出现账户存款100,然后查询存储结果为0,B取款失败,但是查询余额为100。

而同步锁出现的目的就是为了解决多线程安全问题。

上边的举例对应的代码如下:

银行类:

/**
* 银行类
* */
public class Bank {
private int money = 0; /**
* 存款
*/
public void deposit(int money) {
this.money += money;
System.out.println(Thread.currentThread().getName() + ":存款" + money + ",賬戶餘額:" + this.money);
} /**
* 取款
*/
public void withdrawal(int money) {
if (this.money - money < 0) {
System.out.println(Thread.currentThread().getName() + ":余额不足");
return;
}
this.money -= money;
System.out.println(Thread.currentThread().getName() + ":取款" + money + ",賬戶餘額:" + this.money);
} /**
* 查詢賬戶餘額
*/
public void look() {
System.out.println(Thread.currentThread().getName() + ":查詢賬戶餘額為:" + this.money);
}
}

取款、存款A、B线程模拟:

public class LockTest {
public static void main(String[] args) {
final Bank bank = new Bank(); Thread threadA = new Thread(new Runnable() {
public void run() {
while (true) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
bank.deposit(100);
bank.look(); System.out.println("-----------------------");
}
}
}, "thread-a"); Thread threadB = new Thread(new Runnable() {
public void run() {
while (true) {
System.out.println("-----------------------");
bank.withdrawal(100);
bank.look(); try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}, "thread-b"); threadB.start();
threadA.start();
}
}

此时打印的结果为:

-----------------------
thread-b:余额不足
thread-b:查詢賬戶餘額為:0
-----------------------
thread-a:存款100,賬戶餘額:100
thread-a:查詢賬戶餘額為:0
thread-b:取款100,賬戶餘額:0
thread-b:查詢賬戶餘額為:0
-----------------------

同步代码块:

注意:同步是一种高开销的操作,因此应该尽量减少同步的内容。通常没有必要同步整个方法,使用synchronized代码块同步关键代码即可。

/**
* 银行类
* */
class Bank {
private int money = 0; /**
* 存款
*/
public void deposit(int money) {
synchronized (this) {
this.money += money;
System.out.println(Thread.currentThread().getName() + ":存款" + money + ",賬戶餘額:" + this.money);
}
} /**
* 取款
*/
public void withdrawal(int money) {
synchronized (this) {
if (this.money - money < 0) {
System.out.println(Thread.currentThread().getName() + ":余额不足");
return;
}
this.money -= money;
System.out.println(Thread.currentThread().getName() + ":取款" + money + ",賬戶餘額:" + this.money);
}
} /**
* 查詢賬戶餘額
*/
public void look() {
synchronized (this) {
System.out.println(Thread.currentThread().getName() + ":查詢賬戶餘額為:" + this.money);
}
}
}

测试:

-----------------------
thread-b:余额不足
thread-b:查詢賬戶餘額為:0
thread-a:存款100,賬戶餘額:100
thread-a:查詢賬戶餘額為:100
-----------------------
-----------------------
thread-b:取款100,賬戶餘額:0
thread-b:查詢賬戶餘額為:0
thread-a:存款100,賬戶餘額:100
thread-a:查詢賬戶餘額為:100
-----------------------
-----------------------
thread-b:取款100,賬戶餘額:0
thread-b:查詢賬戶餘額為:0
thread-a:存款100,賬戶餘額:100
thread-a:查詢賬戶餘額為:100

同步方法:

同步方法就是使用synchronized关键字修饰某个方法,这个方法就是同步方法。这个同步方法(非static方法)无须显式指定同步监视器,同步方法的同步监视器是this,也就是调用该方法的对象。通过同步方法可以非常方便的实现线程安全的类,线程安全的类有如下特征:

每个线程调用该对象的任意synchronized方法之后,都能得到正确的结果;
每个线程调用该对象的任意synchronized方法之后,该对象状态依然能保持合理状态。
/**
* 银行类
*/
class Bank {
private int money = 0; /**
* 存款
*/
public synchronized void deposit(int money) {
this.money += money;
System.out.println(Thread.currentThread().getName() + ":存款" + money + ",賬戶餘額:" + this.money);
} /**
* 取款
*/
public synchronized void withdrawal(int money) {
if (this.money - money < 0) {
System.out.println(Thread.currentThread().getName() + ":余额不足");
return;
}
this.money -= money;
System.out.println(Thread.currentThread().getName() + ":取款" + money + ",賬戶餘額:" + this.money);
} /**
* 查詢賬戶餘額
*/
public synchronized void look() {
System.out.println(Thread.currentThread().getName() + ":查詢賬戶餘額為:" + this.money);
}
}

测试:

-----------------------
thread-b:余额不足
thread-b:查詢賬戶餘額為:0
-----------------------
thread-a:存款100,賬戶餘額:100
thread-b:取款100,賬戶餘額:0
thread-b:查詢賬戶餘額為:0
thread-a:查詢賬戶餘額為:0
-----------------------
-----------------------
thread-a:存款100,賬戶餘額:100
thread-a:查詢賬戶餘額為:100
-----------------------

重入锁Lock

JavaSE5.0中新增了一个java.util.concurrent包来支持同步。ReentrantLock类是可重入、互斥、实现了Lock接口的锁, 它与使用synchronized方法和快具有相同的基本行为和语义,并且扩展了其能力。
ReenreantLock类的常用方法有:

Lock lock=new ReentrantLock() ;// 创建一个ReentrantLock实例
lock.lock(); // 获得锁
lock.unlock();// 释放锁

注意:ReentrantLock()还有一个可以创建公平锁的构造方法,但由于能大幅度降低程序运行效率,不推荐使用。

/**
* 银行类
*/
class Bank {
private int money = 0;
private Lock lock=new ReentrantLock();
/**
* 存款
*/
public void deposit(int money) {
lock.lock();
try{
this.money += money;
System.out.println(Thread.currentThread().getName() + ":存款" + money + ",賬戶餘額:" + this.money);
}finally{
lock.unlock();
}
} /**
* 取款
*/
public void withdrawal(int money) {
lock.lock();
try{
if (this.money - money < 0) {
System.out.println(Thread.currentThread().getName() + ":余额不足");
return;
}
this.money -= money;
System.out.println(Thread.currentThread().getName() + ":取款" + money + ",賬戶餘額:" + this.money);
}finally{
lock.unlock();
}
} /**
* 查詢賬戶餘額
*/
public void look() {
lock.lock();
try{
System.out.println(Thread.currentThread().getName() + ":查詢賬戶餘額為:" + this.money);
}finally{
lock.unlock();
}
}
}

测试:

-----------------------
thread-b:余额不足
thread-b:查詢賬戶餘額為:0
thread-a:存款100,賬戶餘額:100
thread-a:查詢賬戶餘額為:100
-----------------------
-----------------------
thread-b:取款100,賬戶餘額:0
thread-b:查詢賬戶餘額為:0
thread-a:存款100,賬戶餘額:100
thread-a:查詢賬戶餘額為:100
-----------------------

Java-JUC(七):同步锁的几种方式的更多相关文章

  1. Java反射获取class对象的三种方式,反射创建对象的两种方式

    Java反射获取class对象的三种方式,反射创建对象的两种方式 1.获取Class对象 在 Java API 中,提供了获取 Class 类对象的三种方法: 第一种,使用 Class.forName ...

  2. Java - "JUC" ReentrantLock获取锁

    [Java并发编程实战]-----“J.U.C”:ReentrantLock之一简介 ReentrantLock介绍 ReentrantLock是一个可重入的互斥锁,又被称为“独占锁”. 顾名思义,R ...

  3. JUC——线程同步锁(Condition精准控制)

    在进行锁处理的时候还有一个接口:Condition,这个接口可以由用户来自己进行锁的对象创建. Condition的作用是对锁进行更精确的控制. Condition的await()方法相当于Objec ...

  4. JAVA并发,同步锁性能测试

    测试主要从运行时间差来体现,数据量越大,时间差越明显,例子如下: package com.xt.thinks21_2; /** * 同步锁性能测试 * * @author Administrator ...

  5. Java并发--线程间协作的两种方式:wait、notify、notifyAll和Condition

    在前面我们将了很多关于同步的问题,然而在现实中,需要线程之间的协作.比如说最经典的生产者-消费者模型:当队列满时,生产者需要等待队列有空间才能继续往里面放入商品,而在等待的期间内,生产者必须释放对临界 ...

  6. Java中 实现多线程成的三种方式(继承,实现,匿名内部类)

    ---------------------------------------------------------------------------------------------------- ...

  7. Java使用基本字节流OutputStream的四种方式对于数据复制(文本,音视频,图像等数据)

    //package 字符缓冲流bufferreaderDemo; import java.io.BufferedOutputStream; import java.io.FileInputStream ...

  8. Java字符流读写数据的两种方式

    第一种方式:逐个字符进行读写操作(代码注释以及详细内容空闲补充) package IODemo; import java.io.FileReader; import java.io.FileWrite ...

  9. java 学习笔记 读取配置文件的三种方式

    package com.itheima.servlet.cfg; import java.io.FileInputStream; import java.io.FileNotFoundExceptio ...

随机推荐

  1. 【转】Mapped Statements collection does not contain value for解决

    最近一直在弄springMVC+mybatis的整合,因为接触到这个框架之后发现这个框架确实要比ssh好得多所以我自己也在配置这个框架.但是在配置的过程中我遇到了一些问题,这些问题当我配置完成之后访问 ...

  2. spring-boot 速成(5) profile区分环境

    maven中的profile概念,在spring-boot中一样适合,只要约定以下几个规则即可: 一.不同环境的配置文件以"application-环境名.yml"命名 举个粟子: ...

  3. Java中static、final用法小结(转)

    一.final 1.final变量: 当你在类中定义变量时,在其前面加上final关键字,那便是说,这个变量一旦被初始化便不可改变,这里不可改变的意思对基本类型来说是其值不可变,而对于对象变量来说其引 ...

  4. LightOJ 1074 - Extended Traffic (SPFA)

    http://lightoj.com/volume_showproblem.php?problem=1074 1074 - Extended Traffic   PDF (English) Stati ...

  5. LPC43xx SGPIO Experimentation

    LPC4350 SGPIO Experimentation The NXP LPC43xx microcontrollers have an interesting, programmable ser ...

  6. how convert large HEX string to binary array ?

    how convert large HEX string to binary I have a string with 14 characters . This is a hex represanta ...

  7. SOC 与 ARM

    SOC是指片上系统,意思是一个芯片就构成一个包括了存储.CPU.甚至还有AD.UART等等其他资源的系统!而ARM只是CPU的一种,有的片上系统是51.nios.PIC.等等不一而是!特别是nios, ...

  8. [置顶] Linux下发布QT程序

    Linux下发布QT程序 概述 无论在windows下还是在linux下,可执行程序的运行都依赖于相关的运行库,我们需要将依赖的库找到放到特定的位置,让可执行文件能够找到.在不知道可执行文件依赖哪些库 ...

  9. 为什么我不再用 .NET 框架

    .NET平台很棒.真的很棒.直到它不再那么棒.我为什么不再用.NET?简单来说,它限制了我们选择的能力(对我来说很重要),转移了我们的注意力,使得我们向内认知它的安全性,替代了帮助我们认知外面广阔世界 ...

  10. 查看Oracle数据库名和实例名的命令

      查看数据库名 SQL> select name from v$database; NAME --------- ORCL SQL> desc v$database; 名称       ...