JDK5.0,Java提供了更强大的线程同步机制,通过显式定义同步锁对象来实现同步,同步锁使用Lock对象充当
java.util.concurrent.locks.Lock接口是控制多个线程对共享资源进行访问的工具,锁提供了对共享资源的独占访问,每次只能有一个线程对Lock对象加锁,线程开始访问共享资源之前应先获得Lock对象
ReentrantLock(可重入锁)类实现了Lock,它拥有与synchronized相同的并发性和内存语义,在实现线程安全的控制中,常用ReentrantLock,可显式加锁、释放锁
使用方法
class A{
private final ReentrantLock lock = new ReentrantLock();
public void m(){
lock.lock();
try{
//保证线程安全的代码
}finally {
lock.unlock(); //若同步代码有异常,要将unlock()写入finally语句块
}
}
}
买票案例
import java.util.concurrent.locks.ReentrantLock;
//测试Lock锁
public class TestLock {
public static void main(String[] args) {
TestLock2 testLock2 = new TestLock2();
new Thread(testLock2).start();
new Thread(testLock2).start();
new Thread(testLock2).start();
}
}
class TestLock2 implements Runnable{
int ticketNums = 10;
//定义Lock锁
private final ReentrantLock lock = new ReentrantLock();
@Override
public void run() {
while(true){
try{
lock.lock(); //加锁
if(ticketNums > 0){
try {
Thread.sleep(1000);
}catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(ticketNums--);
}else {
break;
}
}finally {
lock.unlock(); //解锁
}
}
}
}
synchronized与Lock的对比
1.synchronized是隐式锁(出作用域后自动释放),Lock是显式锁(手动开启、关闭)
2.Lock只有代码块锁,synchronized有代码块锁和方法锁
3.使用Lock锁,JVM调度线程花费时间少、性能更好 ,有更好的扩展性(子类多)
4.Lock > 同步代码块(方法体内) > 同步方法(方法体外)