Java 线程同步属于 Java 多线程与并发编程的外围点,须要重点把握,上面我就来详解 Java 线程同步的 4 种次要的实现形式 @mikechen
目录
- 什么是线程同步
-
线程同步的几种形式
- 1、应用 synchronized 关键字
- 2. 应用 ReentrantLock
- 3. 应用原子变量实现线程同步
- 4.ThreadLocal 实现线程同步
什么是线程同步
当应用多个线程来拜访同一个数据时,将会导致数据不精确,相互之间产生抵触,非常容易呈现线程平安问题,如下图所示:
比方多个线程都在操作同一数据,都打算批改商品库存,这样就会导致数据不统一的问题。
线程同步的实在意思,其实是“排队”:几个线程之间要排队,一个一个对共享资源进行操作,而不是同时进行操作。
所以咱们用同步机制来解决这些问题,退出同步锁以防止在该线程没有实现操作之前,被其余线程的调用,从而保障了该变量的唯一性和准确性。
线程同步的几种形式
1、应用 synchronized 关键字
这种形式比拟灵便, 润饰一个代码块,被润饰的代码块称为同步语句块。
其作用的范畴是大括号 {} 括起来的代码,作用的对象是调用这个代码块的对象,如下格局:
synchronized(对象) {
// 失去对象的锁,能力操作同步代码
须要被同步代码;
}
![]()
通常没有必要同步整个办法,应用 synchronized 代码块同步要害代码即可。
具体的示例如下:
public class SynchronizedThread {
class Bank {
private int account = 200;
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();}
}
![]()
2. 应用 ReentrantLock
ReentrantLock 类是可重入、互斥、实现了 Lock 接口的锁,它与应用 synchronized 办法具备雷同的根本行为和语义,并且扩大了其能力。
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();
}
}
}
![]()
synchronized 与 Lock 的比照
ReentrantLock 是显示锁, 手动开启和敞开锁,别忘记敞开锁;
synchronized 是隐式锁,出了作用域主动开释;
ReentrantLock 只有代码块锁,synchronized 有代码块锁和办法锁;
应用 ReentrantLock 锁,JVM 将破费较少的工夫来调度线程,线程更好, 并且具备更好的扩展性(提供更多的子类);
优先应用程序:
ReentrantLock> synchronized 同步代码块 > synchronized 同步办法
3. 应用原子变量实现线程同步
为了实现线程同步,咱们将应用原子变量 (Atomic* 结尾的)来实现。
比方典型代表:AtomicInteger 类存在于 java.util.concurrent.atomic 中, 该类示意反对原子操作的整数,采纳 getAndIncrement 办法以原子办法将以后的值递减。
具体示例如下:
private AtomicInteger account = new AtomicInteger(100);
public AtomicInteger getAccount() {return account;}
public void save(int money) {account.addAndGet(money);
}
![]()
4.ThreadLocal 实现线程同步
如果应用 ThreadLocal 治理变量,则每一个应用该变量的线程都取得该变量的正本,正本之间互相独立,这样每一个线程都能够随便批改本人的变量正本,而不会对其余线程产生影响,从而实现线程同步。
具体代码示例如下:
// 只改 Bank 类,其余代码与上同
public class Bank{
// 创立一个线程本地变量 ThreadLocal
private static ThreadLocal<Integer> account = new ThreadLocal<Integer>(){
@Override
// 返回以后线程的 "初始值"
protected Integer initialValue(){return 100;}
};
public void save(int money){
// 设置线程正本中的值
account.set(account.get()+money);
}
public int getAccount(){
// 返回线程正本中的值
return account.get();}
}
![]()
以上
作者简介
陈睿 |mikechen,10 年 + 大厂架构教训,《BAT 架构技术 500 期》系列文章作者,专一于互联网架构技术。
浏览 mikechen 的互联网架构更多技术文章合集
Java 并发 |JVM|MySQL|Spring|Redis| 分布式 | 高并发
关注「mikechen 的互联网架构」公众号,回复 【架构】 支付《Java 进阶架构思维导图 &Java 进阶架构文章合集》