synchronized关键字的意义

为了防止代码块受并发访问的干扰,Java语言提供一个 synchronized关键字达 到这一目的,并且 Java SE 5.0引入了 ReentrantLock 类。synchronized 关键字自动提供一个 锁以及相关的“ 条件”, 对于大多数需要显式锁的情况, 这是很便利的。

synchronized关键字的作用

synchronized关键字是对某个对象加锁

举个例子,如以下代码段:

1
2
3
4
5
6
7
private Object o = new Object();
public void m() {
synchronized(o) { //任何线程要执行下面的代码,必须先拿到o的锁
count--;
System.out.println(Thread.currentThread().getName() + " count = " + count);
}
}

还有另一种用法

1
2
3
4
5
6
public void m() {   
synchronized(this) { //任何线程要执行下面的代码,必须先拿到this的锁
count--;
System.out.println(Thread.currentThread().getName() + " count = " + count);
}
}

这种方式等同于如下代码段:

1
2
3
4
public synchronized void m() { //等同于在方法的代码执行时要synchronized(this)   
count--;
System.out.println(Thread.currentThread().getName() + " count = " + count);
}

注意:当synchronized关键字修饰方法时,锁定的是this,也就是该对象本身;当synchronized关键字修饰的是静态方法或者静态变量时,锁定的是对象.class

synchronized关键字的注意点

  1. 同步方法和非同步方法可以同时调用,原因是调用非同步方法时不需要拿到锁

  2. 在应用场景中,对写方法(set)加锁,对读方法(get)不加锁,容易产生脏读问题,原因是,在①线程写入操作还没有完成时,此时②线程读取数据,此时读到的数据不是实际写入的数据

  3. 一个同步方法可以调用另一个同步方法,一个线程已经拥有某个对象的锁,再次申请的时候,仍然会得到该对象的锁,也就是说synchronized获得的锁是可重入的

  4. 可重入锁的实现机制是:每一个锁关联一个线程持有者和计数器,当计数器为 0 时表示该锁没有被任何线程持有,那么任何线程都可能获得该锁而调用相应的方法;当某一线程请求成功后,JVM会记下锁的持有线程,并且将计数器置为 1;此时其它线程请求该锁,则必须等待;而该持有锁的线程如果再次请求这个锁,就可以再次拿到这个锁,同时计数器会递增;当线程退出同步代码块时,计数器会递减,如果计数器为 0,则释放该锁。

  5. 子类可以调用父类的同步方法,锁住的是子类对象

  6. 在同步方法执行的过程中,如果出现异常,默认情况下锁会被释放,若不想被释放,则需要catch异常

  7. 同步代码块中的语句越少越好,采用细粒度的锁,可以使线程争用的时间变短,从而提高效率

  8. 锁定的某个对象o,如果o的属性发生改变,不影响锁的使用,但是如果o变成另一个对象,则锁定的对象发生改变,如以下代码:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    public class T {
    Object o = new Object();
    void m() {
    synchronized(o) {
    while(true) {
    try {
    TimeUnit.SECONDS.sleep(1);
    } catch (InterruptedException e) {
    e.printStackTrace();
    }
    System.out.println(Thread.currentThread().getName());
    }
    }
    }
    public static void main(String[] args) {
    T t = new T();
    //启动第一个线程
    new Thread(t::m, "t1").start();

    try {
    TimeUnit.SECONDS.sleep(3);
    } catch (InterruptedException e) {
    e.printStackTrace();
    }
    //创建第二个线程
    Thread t2 = new Thread(t::m, "t2");
    //锁对象发生改变,所以t2线程得以执行,如果注释掉这句话,线程2将永远得不到执行机会
    t.o = new Object();
    t2.start();

    }
    }

    输出:
    t1
    t1
    t1
    t2
    t1
    t2
    t1
    t2
    t1
    t2
    t1

    Process finished with exit code -1
  9. 不要用字符串常量作为锁定对象,如下:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    public class T {
    String s1 = "Hello";
    String s2 = "Hello";
    void m1() {
    synchronized(s1) {

    }
    }
    void m2() {
    synchronized(s2) {

    }
    }
    }

    m1和m2其实锁定的是同一个对象;此时有可能发生诡异的现象,比如你用到的一个类库,在该类库中代码中也锁定了字符串“Hello”,这时就有可能发生死锁阻塞,因为你的程序和你用到的类库在不经意间使用了同一把锁

synchronized的原理

同步代码块

monitorenter :

每个对象有一个监视器锁(monitor)。当monitor被占用时就会处于锁定状态,线程执行monitorenter指令时尝试获取monitor的所有权,过程如下:

1、如果monitor的进入数为0,则该线程进入monitor,然后将进入数设置为1,该线程即为monitor的所有者。

2、如果线程已经占有该monitor,只是重新进入,则进入monitor的进入数加1.

3.如果其他线程已经占用了monitor,则该线程进入阻塞状态,直到monitor的进入数为0,再重新尝试获取monitor的所有权。

monitorexit:

  执行monitorexit的线程必须是objectref所对应的monitor的所有者。

​ 指令执行时,monitor的进入数减1,如果减1后进入数为0,那线程退出monitor,不再是这个monitor的所有者。其他被这个monitor阻塞的线程可以尝试去获取这个 monitor 的所有权。

  Synchronized的语义底层是通过一个monitor的对象来完成,其实wait/notify等方法也依赖于monitor对象,这就是为什么只有在同步的块或者方法中才能调用wait/notify等方法,否则会抛出java.lang.IllegalMonitorStateException的异常的原因。

同步方法

从反编译的结果来看,方法的同步并没有通过指令monitorenter和monitorexit来完成(理论上其实也可以通过这两条指令来实现),不过相对于普通方法,其常量池中多了ACC_SYNCHRONIZED标示符。JVM就是根据该标示符来实现方法的同步的:当方法调用时,调用指令将会检查方法的 ACC_SYNCHRONIZED 访问标志是否被设置,如果设置了,执行线程将先获取monitor,获取成功之后才能执行方法体,方法执行完后再释放monitor。在方法执行期间,其他任何线程都无法再获得同一个monitor对象。 其实本质上没有区别,只是方法的同步是一种隐式的方式来实现,无需通过字节码来完成。

参考资料:

https://www.cnblogs.com/huangyin/p/6586469.html