Unsafe-CAS操作一个对象的status值

2024-03-26 15:04
文章标签 操作 对象 cas status unsafe

本文主要是介绍Unsafe-CAS操作一个对象的status值,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

如下代码主要展示了如何使用 CAS(Compare and Swap)操作来实现多线程并发控制。

在这个例子中,CAS 主要用于修改对象中的 statusname 字段。

在 Java 中,CAS 操作是通过 Unsafe 类来实现的,它可以直接操作对象的内存,而不受 Java 内存模型的限制。在这段代码中,通过 Unsafe 类获取了 statusname 字段在对象中的偏移量 statusOffsetnameOffSet,然后通过 CAS 操作来修改这些字段的值。

为什么要通过偏移量来进行 CAS 操作呢?这是因为 CAS 操作是基于对象内存地址进行的,而不是对象的引用或名称。通过偏移量,可以直接确定对象内部字段在内存中的位置,从而正确地执行 CAS 操作。

在这段代码中,通过获取 statusname 字段在对象中的偏移量,可以在多线程环境下安全地对这些字段进行修改,而不会出现数据竞争或错误修改的情况。

需要注意的是,直接操作对象内存是一种底层的编程技巧,需要谨慎使用,并且在高级应用中建议使用更高级的并发工具和模式来实现并发控制。


public class CASTest {private volatile int status;private static final Unsafe unsafe;// 对于 Java AbstractQueuedSynchronizer (AQS) 中的 stateOffset 字段,// 通常会使用 static final 修饰是因为这个偏移量值在整个类中都是不变的,且所有实例都需要使用相同的偏移量来进行 CAS 操作。// 即:不管有多少个 Bean 对象,对于Bean对象中的status操作的偏移量值是固定的,可以使用 static final// statusOffset 只是一个相对的偏移量,跟status具体的值无关,所以可以多实例共享statusOffsetprivate static final long statusOffset;private volatile String name;private static final long nameOffSet;static {try {// sun.misc 中的class无法像AQS中那样通过 private static final Unsafe unsafe = Unsafe.getUnsafe(); 直接调用Field field = Unsafe.class.getDeclaredField("theUnsafe");field.setAccessible(true);unsafe = (Unsafe) field.get(null);statusOffset = unsafe.objectFieldOffset(CASTest.class.getDeclaredField("status"));nameOffSet = unsafe.objectFieldOffset(CASTest.class.getDeclaredField("name"));} catch (Exception e) {throw new RuntimeException(e);}}public static void main(String[] args) {CASTest casTest = new CASTest();System.out.println("statusOffset:" + CASTest.statusOffset);System.out.println("status:" + casTest.status);System.out.println("name:" + casTest.name);System.out.println("nameOffSet:" + CASTest.nameOffSet);// 多线程并发cas操作status 0->1for (int i = 0; i < 10; i++) {int finalI = i;new Thread(() -> {boolean nameSetSuccess = unsafe.compareAndSwapObject(casTest, nameOffSet, null, "张三");if (nameSetSuccess) {System.out.println(finalI + "==============-name:===============" + casTest.name);System.out.println(finalI + "============-nameOffSet:================" + nameOffSet);} else {System.out.println(finalI + "-name:" + casTest.name);System.out.println(finalI + "-nameOffSet:" + nameOffSet);}boolean success = unsafe.compareAndSwapInt(casTest, statusOffset, 0, 1);if (success) {System.out.println(finalI + "=============-statusOffset:===========" + statusOffset);System.out.println(finalI + "=============-status:=============" + casTest.status);} else {System.out.println(finalI + "-statusOffset:" + statusOffset);System.out.println(finalI + "-status:" + casTest.status);}}).start();}}
}

这段代码主要展示了如何使用 CAS(Compare and Swap)操作来实现多线程并发控制,并通过偏移量来操作对象中的字段。下面是对代码的分析及流程讲解:

  1. 首先定义了一个 CASTest 类,其中包含了一个 volatile int status 和一个 volatile String name 字段,以及相应的静态变量 Unsafe unsafelong statusOffsetlong nameOffSet

  2. 在静态代码块中,通过反射的方式获取 Unsafe 对象,并获取了 statusname 字段在内存中的偏移量 statusOffsetnameOffSet

  3. main 方法中,创建了一个 CASTest 对象 casTest,并打印了 statusOffsetstatusnamenameOffSet 的值。

  4. 接着使用了一个循环创建了10个线程,每个线程内部执行了以下操作:

    • 使用 Unsafe.compareAndSwapObject 方法尝试将 name 字段由 null 修改为 "张三",并根据操作结果输出相关信息。
    • 使用 Unsafe.compareAndSwapInt 方法尝试将 status 字段由 0 修改为 1,并根据操作结果输出相关信息。
  5. 在多线程并发操作中,每个线程都尝试使用 CAS 操作来修改 namestatus 字段的值。由于 CAS 是一种乐观锁的机制,会比较对象当前值和期望值,如果一致则进行更新,否则不会更新并返回失败。

  6. 通过输出结果可以观察到多线程并发时对 namestatus 字段的操作情况,以及 CAS 操作的成功与失败情况。

总体来说,这段代码演示了如何使用 CAS 操作和偏移量来实现多线程并发控制,尤其是在底层操作对象内存时的一种实践。需要注意的是,对于普通业务开发而言,推荐使用更高级的并发工具和模式来确保数据安全性和可靠性。

结果展示:

这篇关于Unsafe-CAS操作一个对象的status值的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/848897

相关文章

python panda库从基础到高级操作分析

《pythonpanda库从基础到高级操作分析》本文介绍了Pandas库的核心功能,包括处理结构化数据的Series和DataFrame数据结构,数据读取、清洗、分组聚合、合并、时间序列分析及大数据... 目录1. Pandas 概述2. 基本操作:数据读取与查看3. 索引操作:精准定位数据4. Group

Python操作PDF文档的主流库使用指南

《Python操作PDF文档的主流库使用指南》PDF因其跨平台、格式固定的特性成为文档交换的标准,然而,由于其复杂的内部结构,程序化操作PDF一直是个挑战,本文主要为大家整理了Python操作PD... 目录一、 基础操作1.PyPDF2 (及其继任者 pypdf)2.PyMuPDF / fitz3.Fre

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

MySQL 强制使用特定索引的操作

《MySQL强制使用特定索引的操作》MySQL可通过FORCEINDEX、USEINDEX等语法强制查询使用特定索引,但优化器可能不采纳,需结合EXPLAIN分析执行计划,避免性能下降,注意版本差异... 目录1. 使用FORCE INDEX语法2. 使用USE INDEX语法3. 使用IGNORE IND

Python使用openpyxl读取Excel的操作详解

《Python使用openpyxl读取Excel的操作详解》本文介绍了使用Python的openpyxl库进行Excel文件的创建、读写、数据操作、工作簿与工作表管理,包括创建工作簿、加载工作簿、操作... 目录1 概述1.1 图示1.2 安装第三方库2 工作簿 workbook2.1 创建:Workboo

Ubuntu 24.04启用root图形登录的操作流程

《Ubuntu24.04启用root图形登录的操作流程》Ubuntu默认禁用root账户的图形与SSH登录,这是为了安全,但在某些场景你可能需要直接用root登录GNOME桌面,本文以Ubuntu2... 目录一、前言二、准备工作三、设置 root 密码四、启用图形界面 root 登录1. 修改 GDM 配

JSONArray在Java中的应用操作实例

《JSONArray在Java中的应用操作实例》JSONArray是org.json库用于处理JSON数组的类,可将Java对象(Map/List)转换为JSON格式,提供增删改查等操作,适用于前后端... 目录1. jsONArray定义与功能1.1 JSONArray概念阐释1.1.1 什么是JSONA

Java操作Word文档的全面指南

《Java操作Word文档的全面指南》在Java开发中,操作Word文档是常见的业务需求,广泛应用于合同生成、报表输出、通知发布、法律文书生成、病历模板填写等场景,本文将全面介绍Java操作Word文... 目录简介段落页头与页脚页码表格图片批注文本框目录图表简介Word编程最重要的类是org.apach

Python实现对阿里云OSS对象存储的操作详解

《Python实现对阿里云OSS对象存储的操作详解》这篇文章主要为大家详细介绍了Python实现对阿里云OSS对象存储的操作相关知识,包括连接,上传,下载,列举等功能,感兴趣的小伙伴可以了解下... 目录一、直接使用代码二、详细使用1. 环境准备2. 初始化配置3. bucket配置创建4. 文件上传到os

mysql表操作与查询功能详解

《mysql表操作与查询功能详解》本文系统讲解MySQL表操作与查询,涵盖创建、修改、复制表语法,基本查询结构及WHERE、GROUPBY等子句,本文结合实例代码给大家介绍的非常详细,感兴趣的朋友跟随... 目录01.表的操作1.1表操作概览1.2创建表1.3修改表1.4复制表02.基本查询操作2.1 SE