专注Java教育14年 全国咨询/投诉热线:444-1124-454
赢咖4LOGO图
始于2009,口口相传的Java黄埔军校
首页 hot资讯 线程同步详解

线程同步详解

更新时间:2020-11-09 17:57:54 来源:赢咖4 浏览1221次


同步(Synchronization)在计算机语言中是指对在一个系统中所发生的事件之间进行协调,在时间上出现一致性与统一化的现象。而Java语言是允许多线程并发控制的,当多个线程同时操作一个可共享的资源变量时(如数据的增删改查),将会导致数据不准确,相互之间产生冲突,因此加入同步锁以避免在该线程没有完成操作之前,被其他线程的调用,从而保证了该变量的唯一性和准确性。本文我们就来和大家讲讲线程同步的相关知识。


1、线程同步方法

即有synchronized关键字修饰的方法。由于java的每个对象都有一个内置锁,当用此关键字修饰方法时,内置锁会保护整个方法。在调用该方法前,需要获得内置锁,否则就处于阻塞状态。

public synchronized void save(){}

注: synchronized关键字也可以修饰静态方法,此时如果调用该静态方法,将会锁住整个类


2、线程同步代码块

即有synchronized关键字修饰的语句块。被该关键字修饰的语句块会自动被加上内置锁,从而实现同步。

public class Bank {

private int count =0;//账户余额

//存钱

public void addMoney(int money){

synchronized (this) {

count +=money;

}

System.out.println(System.currentTimeMillis()+"存进:"+money);

}

//取钱

public void subMoney(int money){

synchronized (this) {

if(count-money < 0){

System.out.println("余额不足");

return;

}

count -=money;

}

System.out.println(+System.currentTimeMillis()+"取出:"+money);

}

//查询

public void lookMoney(){

System.out.println("账户余额:"+count);

}

}

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


3、使用特殊域变量(volatile)实现线程同步

1)volatile关键字为域变量的访问提供了一种免锁机制;

2)使用volatile修饰域相当于告诉虚拟机该域可能会被其他线程更新;

3)因此每次使用该域就要重新计算,而不是使用寄存器中的值;

4)volatile不会提供任何原子操作,它也不能用来修饰final类型的变量。

public class SynchronizedThread {

class Bank {

private volatile int account = 100;

public int getAccount() {

return account;

}

/**

* 用同步方法实现

*

* @param money

*/

public synchronized void save(int money) {

account += money;

}

/**

* 用同步代码块实现

*

* @param money

*/

public void save1(int money) {

synchronized (this) {

account += money;

}

}

}

class NewThread implements Runnable {

private Bank bank;

public NewThread(Bank bank) {

this.bank = bank;

}

@Override

public void run() {

for (int i = 0; i < 10; i++) {

// bank.save1(10);

bank.save(10);

System.out.println(i + "账户余额为:" +bank.getAccount());

}

}

}

/**

* 建立线程,调用内部类

*/

public void useThread() {

Bank bank = new Bank();

NewThread new_thread = new NewThread(bank);

System.out.println("线程1");

Thread thread1 = new Thread(new_thread);

thread1.start();

System.out.println("线程2");

Thread thread2 = new Thread(new_thread);

thread2.start();

}

public static void main(String[] args) {

SynchronizedThread st = new SynchronizedThread();

st.useThread();

}

多线程中的非同步问题主要出现在对域的读写上,如果让域自身避免这个问题,则就不需要修改操作该域的方法。用final域,有锁保护的域和volatile域可以避免非同步的问题。


4、使用重入锁(Lock)实现线程同步

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

ReentrantLock() : 创建一个ReentrantLock实例

lock() : 获得锁

unlock() : 释放锁

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

//只给出要修改的代码,其余代码与上同

class Bank {

private int account = 100;

//需要声明这个锁

private Lock lock = new ReentrantLock();

public int getAccount() {

return account;

}

//这里不再需要synchronized

public void save(int money) {

lock.lock();

try{

account += money;

}finally{

lock.unlock();

}

}


以上就是对线程同步的详细讲解,简而言之,线程同步就是为了防止多个线程同时访问同一个数据对象时,对数据造成破坏,也是保证多线程安全访问资源的一种保障。当然,线程同步也只是多线程知识里面的沧海一粟,想要学好多线程,学好Java,快来观看本站的Java基础教程吧!


提交申请后,顾问老师会电话与您沟通安排学习

免费课程推荐 >>
技术文档推荐 >>