使用synchronized关键字来修饰某个方法,该方法被称为同步方法。
对于synchronized修饰的实例方法(非static方法),无需显示指定同步监视器,同步方法的同步监视器是this,也就是调用该方法的对象。
通过同步方法可以非常方便的实现线程安全的类,线程安全的类具有如下特征:
该类的对象可以被多个线程对象安全访问
每个线程调用该对象的任意方法后将得到正确的结果
每个线程调用该对象的任意方法后,该对象状态依然保持合理状态
public class Account {
private String account;//账号
private double balance;//余额
public Account() {
super();
}
public Account(String account, double balance) {
super();
this.account = account;
this.balance = balance;
}
public synchronized void drawMoney(double drawMoney) {
if(balance >= drawMoney) {
System.out.println(Thread.currentThread().getName() + "取钱成功!吐出钞票:" + drawMoney);
try {
Thread.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
balance -= drawMoney;
System.out.println("\t余额为:" + balance);
}else {
System.out.println(Thread.currentThread().getName() + "取钱失败!余额不足");
}
}
public String getAccount() {
return account;
}
public void setAccount(String account) {
this.account = account;
}
public double getBalance() {
return balance;
}
public void setBalance(double balance) {
this.balance = balance;
}
}
public class DrawThread extends Thread{
private Account account;//模拟账户
private double drawMoney;//当前取钱线程想要取的钱数
public DrawThread(String name , Account account, double drawMoney) {
super(name);
this.account = account;
this.drawMoney = drawMoney;
}
public void run() {
account.drawMoney(drawMoney);
}
public static void main(String[] args) {
Account account = new Account("123456" , 1000);
new DrawThread("A", account, 800).start();
new DrawThread("B", account, 800).start();
}
}
使用synchronized关键自修饰drawMoney方法,该方法为同步方法,该同步方法的同步监视器是this,因此,对同一account而言,任意时刻只有一个线程能对account对象锁定,这样可以保障线程安全。
synchronized关键字只能修饰方法,和代码块。
线程安全是以降低程序的运行效率作为代价的,为了减少线程安全带来的负面影响
不要对线程安全类的所有方法都进行同步,对共享资源同步就行了
如果可变类有两种运行环境,单线程环境/多线程环境,则应该为可变类提供两种版本,线程安全/线程不安全版本,单线程环境中提供线程不安全版本保证性能,多线程环境中使用线程安全版本
释放同步监听器的锁定:任何线程进入同步代码块,同步监听器,必须先获得对同步监视器的锁定,何时释放呢?
同步方法/同步代码块执行完了,当前线程,释放同步锁。
同步方法/同步代码块执行过程中,遇到break,return终止执行,释放同步监视器
同步方法/同步代码块执行过程中,出现了未处理的error 和 exception , 导致异常结束 , 释放同步监视器
同步方法/同步代码块执行过程中,程序执行了同步监视器对象的wait()方法,则当前线程暂停,并释放同步监视器