Java多线程-synchronized的使用
Java的同步主要是通过synchronized和Lock来实现。
本文主要说的是Java关键字synchronized的使用。
相关概念
- 监视器
在Java中每个对象都有一个锁标记(monitor),也称监视器,多线程同时访问某个对象时,线程只有获取了该对象的锁才能访问。
在Java中,可以使用关键字synchronized来标记一个方法或者代码块。
注意
synchronized可修饰方法、代码块,但不可去修饰属性、构造方法。
使用(同步方法与同步代码块)
方法同步
1 | // 方法同步 |
synchronized用在方法上,则线程调用此方法时会获取该实例的对象锁,方法未结束之前其他线程只能等待。
同时其他线程也无法访问该实例的其他synchronized方法,因为一个对象只有一把锁,当一个线程获取了该对象的锁之后,其他线程就无法获取该对象锁。
同理,当一个线程访问该实例的synchronized方法时,其他线程能访问该对象的非synchronized方法,因为没使用到对象锁。
上面的前提是各线程调用的是同一对象实例,否则synchronized无效。
但若该synchronized方法为类方法,即方法修饰符为static,那么使用synchronized调用此方法的线程会获取该类的锁,只要该线程没有退出当前方法,其他线程也无法访问该方法。又若一个线程访问的是该对象的普通(非static)synchronized方法,另一个线程访问的是该对象(该类)的static synchronized方法,也不会发生互斥现象。因为它们一个获取的是对象锁,另一个是类锁,不冲突。
代码块同步
1 | public void test2() { |
而用在代码块上,当线程运行到该代码块时,就会获得object对象的对象锁,如果多线程使用同一个object对象,那么此时就会同步,使得其他线程无法同时访问该代码块。使用synchronized代码块只对需要的部分进行同步,效率也更高。
当object为this时,代表获取的是当前实例的锁。即
1 | public void test3() { |
最后,使用synchronized出现异常时,JVM会自动释放当前线程占用的锁,因此不会由于异常导致出现死锁现象。