降低锁的竞争可以提高并发程序的性能和可伸缩性,有3种方式可以降低锁的竞争:
1. 减少锁的持有时间(缩小锁的范围)
2. 降低锁的请求频率(降低锁的粒度)
3. 放弃使用独占锁,使用并发容器,原子变量,读写锁等等来代替它。
减少锁的持有时间(减小锁的范围):
减少锁的持有时间实际上就是减小锁的控制范围,将一些并不需要锁的操作从同步代码块中移除。如下所示,需要进行同步操作的只有attributes.get(key);这一行代码。
//可以优化的代码
class AttributeStore{
private final Map<String,String> attributes=new HashMap<String,String>();
public synchronized boolean userLocationMatches(String username,String regex){
String key="user."+username;
String location=attributes.get(key);
if(location==null)
return false;
else
return Pattern.matches(regex,location);
}
}
缩小锁的范围如下,将不需要同步的内容移出代码块。
//优化之后的代码
class AttributeStore{
private final Map<String,String> attributes=new HashMap<String,String>();
public boolean userLocationMatches(String username,String regex){
String key="user."+username;
String location;
synchronized (this) {
location=attributes.get(key);
}
if(location==null)
return false;
else
return Pattern.matches(regex,location);
}
}
降低锁的请求频率(降低锁的粒度):
通过将粗粒度的锁分解为多个细粒度的锁,从而将原来到一个锁的请求分担到多个锁。常用的方案是锁分解或锁分段(一个锁分解为两个锁称为锁分解,一个锁分解为多个锁称为锁分段)。
在代码中,当一个锁需要同时保护多个互相独立的共享状态变量的时候,可以考虑锁分解或锁分段。
先来看一个
锁分解的例子:
//可以锁分解的代码
class ServerStatus{
private Set<String> users;
private Set<String> queries;
public synchronized void addUser(String user){
users.add(user);
}
public synchronized void removeUser(String user){
users.remove(user);
}
public synchronized void addQuery(String query){
queries.add(query);
}
public synchronized void removeQuery(String query){
queries.remove(query);
}
}
在上面的代码中,同一个ServerStatus对象锁用于保护2个独立的共享变量,可以使用锁分解。
//优化后的代码
class ServerStatus{
private Set<String> users;
private Set<String> queries;
public void addUser(String user){
synchronized (users) {
users.add(user);
}
}
public void removeUser(String user){
synchronized (users) {
users.remove(user);
}
}
public void addQuery(String query){
synchronized (queries) {
queries.add(query);
}
}
public void removeQuery(String query){
synchronized (queries) {
queries.remove(query);
}
}
}
锁分段的典型应用是ConcurrentHashMap。在Collections.synchronizedMap()方法中,使用组合的方式将传入Map的方法放入同步代码块中执行,所有的同步代码块使用同一个对象锁。为了提高容器的性能,在ConcurrentHashMap容器中使用16个对象锁,每个对象锁保护所有散列桶的1/16,其中第N个散列桶由第(N%16)个对象锁来保护。大致的思路如下:
class MyMap<K,V>{
static final class Node<K,V>{
private K key;
private V value;
private Node<K,V> next;
public Node<K, V> getNext() {
return next;
}
//...set get equals hashCode...//
}
private final static int N_LOCKS=16;
private Object[] mylocks;
private Node<K,V>[] buckets;
public MyMap(int num) {
mylocks=new Object[N_LOCKS];
for(int i=0;i<N_LOCKS;i++){
mylocks[i]=new Object();
}
buckets=new Node[num];
}
public V get(K key){
int bucketIndex=key.hashCode()%buckets.length;//定位目标所在的桶
synchronized (mylocks[bucketIndex%N_LOCKS]) {//获取桶对应的锁
for(Node<K,V> node=buckets[bucketIndex];node!=null;node=node.getNext()){
if(key.equals(node.key))
return node.value;
}
return null;
}
}
//......
}
放弃使用独占锁:
我们可以放弃使用独占锁,使用并发容器,原子变量,读写锁等等来代替他。
分享到:
相关推荐
Java并发编程中,锁是一种关键机制,用于控制多个线程对共享资源的访问。本文主要探讨了Java中的两种广义锁概念——乐观锁和悲观锁,以及自旋锁和适应性自旋锁的区别和应用场景。 1. 乐观锁与悲观锁: 乐观锁认为在...
- 避免长时间持有锁,减少锁竞争。 - 使用线程池管理线程,提高资源利用率。 - 使用并发工具而非手工同步,如可能的话,优先选择不可变对象。 7. **Java内存模型** - **JMM(Java Memory Model)** 规定了线程...
- **最小化锁的使用** 减少锁的使用可以降低线程间的竞争,提高并发性能。 - **避免长时间持有锁** 长时间持有锁会导致其他线程等待,影响整体吞吐量。 - **使用并发原语** 例如`java.util.concurrent.atomic`...
在Java并发编程中,`Lock`接口和`ReentrantLock`类扮演着至关重要的角色。它们为多线程环境提供了更高级别的控制和灵活性,弥补了内置锁(也称为监视器锁或内部锁)的不足。`ReentrantLock`是Java 5.0引入的,作为对...
Java并发编程中,锁是确保线程安全的关键机制。本文主要讨论了四种锁类型:乐观锁、悲观锁、自旋锁以及Java中的synchronized同步锁,并深入解析了synchronized锁的内部机制,包括其核心组件、实现方式以及锁的状态。...
- **锁分离**:讲解锁分离技术,以及如何通过减少锁的竞争来提高程序性能。 - **无锁编程**:探讨无锁编程的基本原理和技术细节,包括使用原子变量进行并发控制的方法。 - **并发集合**:详细介绍Java提供的各种并发...
### Java并发编程实践 #### 一、并发编程基础 ##### 1.1 并发与并行的区别 在Java并发编程中,首先需要理解“并发”(Concurrency)和“并行”(Parallelism)的区别。“并发”指的是多个任务在同一时间段内交替...
在Java多线程并发编程中,ReentrantReadWriteLock(可重入读写锁)是一个重要的同步工具,它属于Java并发包(java.util.concurrent.locks)中的一个类。这个锁提供了比标准的synchronized关键字更细粒度的控制,允许...
- **减少锁的竞争**:尽可能减少使用锁的范围和时间。 - **使用局部变量**:使用线程本地变量(ThreadLocal)来存储经常使用的数据,减少访问共享资源的次数。 综上所述,“JAVA并发编程实践”这一主题涵盖了Java并发...
Java并发编程是Java开发中的重要领域,特别是在多线程应用中,锁机制是保障线程安全、实现资源有效管理的关键工具。本资料“Java并发锁简介-动力节点共9页.pdf.zip”似乎提供了关于Java并发锁的基础介绍。下面将详细...
10. **并发性能优化**:笔记会探讨如何通过优化并发策略来提高程序性能,例如减少锁的竞争、合理使用并发工具以及避免活锁和死锁。 11. **案例分析与实战**:笔记可能包含实际的并发编程案例,帮助读者更好地将理论...
深入地探讨Java并发编程实践中的显示锁概念,我们聚焦于《Java并发编程实践-电子书-07章》所提供的丰富内容。本章着重于展示如何使用显示锁(也称为高级锁),并详细介绍了Lock和ReentrantLock接口以及它们在并发...
- **减少锁的竞争**:使用细粒度锁或无锁编程。 - **减少上下文切换**:合理使用线程池,减少线程创建的数量。 ### 六、并发工具类与实用技巧 #### 6.1 CyclicBarrier与CountDownLatch - **CyclicBarrier**:可以...
Java线程亲和性(Thread Affinity)是一个高级并发编程概念,主要涉及到操作系统调度和硬件资源的优化。在多核处理器系统中,线程亲和性允许开发者指定某个线程应该运行在哪个特定的处理器核心上,从而提高性能、...
##### 5.1 减少锁的竞争 - 使用局部变量代替共享变量。 - 尽量减少锁的粒度,采用细粒度锁。 ##### 5.2 使用并发容器 - `ConcurrentHashMap`:线程安全的哈希表,相比`synchronized`的`HashMap`性能更高。 - `...
根据提供的信息,“JAVA并发编程实践.pdf...掌握Java并发编程的基本原理和技术对于提升系统性能、减少延迟和提高用户体验有着重要意义。希望通过对以上知识点的学习,能够帮助开发者更好地理解和运用Java并发编程技术。
《Java并发编程实践》是Java开发者深入理解并发编程的重要参考资料,尤其对于想要提升多线程应用设计和性能优化技能的程序员来说,这本书提供了丰富的实践经验和深入的理论知识。以下是根据提供的章节内容概述的一些...
Java并发机制的底层实现原理涉及到多个方面,包括了本地内存与线程安全的问题、volatile关键字的使用、synchronized关键字的原理以及Java并发在处理器层面是如何实现的。通过这些机制,Java能够有效地管理多线程环境...
Java并发编程是指在Java语言中编写多线程和多任务执行的程序,以便更高效地利用计算机的多核处理器资源。并发编程是Java高级编程技能中的重要组成部分,尤其是在需要处理大量数据、提供快速响应、实现高吞吐量和高可...
本篇文章将深入探讨Java并发编程的相关知识点,主要基于提供的两个文件——"Java并发编程实战(中文版).pdf"和"Java Concurrency in Practice.pdf"。 1. **线程与并发** - **线程基础**:Java中的线程是并发执行...