三、线程同步之Sysnchronized关键字
线程同步
问题引入
观察一面一段小程序:
public class Main {private static int amount = 0;public static void main(String[] args) {System.out.println(++amount);new MyThread("thread1").start();new MyThread("thread2").start();}private static void calc(String tag){++amount;try {Thread.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(tag + "==>" + amount);}static class MyThread extends Thread {private String name;public MyThread(String name) {super();this.name = name;}@Overridepublic void run() {super.run();calc(name);}}}
运行结果:

上面的代码MyThread中先将amount累加1;再睡1S,打印数据。很显然thread1睡眠之后被打断,thread2被执行才会出现这样的情况。
系统对线程的调度是有一定随机性的。当多线程操作资源时,才会出现线程安全的问题。Java 提供了一系列的方案来解决这个问题,下面将一一说明。
Synchronized关键字
同步方法块
synchronized(obj){...//此处是同步代码块}
当线程执行此段代码时必须持有锁,否则不能执行,上面代码修改如下所示即可恢复正常:
//对象锁private static Object obj = new Object();private static void calc(String tag) {//执行此代码必须持有此锁,没有锁只能等待此锁的使用线程释放锁synchronized (obj) {++amount;try {Thread.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(tag + "==>" + amount);}}

同步方法
此关键字还可以直接修饰方法,被修饰的方法为同步方法。同步方法的使用也必须持有锁,此时锁定的是this(当前对象),
private synchronized static void calc(String tag) {// 执行此代码必须持有此锁,没有锁只能等待此锁的使用线程释放锁++amount;try {Thread.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(tag + "==>" + amount);}

释放锁
synchronized修饰的代码块或者同步方法没有显性的释放锁的方法。遇到以下几种情况会释放锁。
- 使用synchronized关键字,必须等待当前线程执行完同步代码块或者同步方法,锁被释放。
- 遇到return、break终止了当前代码块或者同步方法,锁被释放。
- 遇到未处理的异常导致的崩溃,当前锁被释放。
- 线程调用wait方法,释被放锁。
下面两点不会释放锁,需要注意:
- 程序调用Thread.sleep()、Thread.yield(),当前线程不会释放锁。
- 其他线程调用suspend方法,将此方法挂起,当前线程不会释放锁。同时应该避免使用suspend和resume方法控制线程。
访问权限总结
访问权限原则:当前线程持有某个锁时,其他线程无法访问被同一个锁锁定的方法或者代码块;与是否在一个类中,是否是静态无关,只与锁是否被释放有关。
常见的问题如下:
- 一个类中,有多个方法,当前线程持有某锁,其他线程无法访问此锁锁定的其他同步方法,但是可以访问其他非同步方法
- 一个类中有多个代码块或同步方法,当前线程持有某所,其他线程可以访问其他锁锁定的同步代码块

浙公网安备 33010602011771号