当前位置:   article > 正文

Java并发基础(7):锁对象Lock、ReadWriteLock_readlock().lock()

readlock().lock()

目录

1、Lock       

2、读写锁(ReadWriteLock)

需求:


1、Lock       

Lock是java.util.concurrent.locks包下的接口,Lock 实现提供了比使用synchronized 方法和语句可获得的更广泛的锁定操作,它能以更优雅的方式处理线程同步问题

  1. public class LockTest {
  2. public static void main(String[] args) {
  3. final Output output = new Output();
  4. new Thread() {
  5. @Override
  6. public void run() {
  7. output.output("zhangsan");
  8. };
  9. }.start();
  10. new Thread() {
  11. @Override
  12. public void run() {
  13. output.output("lisi");
  14. };
  15. }.start();
  16. }
  17. static class Output{
  18. final Lock lock = new ReentrantLock();
  19. public void output(String s){
  20. lock.lock();
  21. try {
  22. for (int i = 0; i < s.length(); i++) {
  23. System.out.print(s.charAt(i));
  24. Thread.sleep(10);
  25. }
  26. } catch (InterruptedException e) {
  27. e.printStackTrace();
  28. } finally {
  29. lock.unlock();
  30. }
  31. }
  32. }
  33. }
  34. 加lock :zhangsanlisi
  35. 不加lock: zlhanigsasni

这样就实现了和sychronized一样的同步效果,需要注意的是

  • sychronized修饰的方法或者语句块在代码执行完之后锁自动释放
  • 用Lock需要我们手动释放锁,所以为了保证锁最终被释放(发生异常情况),要把互斥区放在try内,释放锁放在finally内。

2、读写锁(ReadWriteLock)

需求:

在对数据进行读写的时候,为了保证数据的一致性和完整性,需要读和写是互斥的,写和写是互斥的,但是读和读是不需要互斥的,这样读和读不互斥性能更高些,来看一下不考虑互斥情况的代码原型:

  1. public class ReadWriteLockTest {
  2.     public static void main(String[] args) {
  3.         final Data data = new Data();
  4.         for (int i = 0; i < 3; i++) {
  5.             new Thread(new Runnable() {
  6.                 public void run() {
  7.                     for (int j = 0; j < 5; j++) {
  8.                         data.set(new Random().nextInt(30));
  9.                     }
  10.                 }
  11.             }).start();
  12.         }        
  13.         for (int i = 0; i < 3; i++) {
  14.             new Thread(new Runnable() {
  15.                 public void run() {
  16.                     for (int j = 0; j < 5; j++) {
  17.                         data.get();
  18.                     }
  19.                 }
  20.             }).start();
  21.         }
  22.     }
  23. }
  24. class Data {    
  25.     private int data;// 共享数据    
  26.     public void set(int data) {
  27.         System.out.println(Thread.currentThread().getName() + "准备写入数据");
  28.         try {
  29.             Thread.sleep(20);
  30.         } catch (InterruptedException e) {
  31.             e.printStackTrace();
  32.         }
  33.         this.data = data;
  34.         System.out.println(Thread.currentThread().getName() + "写入" + this.data);
  35.     }    
  36.     public void get() {
  37.         System.out.println(Thread.currentThread().getName() + "准备读取数据");
  38.         try {
  39.             Thread.sleep(20);
  40.         } catch (InterruptedException e) {
  41.             e.printStackTrace();
  42.         }
  43.         System.out.println(Thread.currentThread().getName() + "读取" + this.data);
  44.     }
  45. }
  46.         部分输出结果:
  47. Thread-1准备写入数据
  48. Thread-3准备读取数据
  49. Thread-2准备写入数据
  50. Thread-0准备写入数据
  51. Thread-4准备读取数据
  52. Thread-5准备读取数据
  53. Thread-2写入12
  54. Thread-4读取12
  55. Thread-5读取5
  56. Thread-1写入12

我们要实现写入和写入互斥,读取和写入互斥,读取和读取互斥,在set和get方法加入sychronized修饰符:

  1. public synchronized void set(int data) {...}    
  2. public synchronized void get() {...}
  3.         部分输出结果:
  4. Thread-0准备写入数据
  5. Thread-0写入9
  6. Thread-5准备读取数据
  7. Thread-5读取9
  8. Thread-5准备读取数据
  9. Thread-5读取9
  10. Thread-5准备读取数据
  11. Thread-5读取9
  12. Thread-5准备读取数据
  13. Thread-5读取9

 我们发现,虽然写入和写入互斥了,读取和写入也互斥了,但是读取和读取之间也互斥了,不能并发执行,效率较低,用读写锁实现代码如下:

  1. class Data {    
  2.     private int data;// 共享数据
  3.     private ReadWriteLock rwl = new ReentrantReadWriteLock();    
  4.     public void set(int data) {
  5.         rwl.writeLock().lock();// 取到写锁
  6.         try {
  7.             System.out.println(Thread.currentThread().getName() + "准备写入数据");
  8.             try {
  9.                 Thread.sleep(20);
  10.             } catch (InterruptedException e) {
  11.                 e.printStackTrace();
  12.             }
  13.             this.data = data;
  14.             System.out.println(Thread.currentThread().getName() + "写入" + this.data);
  15.         } finally {
  16.             rwl.writeLock().unlock();// 释放写锁
  17.         }
  18.     }    
  19.     public void get() {
  20.         rwl.readLock().lock();// 取到读锁
  21.         try {
  22.             System.out.println(Thread.currentThread().getName() + "准备读取数据");
  23.             try {
  24.                 Thread.sleep(20);
  25.             } catch (InterruptedException e) {
  26.                 e.printStackTrace();
  27.             }
  28.             System.out.println(Thread.currentThread().getName() + "读取" + this.data);
  29.         } finally {
  30.             rwl.readLock().unlock();// 释放读锁
  31.         }
  32.     }
  33. }
  34.         部分输出结果:
  35. Thread-4准备读取数据
  36. Thread-3准备读取数据
  37. Thread-5准备读取数据
  38. Thread-5读取18
  39. Thread-4读取18
  40. Thread-3读取18
  41. Thread-2准备写入数据
  42. Thread-2写入6
  43. Thread-2准备写入数据
  44. Thread-2写入10
  45. Thread-1准备写入数据
  46. Thread-1写入22
  47. Thread-5准备读取数据

从结果可以看出实现了我们的需求,这只是锁的基本用法,锁的机制还需要继续深入学习。


原文:https://blog.csdn.net/ghsau/article/details/7461369

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/我家小花儿/article/detail/956091
推荐阅读
相关标签
  

闽ICP备14008679号