本节我们来讲使用java.util.concurrent.locks.Lock来实现线程间的同步,在讲这个之前啊,我们先看看官方的Api里面对于该接口的描述,我们会发现实现该接口的类只有两个:ReentrantLock和ReentrantReadWriteLock,本节我们主要讨论ReentrantLock,ReentrantReadWriteLock放在后面的章节在细说。
我们先来看看这个Lock,它是Java提供的同步代码块的另外一种机制,它的功能比synchronized关键字要强大很多倍。它的好处主要有以下几个:
1、支持更灵活的同步代码块结构。使用synchronized关键字只能在同一个代码块中实现同步,而Lock接口则允许实现更复杂的临界区结构。
2、相比synchronized关键字,Lock接口提供了更多的功能,比如说tryLock()方法,如果锁可用,则获取锁,并立即返回值 true。如果锁不可用,则此方法将立即返回值 false并继续往下执行代码。而使用synchronized关键字则会挂起当前线程,直到其他线程执行完毕。
3、Lock接口的ReentrantReadWriteLock实现类,运行读写分离。
好了,接下来我们来看个例子,将第二节代码中添加如下方法
ok,我们运行第二节中的测试类,发现Lock实现了同样的功能。
其实Lock接口还有更多的功能,我上面的例子只是做一个抛砖引玉的作用,更多的功能(比如:tryLock()方法、递归调用Lock锁等等),还有待大家自己写测试例子体会。
我们先来看看这个Lock,它是Java提供的同步代码块的另外一种机制,它的功能比synchronized关键字要强大很多倍。它的好处主要有以下几个:
1、支持更灵活的同步代码块结构。使用synchronized关键字只能在同一个代码块中实现同步,而Lock接口则允许实现更复杂的临界区结构。
2、相比synchronized关键字,Lock接口提供了更多的功能,比如说tryLock()方法,如果锁可用,则获取锁,并立即返回值 true。如果锁不可用,则此方法将立即返回值 false并继续往下执行代码。而使用synchronized关键字则会挂起当前线程,直到其他线程执行完毕。
3、Lock接口的ReentrantReadWriteLock实现类,运行读写分离。
好了,接下来我们来看个例子,将第二节代码中添加如下方法
private Lock lock = new ReentrantLock();
/**
* 存钱
* @param num 数量
*/
public void addMoney(int num){
//加锁
lock.lock();
System.out.println("-->开始存钱......");
double temp = money;
try {
TimeUnit.MILLISECONDS.sleep(21l);
} catch (InterruptedException e) {
e.printStackTrace();
}
money = temp + num;
System.out.println("-->存钱结束......");
//释放所
lock.unlock();
}
/**
* 取钱
* @param num 数量
*/
public void subMoney(int num){
//加锁
lock.lock();
System.out.println("-->开始取钱......");
double temp = money;
try {
TimeUnit.MILLISECONDS.sleep(19l);
} catch (InterruptedException e) {
e.printStackTrace();
}
money = temp - num;
System.out.println("-->取钱结束......");
//释放所
lock.unlock();
}
ok,我们运行第二节中的测试类,发现Lock实现了同样的功能。
其实Lock接口还有更多的功能,我上面的例子只是做一个抛砖引玉的作用,更多的功能(比如:tryLock()方法、递归调用Lock锁等等),还有待大家自己写测试例子体会。