Lock应用之 线程同步

简介:

在Java5 诞生之前,Java线程同步和协助只能使用synchronized锁机制,这种锁机制由JVM直接提供实现,所以经常又称内部锁,也称作对象监视器,由于这种锁有性能缺陷而且灵活不足,所以牛人在Java5添加了Lock接口及相关实现,为Java多线程同步与协作提供了另一种选择,当然,后来者不管功能还是性能都比前者优秀,以致有人倡导全部使用Lock抛弃synchronized。有了竞争就会迫使进步,果然Java 6对内部锁进行优化,性能大幅度提升,大部分情况下与Lock可以相媲美(但非公平竞争情况下Lock还是存在性能优势)而且JCP专家组提倡简单正常情况下使用synchronized内部锁,并称将来内部锁还会继续进行优化,所以现在synchronized还是广为使用。然而这只是性能方面,在功能方面Lock提供了更多的灵活性,如轮询锁、定时锁、可中断的锁获取、非块结构的加解锁等,并且在Java5以后的并发包中广为使用,所以学习研究Lock还是必须的。


1
2
3
4
public  interface  Lock {
void  lock();
void  unlock();
}


对于简单线程同步,Lock使用lock()方法请求并加锁,需要自己使用unlock()方法释放锁,并不需要限定代码范围,如果忘记会造成死锁,这是灵活性带来的负面影响,所以使用时都会增加try{...}finally{lock.unlock()},以保证锁的释放;synchronized在代码块范围外会自动释放锁,一开始就定义好锁定的代码范围,比较死板但可靠,另外synchronized经常用于修饰类成员方法,线程调用这种方法前需要先获得对应的锁,这种使用方式很顺手,简单明了,Lock没有这种语法。


Lock和synchronized对线程同步操作示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
import  java.util.concurrent.TimeUnit;
import  java.util.concurrent.locks.Lock;
import  java.util.concurrent.locks.ReentrantLock;
public  class  TestLock {
     private  int  mCount;
     private  Lock mLock;
                       
     public  TestLock(){
         this .mCount =  0 ;
         this .mLock =  new  ReentrantLock();
     }
                       
     public  static  void  main(String[] args)  throws  InterruptedException {
         TestLock testLock =  new  TestLock();
                           
         for ( int  i= 0 ; i< 2 ; i++){
             new  Thread(testLock. new  SThread()).start();
         }
                           
         TimeUnit.SECONDS.sleep( 3 );
                           
         for ( int  i= 0 ; i< 2 ; i++){
             new  Thread(testLock. new  LThread()).start();
         }
     }
                       
     private  void  increaseCount() {
         mCount++;
         System.out.println(Thread.currentThread().getName() +  " do lock, count: "  + mCount);
     }
                       
     private  void  decreaseCount() {
         mCount--;
         System.out.println(Thread.currentThread().getName() +  " do unlock, count: "  + mCount);
     }
                       
     private  void  doCountWithLSync() {
         mLock.lock();
         try {
             increaseCount();
                               
             mLock.lock(); //re-enter
                               
             try {
                 increaseCount();
             }
             finally {
                 mLock.unlock();
                 decreaseCount();
             }                            
         }
         finally {
             mLock.unlock();
             decreaseCount();
         }
     }
     private  void  doCountWithSSync() {
         synchronized ( this ){
             increaseCount();
                               
             synchronized ( this ){ //re-enter
                 increaseCount();
             }
                               
             decreaseCount();
         }
                           
         decreaseCount();
     }
     private  class  LThread  implements  Runnable{
         public  void  run(){
             doCountWithLSync();  
         }
     }
                       
     private  class  SThread  implements  Runnable{
         public  void  run(){   
             doCountWithSSync();
         }    
     }
}





     本文转自sarchitect 51CTO博客,原文链接:http://blog.51cto.com/stevex/1300164,如需转载请自行联系原作者


相关文章
|
26天前
|
缓存 Java 开发者
Java多线程并发编程:同步机制与实践应用
本文深入探讨Java多线程中的同步机制,分析了多线程并发带来的数据不一致等问题,详细介绍了`synchronized`关键字、`ReentrantLock`显式锁及`ReentrantReadWriteLock`读写锁的应用,结合代码示例展示了如何有效解决竞态条件,提升程序性能与稳定性。
111 6
|
25天前
|
监控 Java 数据库连接
Java线程管理:守护线程与用户线程的区分与应用
在Java多线程编程中,线程可以分为守护线程(Daemon Thread)和用户线程(User Thread)。这两种线程在行为和用途上有着明显的区别,了解它们的差异对于编写高效、稳定的并发程序至关重要。
29 2
|
1月前
|
数据采集 存储 数据处理
Python中的多线程编程及其在数据处理中的应用
本文深入探讨了Python中多线程编程的概念、原理和实现方法,并详细介绍了其在数据处理领域的应用。通过对比单线程与多线程的性能差异,展示了多线程编程在提升程序运行效率方面的显著优势。文章还提供了实际案例,帮助读者更好地理解和掌握多线程编程技术。
|
1月前
|
存储 监控 安全
深入理解ThreadLocal:线程局部变量的机制与应用
在Java的多线程编程中,`ThreadLocal`变量提供了一种线程安全的解决方案,允许每个线程拥有自己的变量副本,从而避免了线程间的数据竞争。本文将深入探讨`ThreadLocal`的工作原理、使用方法以及在实际开发中的应用场景。
56 2
|
1月前
|
安全 Java 开发者
Java 多线程并发控制:深入理解与实战应用
《Java多线程并发控制:深入理解与实战应用》一书详细解析了Java多线程编程的核心概念、并发控制技术及其实战技巧,适合Java开发者深入学习和实践参考。
58 6
|
1月前
|
存储 安全 Java
Java多线程编程中的并发容器:深入解析与实战应用####
在本文中,我们将探讨Java多线程编程中的一个核心话题——并发容器。不同于传统单一线程环境下的数据结构,并发容器专为多线程场景设计,确保数据访问的线程安全性和高效性。我们将从基础概念出发,逐步深入到`java.util.concurrent`包下的核心并发容器实现,如`ConcurrentHashMap`、`CopyOnWriteArrayList`以及`BlockingQueue`等,通过实例代码演示其使用方法,并分析它们背后的设计原理与适用场景。无论你是Java并发编程的初学者还是希望深化理解的开发者,本文都将为你提供有价值的见解与实践指导。 --- ####
|
1月前
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
47 4
|
2月前
|
存储 并行计算 安全
C++多线程应用
【10月更文挑战第29天】C++ 中的多线程应用广泛,常见场景包括并行计算、网络编程中的并发服务器和图形用户界面(GUI)应用。通过多线程可以显著提升计算速度和响应能力。示例代码展示了如何使用 `pthread` 库创建和管理线程。注意事项包括数据同步与互斥、线程间通信和线程安全的类设计,以确保程序的正确性和稳定性。
|
2月前
|
监控 Java
在实际应用中选择线程异常捕获方法的考量
【10月更文挑战第15天】选择最适合的线程异常捕获方法需要综合考虑多种因素。没有一种方法是绝对最优的,需要根据具体情况进行权衡和选择。在实际应用中,还需要不断地实践和总结经验,以提高异常处理的效果和程序的稳定性。
30 3
|
2月前
|
调度 Android开发 开发者
构建高效Android应用:探究Kotlin多线程优化策略
【10月更文挑战第11天】本文探讨了如何在Kotlin中实现高效的多线程方案,特别是在Android应用开发中。通过介绍Kotlin协程的基础知识、异步数据加载的实际案例,以及合理使用不同调度器的方法,帮助开发者提升应用性能和用户体验。
66 4