二十二、同步方法
作者:互联网
同步方法:
由于我们可以通过private关键字来保证数据对象只能被方法访问,所以我们只需要针对方法提出一套机制,这套机制就是synchronized关键字,它包括两种用法:synchronized方法和synchronized块.
- 同步方法:public synchronized void method(int args){}
synchronized方法控制对“对象”的访问,每个对象对应一把锁,每个synchronized方法都必须获得调用该方法的对象的锁才能执行,否则线程会阻塞,方法一旦执行,就独占该锁,直到该方法返回才释放锁,后面被阻塞的线程才能获得这个锁,继续执行。
- 缺陷:若将一个大的方法申明为synchronized将会影响效率
同步块:synchronized (Obj ){ }
Obj称之为同步监视器
- Obj可以是任何对象﹐但是推荐使用共享资源作为同步监视器
- 同步方法中无需指定同步监视器﹐因为同步方法的同步监视器就是this ,就是这个对象本身,或者是class[反射]
同步监视器的执行过程
- 第一个线程访问,锁定同步监视器﹐执行其中代码
- 第二个线程访问,发现同步监视器被锁定,无法访问
- 第一个线程访问完毕,解锁同步监视器
- 第二个线程访问,发现同步监视器没有锁,然后锁定并访问
PS:锁的对象就是变化的对象
1. 买票
public class UnSafeBuyTicket {
public static void main(String[] args) {
BuyTicket buyTicket = new BuyTicket();
new Thread(buyTicket, "张三").start();
new Thread(buyTicket, "李四").start();
new Thread(buyTicket, "王五").start();
}
}
class BuyTicket implements Runnable {
private int ticket = 30;
@Override
public void run() {
synchronized ((Object) ticket) {
while (true) {
if (ticket > 0) {
System.out.println(Thread.currentThread().getName() + "买到的票号为:" + ticket--);
} else {
break;
}
}
}
}
}
2. 取钱
public class UnSafeBank {
public static void main(String[] args) {
Withdraw withdraw = new Withdraw(80);
Thread t1 = new Thread(withdraw, "他");
Thread t2 = new Thread(withdraw, "她");
t1.start();
t2.start();
}
}
class Withdraw implements Runnable {
//余额
private int balance = 100;
//取出
private int take;
public Withdraw(int take) {
this.take = take;
}
@Override
public void run() {
synchronized ((Object) balance) {
if (balance - take > 0) {
try {
if (Thread.currentThread().getName().equals("他")) {
Thread.sleep(1000);
}
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
balance = balance - take;
System.out.println(Thread.currentThread().getName() + "取出:" + take + ",剩余:" + balance);
} else {
System.out.println("余额不足");
}
}
}
}
3. 列表
public class UnSafeList {
public static void main(String[] args) throws InterruptedException {
List list = new ArrayList();
for (int i = 0; i < 100000; i++) {
new Thread(() -> {
synchronized (list) {
list.add(Thread.currentThread().getName());
}
}).start();
}
//让main线程等待上面的子线程执行完成
Thread.sleep(1000);
System.out.println(list.size());
}
}
结果:
标签:二十二,同步,Thread,synchronized,监视器,new,方法,public 来源: https://www.cnblogs.com/epiphany8/p/16272145.html