基本线程同步(五)使用Lock同步代码块

声明:本文是《 Java 7 Concurrency Cookbook 》的第二章,作者: Javier Fernández González     译者:许巧辉 校对:方腾飞

使用Lock同步代码块

Java提供另外的机制用来同步代码块。它比synchronized关键字更加强大、灵活。它是基于Lock接口和实现它的类(如ReentrantLock)。这种机制有如下优势:

  • 它允许以一种更灵活的方式来构建synchronized块。使用synchronized关键字,你必须以结构化方式得到释放synchronized代码块的控制权。Lock接口允许你获得更复杂的结构来实现你的临界区。
  • Lock 接口比synchronized关键字提供更多额外的功能。新功能之一是实现的tryLock()方法。这种方法试图获取锁的控制权并且如果它不能获取该锁,是因为其他线程在使用这个锁,它将返回这个锁。使用synchronized关键字,当线程A试图执行synchronized代码块,如果线程B正在执行它,那么线程A将阻塞直到线程B执行完synchronized代码块。使用锁,你可以执行tryLock()方法,这个方法返回一个 Boolean值表示,是否有其他线程正在运行这个锁所保护的代码。
  • 当有多个读者和一个写者时,Lock接口允许读写操作分离。
  • Lock接口比synchronized关键字提供更好的性能。

在这个指南中,你将学习如何通过锁来同步代码块和通过Lock接口及其实现者ReentrantLock类来创建临界区,实现一个程序来模拟打印队列。

准备工作

这个指南的例子使用Eclipse IDE实现。如果你使用Eclipse或其他IDE,如NetBeans,打开它并创建一个新的Java项目。

如何做…

按以下步骤来实现的这个例子:

1.创建PrintQueue类,来实现打印队列。

1 public class PrintQueue {

2.声明一个Lock对象,并且使用ReentrantLock类的一个新对象来初始化它。

1 private final Lock queueLock=new ReentrantLock();

3.实现printJob()方法,它将接收Object对象作为参数,并且不会返回任何值。

1 public void printJob(Object document){

4.在printJob()方法内部,通过调用lock()方法来获取Lock对象的控制权。

1 queueLock.lock();

5.然后,包含以下代码来模拟文档的打印:

1 try {
2 Long duration=(long)(Math.random()*10000);
3 System.out.println(Thread.currentThread().getName()+ ":
4 PrintQueue: Printing a Job during "+(duration/1000)+
5 " seconds");
6 Thread.sleep(duration);
7 catch (InterruptedException e) {
8 e.printStackTrace();
9 }

6.最后,通过调用unlock()方法来释放Lock对象的控制。

1 finally {
2 queueLock.unlock();
3 }

7.创建一个Job类,并指定它实现Runnable接口。

1 public class Job implements Runnable {

8.声明一个PrintQueue类的对象,并通过实现类(Job类)的构造器来初始化这个对象。

1 private PrintQueue printQueue;
2 public Job(PrintQueue printQueue){
3 this.printQueue=printQueue;
4 }

9.实现run()方法,它使用PrintQueue对象来发送一个打印任务。

1 @Override
2 public void run() {
3 System.out.printf("%s: Going to print a document\n", Thread.
4 currentThread().getName());
5 printQueue.printJob(new Object());
6 System.out.printf("%s: The document has been printed\n",
7 Thread.currentThread().getName());
8 }

10.通过创建类名为Main,且包括main()方法来实现这个示例的主类。

1 public class Main {
2 public static void main(String[] args) {

11.创建一个共享的PrintQueue对象。

1 PrintQueue printQueue=new PrintQueue();

12.创建10个Job对象,并且使用10个线程来运行它们。

1 Thread thread[]=new Thread[10];
2 for (int i=0; i<10; i++){
3 thread[i]=new Thread(new Job(printQueue),"Thread "+ i);
4 }

13.启动这10个线程。

1 for (int i=0; i<10; i++){
2 thread[i].start();
3 }

它是如何工作的…

从以下截图,你可以看到执行这个示例一部分的输出:

在 printJob()中,PrintQueue类是这个示例的关键所在。当我们通过锁来实现一个临界区并且保证只有一个执行线程能运行一个代码块,我们必 须创建一个ReentrantLock对象。在临界区的起始部分,我们必须通过使用lock()方法来获得锁的控制权。当一个线程A调用这个方法时,如果 没有其他线程持有这个锁的控制权,那么这个方法就会给线程A分配这个锁的控制权并且立即返回允许线程A执行这个临界区。否则,如果其他线程B正在执行由这 个锁控制的临界区,lock()方法将会使线程A睡眠直到线程B完成这个临界区的执行。

在临界区的尾部,我们必须使用unlock()方法来释放锁的控制权,允许其他线程运行这个临界区。如果你在临界区的尾部没有调用unlock()方法,那么其他正在等待该代码块的线程将会永远等待,造成 死锁情况。如果你在临界区使用try-catch代码块,别忘了在finally部分的内部包含unlock()方法的代码。

不止这些…

Lock 接口(和ReentrantLock类)包含其他方法来获取锁的控制权,那就是tryLock()方法。这个方法与lock()方法的最大区别是,如果一 个线程调用这个方法不能获取Lock接口的控制权时,将会立即返回并且不会使这个线程进入睡眠。这个方法返回一个boolean值,true表示这个线程 获取了锁的控制权,false则表示没有。

注释:考虑到这个方法的结果,并采取相应的措施,这是程序员的责任。如果这个方法返回false值,预计你的程序不会执行这个临界区。如果是这样,你可能会在你的应用程序中得到错误的结果。

ReentrantLock类也允许递归调用(锁的可重入性,译者注),当一个线程有锁的控制权并且使用递归调用,它延续了锁的控制权,所以调用lock()方法将会立即返回并且继续递归调用的执行。此外,我们也可以调用其他方法。

更多信息

你必须要非常小心使用锁来避免死锁,这种情况发生在,当两个或两个以上的线程被阻塞等待将永远不会解开的锁。比如,线程A锁定Lock(X)而线程B锁定 Lock(Y)。如果现在,线程A试图锁住Lock(Y)而线程B同时也试图锁住Lock(X),这两个线程将无限期地被阻塞,因为它们等待的锁将不会被解开。请注意,这个问题的发生是因为这两个线程尝试以相反的顺序获取锁(译者注:锁顺序死锁)。在附录中,提供了一些很好的并发编程设计的建议,适当的设计并发应用程序,来避免这些死锁问题。

参见

  • 在第2章,基本线程同步中的同步方法指南
  • 在第2章,基本线程同步中的在锁中使用多条件的指南
  • 在第8章,测试并发应用程序中的监控Lock接口的指南
  • 文章转自 并发编程网-ifeve.com
时间: 2024-08-29 04:26:47

基本线程同步(五)使用Lock同步代码块的相关文章

线程类中可以用静态代码块做初始化静态变量么?这些静态变量会不会有并发问题呢?

问题描述 线程类中可以用静态代码块做初始化静态变量么?这些静态变量会不会有并发问题呢? public class SalesConfirmationUploadThread extends Thread { private boolean result = false; private final SalesConfirmationPipedInputStream input; private long orderId; private String pin; private JingdongS

银行取款[多线程]{使用同步代码块确保线程同步}

经典例子:老婆(朱丽叶)老公(罗密欧),使用银行卡和存折,或者网银等,同时对同一账户操作的安全问题. 此处用多线程实现,同时取款的模拟实现,使用同步代码块确保线程同步,查看取款安全隐患问题,代码如下: --------------------------------------------------------------------------------------------------------------------------------------  * 线程同步 :使用同步块

线程-关于java sychronized锁代码块对该对象的其它sychonized方法是否有影响的问题。

问题描述 关于java sychronized锁代码块对该对象的其它sychonized方法是否有影响的问题. 代码是书上例子,我简化了,启用线程ob1后,通过主线程调用sychonized方法mysuspend()和myresume()来控制ob1线程的等待与重启.线程从15开始倒数,0.5秒个,主线程在1秒时候暂停文字提示,再过1秒重启线程.例子中的sychonized代码块仅包含wait()判定部分,输出与预期一样,倒数到14的时候中断,之后又恢复.但是当我将sychonized代码块前移

线程-同步代码块的问题。。。。。

问题描述 同步代码块的问题..... 数据库的几张表常用表的数据,我想写到几个静态变量中. 然后提供若干个get方法,同时还有一个set方法重置所有的静态变量. 请问如何能够在set重置方法没完成以前,让get方法的线程等待? 同时几个get方法又不能有相互影响. 就是set没完成,所有的get需要等待,但是不同的get之间没完成不需要等待. 来人啊... 解决方案 http://my.oschina.net/tinglanrmb32/blog/339661 解决方案二: http://blog

线程安全-存储过程怎么让代码块同步执行

问题描述 存储过程怎么让代码块同步执行 问题是这样的:一个表,传入参数(多个) 判断 如果表中有这个数据(有一个或者多个参数标识这条记录唯一),更新这条数据 如果表中不存在这条数据,新增一条. 现在把这个逻辑写到存储过程中了,会出现多个线程同时调用这个存储过程,在查询.没插入之前,有多个线程得到都是数据库没有记录的,导致插入多条数据. 怎么把这个过程变成原子的,锁定这块代码. 锁表不太科学,锁行,因为插入之前,数据是不存在的,也锁不住,求帮忙,非常感谢! 这个问题应该不是事务吧?事务是代码可能执

代码块与函数间的同步,其中一个线程只运行了一次

问题描述 publicclassThreadDemo3{publicstaticvoidmain(String[]args){//TODOAuto-generatedmethodstubThreadTest2test2=newThreadTest2();newThread(test2).start();//这个线程调用同步代码块//让主线程暂停一会儿try{Thread.sleep(1);}catch(Exceptione){//TODO:handleexceptionSystem.out.pr

代码块与函数间的同步。为什么线程0只执行了一次

问题描述 publicclassThreadDemo3{publicstaticvoidmain(String[]args){//TODOAuto-generatedmethodstubThreadTest2test2=newThreadTest2();newThread(test2).start();//这个线程调用同步代码块//让主线程暂停一秒try{Thread.sleep(10);}catch(Exceptione){//TODO:handleexceptionSystem.out.pr

java线程学习4——线程同步之同步代码块

  模拟一个场景,两个人对同一个账户同时取钱   package cn.xy.Thread; public class Account {  /**   * 账户号   */  private String accountNo;  /**   * 账户余额   */  private double balance;  public Account()  {   super();  }  public Account(String accountNo, double balance)  {   s

JAVA之旅(十三)——线程的安全性,synchronized关键字,多线程同步代码块,同步函数,同步函数的锁是this

JAVA之旅(十三)--线程的安全性,synchronized关键字,多线程同步代码块,同步函数,同步函数的锁是this 我们继续上个篇幅接着讲线程的知识点 一.线程的安全性 当我们开启四个窗口(线程)把票陆陆续续的卖完了之后,我们要反思一下,这里面有没有安全隐患呢?在实际情况中,这种事情我们是必须要去考虑安全问题的,那我们模拟一下错误 package com.lgl.hellojava; import javax.security.auth.callback.TextInputCallback