多线程(50)如何实现自旋锁

自旋锁是一种忙等锁,当线程尝试获取锁而锁已被其他线程持有时,该线程会在一个循环中不断尝试获取锁,直到成功为止。与传统的互斥锁相比,自旋锁不会使线程进入睡眠状态,因此如果等待锁的时间非常短,自旋锁的性能可能会更好。但是,如果锁被长时间持有,自旋锁会浪费大量CPU资源。

下面是一个使用Java代码实现的简单自旋锁示例。请注意,该示例仅用于教育目的,实际应用中应使用Java的java.util.concurrent.locks.Lock接口或其他并发工具类,因为它们提供了更完善、更可靠的锁实现。

简单的自旋锁实现

我们将使用AtomicBoolean来实现自旋锁。AtomicBoolean类提供了一种线程安全的布尔值操作方式,其内部使用了无锁的比较并交换(CAS)操作,非常适合用于实现自旋锁。

import java.util.concurrent.atomic.AtomicBoolean;public class SpinLock {private final AtomicBoolean lock = new AtomicBoolean(false);/*** 尝试获取锁,如果锁已被其他线程持有,则持续尝试*/public void lock() {while (!lock.compareAndSet(false, true)) {// 循环尝试获取锁,直到成功为止// 注意:在高并发情况下可能会导致大量CPU资源浪费}}/*** 释放锁*/public void unlock() {lock.set(false);}public static void main(String[] args) {SpinLock spinLock = new SpinLock();// 线程1new Thread(() -> {spinLock.lock();try {System.out.println("Thread 1 acquired the lock");Thread.sleep(1000); // 模拟执行任务} catch (InterruptedException e) {Thread.currentThread().interrupt();} finally {spinLock.unlock();System.out.println("Thread 1 released the lock");}}).start();// 线程2new Thread(() -> {spinLock.lock();try {System.out.println("Thread 2 acquired the lock");} finally {spinLock.unlock();System.out.println("Thread 2 released the lock");}}).start();}
}

在这个示例中,lock()方法使用了一个循环,不断尝试通过compareAndSet()方法将lock变量从false设置为true。只有当compareAndSet()返回true时,当前线程才成功获得锁。unlock()方法则简单地将lock变量置回false以释放锁。

注意事项

  1. 性能问题:自旋锁在锁持有时间非常短且线程竞争不激烈的场景下效率较高,但如果锁被长时间持有,它会导致大量的CPU时间被浪费在无效的锁请求上。
  2. 公平性:上述简单的自旋锁实现不是公平的,即没有考虑请求锁的顺序。在竞争激烈的情况下,某些线程可能会饥饿。
  3. 实际使用:在实际应用中,建议使用Java标准库中提供的锁和并发工具,如ReentrantLock,它们提供了更高级的功能,比如可重入性、公平性选择和条件变量支持。

自旋锁是对高性能并发程序设计的一种基本构建块,正确使用它们可以在特定场景下显著提升性能。然而,设计高效且正确的并发控制机制需要深入理解底层原理和应用场景。

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

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

相关文章

LWIP2.1.3+UCOSIII3.08的系统接口arch.c修改

自己的笔记不保证正确 err_t sys_mbox_trypost_fromisr(sys_mbox_t *mbox, void *msg) {BaseType_t ret;BaseType_t xHigherPriorityTaskWoken pdFALSE;LWIP_ASSERT("mbox ! NULL", mbox ! NULL); // LWIP_ASSERT("mbox->mbx ! NULL", mbox->mbx …

C#WPF的XAML中String回车换行

本文实例演示C#WPF的XAML中String回车换行。 在XAMl中回车换行与C#中的不同,使用\r\n不再起作用。 首先使用String需要先添加引用 xmlns:sys="clr-namespace:System;assembly=mscorlib" 回车使用
或者 换行使用
或者 同时使用时需要添加 xml:…

mac ip 域名 三者之间的关系

mac ip 域名 三者之间的关系 在计算机网络中,MAC地址(Media Access Control Address)、IP地址(Internet Protocol Address)和域名(Domain Name)是三个不同的概念,它们之间有以下关系…

Linux进阶篇:文件传输工具curl命令详解

文件传输工具Linux curl命令详解 一 curl命令介绍 在Linux中curl是一个利用URL规则在命令行下工作的文件传输工具,可以说是一款很强大的http命令行工具。它支持文件的上传和下载,是综合传输工具,但按传统,习惯称url为下载工具。…

package.java文件的作用

你查看springboot的源码,有很多类都有这个文件,在idea不能创建,因为不支持这种命名,只能用记事本创建后复制都项目中。 主要应用是给类添加正常,或者把公用的注解都放到这里,常量不合适,作用范…

在Qt平台上的网络应用编程原理

前言 在网络通信方面的应用编程需要使用套接字(Socket),如在构建网站的服务器、游戏的服务器时。Qt提供了跨平台的类库QTcpServer、QTcpSocket及QUdpSocket供程序员使用,具体用途如下。 QTcpServer用于传输控制协议/网际协议(Transmission Control Proto…

竞赛 基于CNN实现谣言检测 - python 深度学习 机器学习

文章目录 1 前言1.1 背景 2 数据集3 实现过程4 CNN网络实现5 模型训练部分6 模型评估7 预测结果8 最后 1 前言 🔥 优质竞赛项目系列,今天要分享的是 基于CNN实现谣言检测 该项目较为新颖,适合作为竞赛课题方向,学长非常推荐&am…

欧姆龙61F系列液位开关使用教程(补水和排水)

欧姆龙61F系列液位开关使用教程(补水和排水) 本文以61F-LS-CP11-NRA型号的液位开关为例进行说明: 具体的选型文档可参考以下链接中的内容: OMRON欧姆龙-无浮标开关(紧凑插入型)61F-LS液位开关-选型样本说明 补水功能(供水) 如下图所示, 电机电源为3相AC220V; 控制电…

SSRF+Redis未授权getshell

SSRFRedis未授权getshell 1.前言 当一个网站具有ssrf漏洞,如果没有一些过滤措施,比如没过滤file协议,gophere协议,dict等协议,就会导致无法访问的内网服务器信息泄露,甚至可以让攻击者拿下内网服务器权限 …

Git分布式版本控制系统——Git常用命令(二)

五、Git常用命令————分支操作 同一个仓库可以有多个分支,各个分支相互独立,互不干扰 分支的相关命令,具体如下: git branch 查看分支 git branch [name] 创建分支&#x…

5. Mysql的binlog介绍

参考:InnoDB学习(三)之BinLog 1. BinLog介绍 BinLog又称为二进制日志,是MySQL服务层的数据日志,MySQL所有的存储引擎都支持BinLog。 BinLog记录了MySQL中的数据更新和可能导致数据更新的事件,可以用于主从…

轻量带屏解决方案之恒玄芯片移植案例

本文章基于恒玄科技BES2600W芯片的欧智通 Multi-modal V200Z-R开发板 ,进行轻量带屏开发板的标准移植,开发了智能开关面板样例,同时实现了ace_engine_lite、arkui_ui_lite、aafwk_lite、appexecfwk_lite、HDF等部件基于OpenHarmony LiteOS-M内…

Flume配置案例@Source:Kafka,Channel:File,Sink:HDFS

创建flume配置文件 [atguiguhadoop104 flume]$ vim job/kafka_to_hdfs_log.conf 配置内容如下: --------------------- #定义组件 a1.sourcesr1 a1.channelsc1 a1.sinksk1 #配置source1 a1.sources.r1.type org.apache.flume.source.kafka.KafkaSource a1…

【联机不卡顿】幻兽帕鲁教你如何低成本0延迟畅玩 云服务器性价比选择方案 16G低至26/月

更新日期:4月14日(腾讯云16G价格回调了!京东云采购季持续进行) 本文纯原创,侵权必究 《最新对比表》已更新在文章头部—腾讯云文档,文章具有时效性,请以腾讯文档为准! 【腾讯文档实…

算法:位运算

算法&#xff1a;位运算 常见位运算操作基本题型模拟加法数字查找总结 常见位运算操作 在C/C中&#xff0c;有比较丰富的位运算操作符&#xff0c;常见的有&#xff1a; &&#xff1a;按位与 |&#xff1a;按位或 ~&#xff1a;按位取反 ^&#xff1a;按位异或 <<&a…

MySQ数据库: MySQL数据库的安装配置 ,图文步骤详细,一篇即可完成安装完成! MySQL数据库如何与客户端连接

LiuJinTao&#xff1a; 2024年4月14日 文章目录 MySQL的安装配置1. 下载2. 安装 三、 MySQL 启动与停止1. 第一种 方式&#xff1a;2. 第二种方式&#xff1a; 四、MySQL 客户端连接2. 方式二&#xff1a; MySQL的安装配置 1. 下载 官方下载网址&#xff1a;https://www.mysq…

代码随想录刷题随记21-回溯1

代码随想录刷题随记21-回溯1 回溯法解决的问题 回溯法&#xff0c;一般可以解决如下几种问题&#xff1a; 组合问题&#xff1a;N个数里面按一定规则找出k个数的集合 切割问题&#xff1a;一个字符串按一定规则有几种切割方式 子集问题&#xff1a;一个N个数的集合里有多少符…

可视化大屏C位图:​地理信息—地球焦点图

Hello&#xff0c;我是大千UI工场&#xff0c;本期可视化大屏的焦点图&#xff08;C位&#xff09;分享将地球作为焦点图的情形&#xff0c;欢迎友友们关注、评论&#xff0c;如果有订单可私信。 将地球作为可视化大屏焦点图可以有以下几个作用&#xff1a; 全球数据展示&…

SQL server 非聚集索引

CREATE NONCLUSTERED INDEX idx_dwd_kuoutlist2_dt_kuout_kg ON [dwd_kuoutlist2] (dt, kuout_kg,id) 这条SQL语句是在SQL Server中创建一个非聚集索引。具体含义如下&#xff1a; 1. **CREATE NONCLUSTERED INDEX**&#xff1a; - 这部分表明正在创建一个非聚集索引。非聚…

蓝桥杯嵌入式(G431)备赛笔记——DMA+ADC(单通道+多通道)

单通道&#xff1a; 开启循环模式&#xff0c;两个参数设为word u32 adc_tick0; u32 r37_value0; u32 r38_value0; float r37_volt0; float r38_volt0;//DMAADCvoid DMA_ADC() {if(uwTick-adc_tick<100) return;adc_tick uwTick;HAL_ADC_Start_DMA(&hadc2, &r37_v…