关于synchronized与lock的性能比较

本文对比分析了synchronized、Lock及ConcurrentHashMap等并发控制机制在不同环境下的性能表现,发现它们在HotSpot与ART虚拟机上的性能差异显著。

记得当初看教程的时候大家都说lock性能比好不少,最近需要自己设计一个缓存终于要自己尝试一番了。

1.关于两者的实现的比较

A).一般认为synchronized关键字的实现是源自于像信号量之类的线程同步机制,涉及到线程运行状态的切换,在高并发状态下,CPU消耗过多的时间在线程的调度上,从而造成了性能的极大浪费。然而真的如此么?
B).lock实现原理则是依赖于硬件,现代处理器都支持CAS指令,所谓CAS指令简单的来说Compare And Set,CPU循环执行指令直到得到所期望的结果,换句话来说就是当变量真实值不等于当前线程调用时的值的时候(说明其他线程已经将这个值改变),就不会赋予变量新的值。这样就保证了变量在多线程环境下的安全性。

然而,现实情况是当JDK版本高于1.6的时候,synchronized已经被做了CAS的优化:具体是这样的,当执行到synchronized代码块时,先对对象头的锁标志位用lock cmpxchg的方式设置成“锁住“状态,释放锁时,在用lock cmpxchg的方式修改对象头的锁标志位为”释放“状态,写操作都立刻写回主内存。JVM会进一步对synchronized时CAS失败的那些线程进行阻塞操作(调用操作系统的信号量)(此段来摘自别处)。也就是先CAS操作,不行的话继而阻塞线程。

除此之外,系统环境,CPU架构,虚拟机环境都会影响两者的性能关系。

2.用数据说话

1).X86_64 cpu i7 4910mq @4.0ghz ,Windows10 64bit,JDK1.8 hotspot 64bit虚拟机环境

测试代码

测试对某Map对象高并发下的读写线程安全测试
测试对比有synchronized,ReadWriteLock,ConcurrentHashMap,

?
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
public class MapTest {
 
      private Map<integer,string> map = new ConcurrentHashMap<>();
 
      private long starttime;
 
      private AtomicInteger count = new AtomicInteger(t_count);
 
      private final static int t_count = 5000 ;
 
      private final static int rw_count = 10000 ;
 
         Runnable readrun = new Runnable() {
             @Override
             public void run() {
                 int i = rw_count;
                 while (i > 0 ){
                     map.get(i);
                     i--;
                 }
                 System.out.println( "read-mapsize=" +map.size());
                 if (count.decrementAndGet() == 0 )
                     System.out.println( "time=" + (System.currentTimeMillis() - starttime + "ms" ));
             }
         };
 
         Runnable writerun = new Runnable() {
             @Override
             public void run() {
                 int i = rw_count;
                 while (i > 0 ){
                     map.put(i,i+ "" );
                     i--;
                 }
                 System.out.println( "write-mapsize=" +map.size());
                 if (count.decrementAndGet() == 0 )
                     System.out.println( "time=" + (System.currentTimeMillis() - starttime + "ms" ));
             }
         };
 
         public void run(){
             starttime = System.currentTimeMillis();
             for ( int i = 0 ;i < t_count/ 2 ;i ++){
                 new Thread(writerun).start();
                 new Thread(readrun).start();
             }
         }
}
</integer,string>

HashMap 用synchronized重写

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public class SyncHashMap extends HashMap{
 
     @Override
     public Object get(Object key) {
         // TODO Auto-generated method stub
         synchronized ( this ) {
             return super .get(key);
         }
     }
 
     @Override
     public synchronized Object put(Object key, Object value) {
         // TODO Auto-generated method stub
         synchronized ( this ) {
             return super .put(key, value);
         }
 
     }
 
}

用读写锁实现的Map代理类,有些粗糙,没加try finally

?
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
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
public class SyncMapProxy<k,v> implements Map<k,v>{
 
     private Map<k,v> origin;
     private ReadWriteLock lock;
 
     public SyncMapProxy(Map<k, v= "" > origin) {
         this .origin = origin;
         lock = new ReentrantReadWriteLock();
     }
 
     public static  <k,v> SyncMapProxy<k,v> SyncMap(Map<k,v> map){
         return new SyncMapProxy<k,v>(map);
     }
 
     @Override
     public void clear() {
         lock.writeLock().lock();
         origin.clear();
         lock.writeLock().unlock();
     }
 
     @Override
     public boolean containsKey(Object key) {
         lock.readLock().lock();
         boolean res = origin.containsKey(key);
         lock.readLock().unlock();
         return res;
     }
 
     @Override
     public boolean containsValue(Object value) {
         lock.readLock().lock();
         boolean res = origin.containsKey(value);
         lock.readLock().unlock();
         return res;
     }
 
     @Override
     public Set<entry<k, v= "" >> entrySet() {
         lock.readLock().lock();
         Set<entry<k, v= "" >> res = origin.entrySet();
         lock.readLock().unlock();
         return res;
     }
 
     @Override
     public V get(Object key) {
         lock.readLock().lock();
         V res = origin.get(key);
         lock.readLock().unlock();
         return res;
     }
 
     @Override
     public boolean isEmpty() {
         return origin.isEmpty();
     }
 
     @Override
     public Set<k> keySet() {
         lock.readLock().lock();
         Set<k> res = origin.keySet();
         lock.readLock().unlock();
         return res;
     }
 
     @Override
     public V put(K key, V value) {
         lock.writeLock().lock();
         V v = origin.put(key, value);
         lock.writeLock().unlock();
         return v;
     }
 
     @Override
     public void putAll(Map<!--? extends K, ? extends V--> map) {
         lock.writeLock().lock();
         origin.putAll(map);
         lock.writeLock().unlock();
     }
 
     @Override
     public V remove(Object key) {
         lock.writeLock().lock();
         V v = origin.remove(key);
         lock.writeLock().unlock();
         return v;
     }
 
     @Override
     public int size() {
         return origin.size();
     }
 
     @Override
     public Collection<v> values() {
         lock.readLock().lock();
         Collection<v> res = origin.values();
         lock.readLock().unlock();
         return res;
     }
}
</v></v></k></k></entry<k,></entry<k,></k,v></k,v></k,v></k,v></k,></k,v></k,v></k,v>

并发量100000,每个线程对Map执行读写100次,总耗时
ConcurrentHashMap:6112ms
synchronized:6121ms
ReadWriteLock:6182ms
Collections.synchronizedMap:6175ms

并发量10000,每个线程对Map执行读写1000次,总耗时
ConcurrentHashMap:1126ms
synchronized:1145ms
ReadWriteLock:2086ms
Collections.synchronizedMap:1170ms

并发量5000,每个线程对Map执行读写10000次,总耗时
ConcurrentHashMap:1206ms
synchronized:4896ms
ReadWriteLock:8505ms
Collections.synchronizedMap:4883ms

并发量1000,每个线程对Map执行读写100000次,总耗时
ConcurrentHashMap:1748ms
synchronized:9341ms
ReadWriteLock:18720ms
Collections.synchronizedMap:8945ms

并发量100,每个线程对Map执行读写1000000次,总耗时
ConcurrentHashMap:1922ms
synchronized:8417ms
ReadWriteLock:16110ms
Collections.synchronizedMap:9604ms

事实证明在以上的配置环境JDK1.8 X86 Windows10下,高并发下这几种方式性能都相差无几,较高和较低并发下,synchronized都比ReadWriteLock来的快,基本是两倍的关系。ConcurrentHashMap作为同步的Map还是时间性能还是最高的。总之在hotspot下都是一个数量级的。


2).下面看另外一种环境
Android6.0 X86_64模拟器镜像,ART Runtime

并发量20,每个线程对Map执行读写1000000次,总耗时
ConcurrentHashMap:10841ms
synchronized:239452ms
ReadWriteLock:16450ms
Collections.synchronized:213429ms

并发量200,每个线程对Map执行读写10000次,总耗时
ConcurrentHashMap:973ms
synchronized:57047ms
ReadWriteLock:1274ms
Collections.synchronized:52746ms

**难以置信的性能差距,synchronized和Lock在Android的Art环境下确实有着一个数量级的差距,可达数十倍之多,但是在Hotspot环境下却恰恰相反,lock在多数情况下反而不如synchronized。
这里估计是Android Art虚拟机尚未对synchronized进行CAS优化,主要还是因为Android现在作为客户端操作系统,对高并发的资源竞争并无必要做优化,以上结果尚不能下定论,看来要去扒一扒Art的源码才能知道具体的原因了。**

总结

如果开发的是服务器程序,并且使用的是最新的hotspot虚拟机,synchronized和lock其实已经相差无几,其底层实现已经差不多了。但是如果你是Android开发者,使用synchronized还是需要考虑其性能差距的。

<think>好的,我需要回答用户关于Javasynchronized关键字Lock接口的区别、使用场景和性能比较的问题。首先,我需要回顾一下用户提供的引用内容,确保信息准确。 用户提供了五个引用,其中提到synchronized是关键字,而Lock是接口;synchronized自动释放锁,Lock需要手动管理;Lock提供更多灵活性,比如可中断、超时、公平锁等。性能方面,早期synchronized性能较差,但后续优化后差距缩小。使用场景方面,synchronized适合简单同步,Lock适合复杂需求。 接下来,我需要按照用户的要求组织回答结构,分点说明区别、使用场景和性能比较。同时,要使用中文,确保行内数学表达式正确,但这里可能不涉及数学公式,所以重点在格式和引用标注。需要检查每个引用的内容是否准确对应到回答中,并在末尾添加相关问题。 可能需要注意的细节:引用[3]提到Lock只适用于代码块锁,而synchronized可以修饰方法和代码块。引用[5]指出Lock是接口,synchronized是关键字。引用[4]展示了Lock的使用示例,需要在回答中提到代码结构差异。性能部分需要结合引用[2]提到的优化情况。 最后,生成相关问题时要围绕主题,比如区别、使用场景、性能优化、死锁处理等。确保引用标识正确,例如[^1]、[^2]等。</think>### JavasynchronizedLock接口的对比分析 #### **核心区别** 1. **实现方式** - `synchronized`是Java内置关键字,通过JVM底层实现锁机制[^5]。 - `Lock`是`java.util.concurrent.locks`包中的接口,需通过具体类(如`ReentrantLock`)实现锁功能。 2. **锁的获取释放** - `synchronized`自动管理锁:进入同步代码块时获取锁,退出时自动释放[^3]。 - `Lock`需手动调用`lock()`和`unlock()`方法,通常配合`try-finally`块确保释放锁[^4]。 3. **功能特性** - `Lock`支持更灵活的功能,如: - **可中断锁**:通过`lockInterruptibly()`响应中断请求。 - **超时获取锁**:`tryLock(long time, TimeUnit unit)`。 - **公平锁**:通过构造函数指定公平性,减少线程饥饿[^3]。 - `synchronized`仅支持非公平锁,且无法中断等待中的线程。 4. **代码范围** - `synchronized`可修饰方法(实例方法、静态方法)或代码块[^3]。 - `Lock`仅适用于代码块锁。 --- #### **性能比较** - **早期版本**:`synchronized`因重量级锁性能较差,而`Lock`通过CAS操作(如`ReentrantLock`)更高效。 - **JDK 6+优化**:JVM对`synchronized`进行了锁升级(偏向锁→轻量级锁→重量级锁),性能差距显著缩小[^2]。 - **实际场景**:高并发且锁竞争激烈时,`Lock`的细粒度控制(如超时)可能更优;简单场景下`synchronized`代码更简洁。 --- #### **使用场景** 1. **优先使用`synchronized`的情况** - 简单的同步需求(如单方法内的线程安全)。 - 需要减少代码复杂度(自动锁管理)[^3]。 2. **优先使用`Lock`的情况** - 需要实现可中断锁、超时控制或公平锁。 - 需在多个条件变量间灵活切换(结合`Condition`类)[^4]。 - 锁竞争激烈且需优化吞吐量。 --- #### **代码示例对比** ```java // 使用synchronized public class Counter { private int count; public synchronized void increment() { count++; } } ``` ```java // 使用Lock public class Counter { private int count; private Lock lock = new ReentrantLock(); public void increment() { lock.lock(); try { count++; } finally { lock.unlock(); } } } ``` (示例参考引用) ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值