随机名言

CAS方法的使用



最近忙着复习,笔记倒是写了很多但没有整理出来,后期抽时间统一上传


1. CAS

在并发编程下能经常看到CAS,全名Compare and Swap(比较和交换)。是JDK提供的非阻塞原子性操作,它通过硬件保证了比较-交换这个操作的原子性,主要是处理器级别提供了原子性操作。和重量级锁(Synchronized)对比,免去了线程上下文切换的开销,是个不错的轻量级锁


实现原理:该方法有四个参数,分别是对象内存位置,对象中变量的偏移量,变量预期值,变量更新值。如果对象obj内存偏移量为offset的变量的值为expect,则使用update替换旧的值expect




2. Unsafe

JDK的Unsafe类提供了一系列的compareAndSwap*方法,下面列出这次用到的方法

  • long objectFieldOffet(Field file) 获取字段的偏移量
  • boolean compareAndSwaoInt(Object obj,long offset,int expect,int update) CAS方法

因为Unsafe是个直接操作内存的不安全类,所以JDK规定只有BootStrap类加载器加载的类才能使用Unsafe类,比如rt.jar包下的类就可以。那么我们自定义的使用Application加载器加载的类怎么使用Unsafe呢?答案是:万能的反射,方法如下


import sun.misc.Unsafe;
import java.lang.reflect.Field;

public class UnsafeUtil {

    public static Unsafe getInstance() throws NoSuchFieldException, IllegalAccessException {
        
        // 获取字段
        Field filed = Unsafe.class.getDeclaredField("theUnsafe");
        
        // 设置可访问
        filed.setAccessible(true);
        
        // 返回Unsafe实例
        return (Unsafe)filed.get(null);
    }
}

注意:使用IDEA方便,而eclipse不能直接导入import sun.misc.Unsafe,需要自行解决


import sun.misc.Unsafe;
import java.lang.reflect.Field;

public class Test {

	// 需要CAS修改的字段
    private int i = 0;

    public int getI() {
        return i;
    }

    public void setI(int i) {
        this.i = i;
    }


    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {

        // 获取Unsafe实例
        Unsafe  unsafe = UnsafeUtil.getInstance();

        // 获取字段 i 的偏移量
        Test test = new Test();
        Field field = test.getClass().getDeclaredField("i");
        long offset = unsafe.objectFieldOffset(field);

        // CAS修改,返回布尔值
        boolean isSuccess = unsafe.compareAndSwapInt(test,offset,0,1);

        // 打印
        System.out.println(isSuccess);
        System.out.println(test.getI());

    }
}

打印

true
1


《Java并发编程之美》

posted @ 2020-04-03 22:33  Howlet  阅读(845)  评论(0编辑  收藏  举报

Copyright © By Howl