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

相关文章

Java第二阶段---09类和对象---第三节 构造方法

第三节 构造方法 1.概念 构造方法是一种特殊的方法,主要用于创建对象以及完成对象的属性初始化操作。构造方法不能被对象调用。 2.语法 //[]中内容可有可无 访问修饰符 类名([参数列表]){ } 3.示例 public class Car {     //车特征(属性)     public String name;//车名   可以直接拿来用 说明它有初始值     pu

动手学深度学习【数据操作+数据预处理】

import osos.makedirs(os.path.join('.', 'data'), exist_ok=True)data_file = os.path.join('.', 'data', 'house_tiny.csv')with open(data_file, 'w') as f:f.write('NumRooms,Alley,Price\n') # 列名f.write('NA

线程的四种操作

所属专栏:Java学习        1. 线程的开启 start和run的区别: run:描述了线程要执行的任务,也可以称为线程的入口 start:调用系统函数,真正的在系统内核中创建线程(创建PCB,加入到链表中),此处的start会根据不同的系统,分别调用不同的api,创建好之后的线程,再单独去执行run(所以说,start的本质是调用系统api,系统的api

Java IO 操作——个人理解

之前一直Java的IO操作一知半解。今天看到一个便文章觉得很有道理( 原文章),记录一下。 首先,理解Java的IO操作到底操作的什么内容,过程又是怎么样子。          数据来源的操作: 来源有文件,网络数据。使用File类和Sockets等。这里操作的是数据本身,1,0结构。    File file = new File("path");   字

MySQL——表操作

目录 一、创建表 二、查看表 2.1 查看表中某成员的数据 2.2 查看整个表中的表成员 2.3 查看创建表时的句柄 三、修改表 alter 3.1 重命名 rename 3.2 新增一列 add 3.3 更改列属性 modify 3.4 更改列名称 change 3.5 删除某列 上一篇博客介绍了库的操作,接下来来看一下表的相关操作。 一、创建表 create

HTML5自定义属性对象Dataset

原文转自HTML5自定义属性对象Dataset简介 一、html5 自定义属性介绍 之前翻译的“你必须知道的28个HTML5特征、窍门和技术”一文中对于HTML5中自定义合法属性data-已经做过些介绍,就是在HTML5中我们可以使用data-前缀设置我们需要的自定义属性,来进行一些数据的存放,例如我们要在一个文字按钮上存放相对应的id: <a href="javascript:" d

封装MySQL操作时Where条件语句的组织

在对数据库进行封装的过程中,条件语句应该是相对难以处理的,毕竟条件语句太过于多样性。 条件语句大致分为以下几种: 1、单一条件,比如:where id = 1; 2、多个条件,相互间关系统一。比如:where id > 10 and age > 20 and score < 60; 3、多个条件,相互间关系不统一。比如:where (id > 10 OR age > 20) AND sco

PHP7扩展开发之对象方式使用lib库

前言 上一篇文章,我们使用的是函数方式调用lib库。这篇文章我们将使用对象的方式调用lib库。调用代码如下: <?php $hello = new hello(); $result = $hello->get(); var_dump($result); ?> 我们将在扩展中实现hello类。hello类中将依赖lib库。 代码 基础代码 这个扩展,我们将在say扩展上增加相关代码。sa

PHP7扩展开发之流操作

前言 啥是流操作?简单来讲就是对一些文件,网络的IO操作。PHP已经把这些IO操作,封装成流操作。这节,我们将使用PHP扩展实现一个目录遍历的功能。PHP示例代码如下: <?phpfunction list_dir($dir) {if (is_dir($dir) === false) {return;} $dh = opendir($dir);if ($dh == false) {ret

浙大数据结构:树的定义与操作

四种遍历 #include<iostream>#include<queue>using namespace std;typedef struct treenode *BinTree;typedef BinTree position;typedef int ElementType;struct treenode{ElementType data;BinTree left;BinTre