java unsafe 详解_Java CAS操作与Unsafe类详解

一、复习

计算机内存模型,synchronized和volatile关键字简介

二、两者对比

sychronized和volatile都解决了内存可见性问题

不同点:

(1)前者是独占锁,并且存在者上下文切换的开销以及线程重新调度的开销;后者是非阻塞算法,不会造成上下文切换的开销。

(2)前者可以保证操作的原子性,但是后者不能保证操作的原子性。

三、在什么情况下才会使用volatile

写入变量是不依赖当前值的,如果是依赖当前值的话,由于获取-计算-写入,三者不是原子性操作,而volatile是保证原子性操作的。

变量没有加锁的时候,如果变量加锁了,是可以保证内存的可见性的因此不需要再使用volatile

四、Java中的原子性操作

原子性操作通俗的来讲就是一组操作,要么都执行成功,要么都执行失败,不存在执行部分成功的情况

使用synchronized关键字既可以保证操作的原子性又可以保证内存的可见性,volatile只能保证内存的可见性,但是不能保证操作的原子性;synchronized固然好,但在高并发的情况下,由于它是一种独占锁,因此会引起性能低下的问题。

五、Java中的CAS操作

定义:CAS(compare and swap)比较并交换,这是JDK提供的一种非阻塞算法,它通过硬件保证了比较-更新的原子性问题。JDK中的Unsafe类提供了一系列的compareAndSwap*方法,下面以compareAndSwapLong为例进行讲解

boolean compare(Object obj,long offset,long expect,long update)

先分别解释一下各个参数,obj是一个对象的引用(也就是对象存储的地址),offset是相对于前面地址的偏移量,expect是一个预想的值,update代表如果和预想的值一样,那么就是使用update这个值来代替,并且返回true,否则返回false

这是处理器提供的一种原子性指令

六、ABA问题

描述:线程1获取变量x的值为A,然后尝试修改为B,但是此时如果有另一个线程修改了x的值为B,同时又修改成了A,那么线程2的这个A和线程1之前的A就不是同一个A了

产生原因:环形依赖,变量的值从A到B,然后又从B到A,这样只能一个方向轮转,如果是从A到B,然后从B到C就不会出现这种情况。

解决方式:JDK中的AtomicStampedReferece给每个变量一个时间戳,从而避免了ABA问题

七、Unsafe类

在JDK中的rt.jar包中有许多方法都是native的,这是一种硬件级别的操作,使用JNI来调用C++底层函数来操作。

1.long objectFieldOffset(Field field)

释义:获取某个对象的中的某个域值所在对象的中的内存偏移量

try{

longvalue=Unsafe.objectFieldOffset(AutomicLong.class.getDeclaredField("value"));

}catch(Exceptione){

e.printStackTrace();

}

2.int arrayBaseOffset(Class arrayClass)

释义:获取数组中的第一个元素地址

3.int arrayIndexOffset(Class arrayClass)

释义:获取数组中第一个元素的字节大小

4.boolean compareAndSwapLong(Object obj,long offset,long expect,long update)

可以见上文

八、源码:

所在包:com.ruigege.OtherFoundationOfConcurrent2

https://github.com/ruigege66/ConcurrentJava

以上就是Java CAS操作与Unsafe类详解的详细内容,更多关于Java CAS操作与Unsafe类的资料请关注聚米学院其它相关文章!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
UnsafeJava中一个非常特殊的,它提供了对底层操作的支持,包括内存操作、线程调度等。使用Unsafe可以实现一些Java中无法实现的操作,并且可以提高代码的性能。 CAS(Compare and Swap)操作Unsafe中非常常见的操作之一。它的作用是比较当前值和期望值是否相等,如果相等,则将新值更新到当前值中,否则不做任何操作CAS操作通常被用于多线程编程中的数据同步,以保证多个线程对共享变量的操作是正确的。 下面是使用Unsafe进行CAS操作的一个示例: ``` import sun.misc.Unsafe; import java.lang.reflect.Field; public class CASDemo { private static Unsafe unsafe; static { try { // 使用反射获取Unsafe Field field = Unsafe.class.getDeclaredField("theUnsafe"); field.setAccessible(true); unsafe = (Unsafe) field.get(null); } catch (Exception e) { e.printStackTrace(); } } public static void main(String[] args) { // 假设初始值为10 int expectedValue = 10; // 将初始值存储在内存地址100的位置 long offset = unsafe.objectFieldOffset(CASDemo.class.getDeclaredFields()[0]); // 使用CAS操作将值从10更新为20 boolean casResult = unsafe.compareAndSwapInt(new CASDemo(), offset, expectedValue, 20); System.out.println(casResult); } private int value = 10; } ``` 在上面的代码中,首先通过反射获取Unsafe的实例,然后将初始值存储在的某个字段中,接着使用`unsafe.compareAndSwapInt()`方法进行CAS操作,将值从10更新为20。这个方法的第一个参数是对象实例,第二个参数是字段的内存地址偏移量,第三个参数是期望值,第四个参数是更新值。如果更新成功,该方法返回true,否则返回false。 需要注意的是,Unsafe使用需要非常小心,因为它可以直接操作内存,容易引发安全问题。在使用Unsafe的时候,应该先了解清楚相关的API文档,并且在实际应用中要进行严格的测试和验证。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值