zoukankan      html  css  js  c++  java
  • 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并发编程之美》

  • 相关阅读:
    Java经典习题7
    Java经典习题6
    java经典习题5
    前后端分离开发——模拟数据mock.js
    微信网页第三方登录原理
    TP5常量
    TP5
    健忘的正则
    JS正则
    apache配置修改
  • 原文地址:https://www.cnblogs.com/Howlet/p/12629408.html
Copyright © 2011-2022 走看看