Java多线程系列——锁

0.引言

在并发编程中,锁是一种重要的同步机制,用于控制对共享资源的访问。Java 提供了多种锁的实现,每种锁都有不同的特性和适用场景。本文将深入介绍 Java 中常见的锁类型,包括内置锁、显式锁、读写锁等,并讨论它们的使用方法和最佳实践。

1. 内置锁(synchronized)

内置锁是 Java 中最基本的锁机制,通过 synchronized 关键字来实现。它可以用于同步方法或同步代码块,保证同一时间只有一个线程可以执行被锁定的代码,从而确保线程安全性。

public synchronized void synchronizedMethod() {// 同步方法体
}// 或者public void synchronizedBlock() {synchronized(this) {// 同步代码块}
}

内置锁的优点是简单易用,但缺点是粒度较粗,无法支持灵活的并发控制。

2. 显式锁(ReentrantLock)

ReentrantLock 是 Java 提供的显式锁实现,它提供了比内置锁更多的功能和灵活性。与 synchronized 相比,ReentrantLock 允许更灵活的加锁与释放锁操作,支持公平性和可中断性。

ReentrantLock lock = new ReentrantLock();lock.lock(); // 获取锁
try {// 临界区代码
} finally {lock.unlock(); // 释放锁
}

显式锁的优点是提供了更多的控制选项,适用于复杂的并发场景,但使用它需要显式地管理锁的获取和释放,容易出错。

3. 读写锁(ReentrantReadWriteLock)

ReentrantReadWriteLock 是一种特殊的锁,它分为读锁和写锁两种。多个线程可以同时持有读锁,但只有一个线程可以持有写锁。这种锁适用于读操作频繁、写操作较少的场景,可以提高并发性能。

ReentrantReadWriteLock rwLock = new ReentrantReadWriteLock();
ReadWriteLock.ReadLock readLock = rwLock.readLock();
ReadWriteLock.WriteLock writeLock = rwLock.writeLock();// 读操作
readLock.lock();
try {// 读操作代码
} finally {readLock.unlock();
}// 写操作
writeLock.lock();
try {// 写操作代码
} finally {writeLock.unlock();
}

读写锁的优点是提高了读操作的并发性能,但在写操作频繁的情况下可能导致读操作的饥饿现象。

4. 其他锁

除了上述常见的锁类型外,Java 还提供了诸如StampedLock、Condition、Semaphore 等更多的锁实现,每种锁都有其特定的使用场景和适用性。

5. 锁的选择和最佳实践

在选择锁时,需要根据具体的业务需求和性能要求来进行权衡。一般来说:

  • 如果只需要简单的同步控制,可以使用内置锁;
  • 如果需要更多的控制选项和灵活性,可以使用显式锁;
  • 如果读操作远远多于写操作,可以考虑使用读写锁;
  • 对于特定场景,还可以选择其他类型的锁。

同时,在使用锁的过程中,需要注意避免死锁、锁竞争和锁粒度过大等问题,合理设计锁的获取顺序,并尽量减少锁的持有时间,以提高程序的并发性能和可维护性。

6.举例

假设我们有一个简单的银行账户类 BankAccount,它包含账户余额和提款方法。多个线程可能同时访问同一个银行账户,我们需要确保在进行提款操作时只有一个线程能够访问账户并更新余额,以避免出现并发问题。

我们可以使用锁来控制对共享资源(即账户余额)的访问,确保在任何时候只有一个线程能够执行更新余额的操作。以下是一个使用内置锁(synchronized)的示例:

public class BankAccount {private double balance;public BankAccount(double initialBalance) {this.balance = initialBalance;}public synchronized void withdraw(double amount) {if (amount <= balance) {balance -= amount;System.out.println(Thread.currentThread().getName() + " withdraws $" + amount + ". Remaining balance: $" + balance);} else {System.out.println(Thread.currentThread().getName() + " tries to withdraw $" + amount + " but insufficient funds.");}}public double getBalance() {return balance;}
}

在这个示例中,withdraw() 方法被标记为 synchronized,这意味着只有一个线程可以同时访问该方法。当一个线程调用 withdraw() 方法时,其他线程必须等待直到当前线程执行完毕。

下面是一个简单的测试类,模拟了多个线程同时对银行账户进行提款操作:

public class Main {public static void main(String[] args) {BankAccount account = new BankAccount(1000);// 创建多个线程同时进行提款操作for (int i = 0; i < 5; i++) {Thread thread = new Thread(() -> {account.withdraw(200);});thread.start();}}
}

 

在这个例子中,即使有多个线程同时尝试提款,由于 withdraw() 方法被 synchronized 修饰,每次只有一个线程能够成功访问并更新账户余额,从而保证了线程安全性。

通过使用锁来控制对共享资源的访问,我们可以避免并发问题,确保多线程环境下程序的正确性和可靠性。

7. 结语

通过本文的介绍,我们了解了 Java 中常见的锁类型及其使用方法。锁是并发编程中重要的同步机制,合理选择和使用锁对于编写高性能、线程安全的并发程序至关重要。希望本文能够帮助读者更好地理解锁的概念和使用技巧,并在实践中运用到自己的项目中去。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/688456.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

设计usb转ttl模块的一些问题

这个是我之前设计的usb转ttl模块&#xff0c;用到的是CH340N芯片&#xff0c;目前遇到的问题以及疑问有以下几个&#xff0c;望大佬们解答&#xff1a; 1 想设计的是一块可以选择3.3V或者5V输出&#xff0c;所以我用了在TTL输出那里加了VCC、VCC3.3V、5V这几个引脚&#xff0c…

114 C++ lambda表达式捕获模式的陷阱分析和展示

一 捕获列表中的 & 捕获外部作用域中的所有变量&#xff0c;&#xff08;不包括静态变量&#xff0c;静态变量不需要捕获&#xff09;&#xff0c;并作为引用在lambda表达式中使用 按照引用这种捕获方式&#xff0c;会导致lambda表达式包含绑定到局部变量的引用。 问题发…

EXCEL中不错的xlookup函数

excel中一般要经常用vlookup函数&#xff0c;但其实经常麻烦要正序&#xff0c;从左边到右边&#xff0c;还要数列&#xff0c;挺麻烦的&#xff0c;xlookup的函数还不错&#xff0c;有个不错的一套视频介绍,B站的&#xff0c;地址是&#xff1a;XLOOKUP函数基础用法&#xff0…

rust的哈希表

新建哈希表 fn main() { use std::collections::HashMap;let mut scores HashMap::new();scores.insert(String::from("Blue"), 10);scores.insert(String::from("Yellow"), 50);println!("{:?}",scores); }访问某个元素 fn main() { use …

GB 18585-2023 壁纸中有害物质限量

壁纸/墙布因其色彩多样&#xff0c;图案丰富&#xff0c;施工方便&#xff0c;价格便宜等多种优势&#xff0c;广泛应用于室内装修材料&#xff0c;在国内&#xff0c;日本&#xff0c;欧美等地区非常普及。 GB 18585-2023壁纸中有害物质限量测试项目&#xff1a; 测试项目 测…

Eliminating Domain Bias for Federated Learning in Representation Space【文笔可参考】

文章及作者信息&#xff1a; NIPS2023 Jianqing Zhang 上海交通大学 之前中的NeurIPS23论文刚今天传到arxiv上&#xff0c;这次我把federated learning的每一轮看成是一次bi-directional knowledge transfer过程&#xff0c;提出了一种促进server和client之间bi-direction…

Day4. 文件IO的基本概念和读写

温习&#xff1a; 文件的拷贝&#xff08;单个字符&#xff09;(fgetc/fputc) #include <stdio.h>int main(void) {FILE* fp NULL;FILE* fq NULL;char ch 0;fp fopen("str.txt","r");if (fp NULL){perror("file to fopen!");retur…

网络模型及传输基本流程

1.OSI 七层模型 OSI &#xff08; Open System Interconnection &#xff0c;开放系统互连&#xff09;七层网络模型称为开放式系统互联参考模型&#xff0c;是一个逻辑上的定义和规范; 把网络从逻辑上分为了 7 层 . 每一层都有相关、相对应的物理设备&#xff0c;比如路由器…

RCS系统之:冲突解决

在RCS系统中&#xff0c;避免碰撞是至关重要的。以下是一些常见的方法和技术用于避免碰撞&#xff1a; 障碍物检测&#xff1a;机器人可以配备各种传感器&#xff0c;如激光雷达、超声波传感器、摄像头等&#xff0c;用于检测周围的障碍物和环境。通过实时监测周围情况&#xf…

插值与拟合算法介绍

在数据处理和科学计算领域,插值与拟合是两种极为重要的数据分析方法。它们被广泛应用于信号处理、图像处理、机器学习、金融分析等多个领域,对于理解和预测数据趋势具有至关重要的作用。本文将深入浅出地介绍这两种算法的基本原理,并结合C语言编程环境探讨如何在CSDN开发者社…

力扣:139. 单词拆分

动态规划&#xff1a; 1.先声明dp数组的含义为下标i表示的是在s变量中i前面的字符串是否在wordDict变量中存在&#xff0c;初始化dp【0】来进行后面dp数组的递推。同时要判断截取的值是否在wirdDict中是否存在&#xff0c;还要判断dp【j】的下标的j前面的字符串是否也在wirdDi…

数组常见算法代码总结

一、数组排序【冒泡排序】&#xff08;优化&#xff09; 1.基本实现思路: 数组排序是通过冒泡排序算法实现的&#xff0c;基本实现思路是比较相邻的元素&#xff0c;把小的元素往前移动或把大的元素往后移动&#xff0c;相邻元素两两进行比较&#xff0c;若大元素在小元素前面…

RocketMQ订阅关系不一致和不能消费时如何排查?

订阅关系不一致 调整任意一个实例的订阅关系和另一个保持一致 消费者不能消费消息 它是最常见的问题之一&#xff0c;也是每个消息队列服务都会遇到的问题 1.确认哪个消息未消费。在这时消费者至少需要手机消息id、消息key、消息发送时间段三者之一 2.确认消息是否发送成功…

HashMap使用静态初始化块添加元素

使用静态初始化块&#xff08;Static Initialization Block&#xff09; 示例代码&#xff1a; import java.util.HashMap; import java.util.Map;public class Main {public static void main(String[] args) {// 使用静态初始化块批量添加元素Map<String, Integer> h…

JVM--- 垃圾收集器详细整理

目录 一、垃圾收集需要考虑的三个事情&#xff1a; 二、垃圾回收针对的区域 三、如何判断对象已死 1.引用计数算法&#xff1a; 2.可达性分析算法 四、引用 五、生存还是死亡&#xff1f; 六、回收方法区 七、垃圾收集算法 1.分代收集理论 2.标记-清除算法 3.标记-复制算…

huggingface库LocalTokenNotFoundError:需要提供token

今天刚开始学习huggingface&#xff0c;跑示例的时候出了不少错&#xff0c;在此记录一下&#xff1a; (gpu) F:\transformer\transformers\examples\pytorch\image-classification>.\run.bat Traceback (most recent call last):File "F:\transformer\transformers\e…

一站式安装对应显卡版本的cuda和torch(windows)

前言 一年前&#xff0c;安装过cuda&#xff0c;觉得并不难&#xff0c;就没有记录。 这次安装还算顺利&#xff0c;就是在找资料的时候&#xff0c;浪费了不少时间 这次就记录下来&#xff0c;方便以后再次安装 总结安装程序&#xff1a; 1、安装python环境 2、安装VS的C环境&…

【机构vip教程】Unittest(1):unittest单元测试框架简介

unittest单元测试框架简介 unittest是python内置的单元测试框架&#xff0c;具备编写用例、组 织用例、执行用例、功能&#xff0c;可以结合selenium进行UI自动化测 试&#xff0c;也可以结合appium、requests等模块做其它自动化测试 官方文档&#xff1a;https://docs.pytho…

机试指南:3-4章

文章目录 第3章 排序与查找(一) 排序1.sort函数&#xff1a;sort(first,last,comp)2.自定义比较规则3.C函数重载&#xff1a;同一个函数名&#xff0c;有不同的参数列表4.机试考试最重要的事情&#xff1a;能把你曾经做过的题目&#xff0c;满分地做出来5.例题例题1&#xff1a…

xtu oj 1162 奇偶校验

题目描述 奇偶校验是一种在通讯中经常使用的&#xff0c;用来确认传输的字节是否正确的手段。 对于一个BYTE(8BIT),我们使用0~6bit来存储数据&#xff0c;称为数据位&#xff0c;第7位存储奇偶校验位。 如果数据位有偶数个1&#xff0c;那么第7位为0&#xff0c;否则为1。现给…