关于volatile修饰的成员变量的性能问题

前言:

  今天偶然看到ConcurrentLinkedQueue的内部类ConcurrentLinkedQueue.Node, 发现其两个成员变量item, next都是由volatile进行修饰, 而且赋值都是用sun的misc包下UNSAFE类进行的。我非常好奇,为啥会有这种操作?(我个人估计是性能考虑), 下面就进行了实验。

UNSAFE类意义:

  sun.misc.Unsafe是JDK内部用的工具类。它通过暴露一些Java意义上说“不安全”的功能给Java层代码,来让JDK能够更多的使用Java代码来实现一些原本是平台相关的、需要使用native语言(例如C或C++)才可以实现的功能。该类不应该在JDK核心类库之外使用。

  JVM的实现可以自由选择如何实现Java对象的“布局”,也就是在内存里Java对象的各个部分放在哪里,包括对象的实例字段和一些元数据之类。sun.misc.Unsafe里关于对象字段访问的方法把对象布局抽象出来,它提供了objectFieldOffset()方法用于获取某个字段相对Java对象的“起始地址”的偏移量,也提供了getInt、getLong、getObject之类的方法可以使用前面获取的偏移量来访问某个Java对象的某个字段。

  Oracle/Sun HotSpot VM所使用的Unsafe对象可以参考这篇博客:http://mishadoff.com/blog/java-magic-part-4-sun-dot-misc-dot-unsafe/

实验:

1. unsafe.putObject(this, itemOffset, item)
   直接在对象的itemOffset位置设置item的引用(越过访问权限)

2. unsafe.putOrderedObject(this, nextOffset, val)
   在对象的itemOffset位置设置item的引用, 只不过这个方法的可见性比直接用 putObject 方法低一点, 其他的线程要过一段时间才可见

putOrderedObject 使用 store-store barrier屏障, 而 putObject还会使用 store-load barrier 屏障(对于Java中的指令屏障不了解的直接可以参考 Java并发编程艺术)

  我参照ConcurrentLinkedQueue的内部类ConcurrentLinkedQueue.Node类写了个类似的类,如下:

class UnsafeNode<T> {
    volatile T item;
    volatile UnsafeNode<T> next;
    static final sun.misc.Unsafe UNSAFE;
    static final long itemOffset;
    static final long nextOffset;

    static {
        try {
            UNSAFE = sun.misc.Unsafe.getUnsafe();
            Class<?> klass = UnsafeNode.class;
            itemOffset = UNSAFE.objectFieldOffset
                    (klass.getDeclaredField("item"));
            nextOffset = UNSAFE.objectFieldOffset
                    (klass.getDeclaredField("next"));
        } catch (Exception e) {
            throw new Error(e);
        }
    }

    UnsafeNode(T item) {
        UNSAFE.putObject(this, itemOffset, item);
    }

    boolean casItem(T cmp, T val) {
        return UNSAFE.compareAndSwapObject(this, itemOffset, cmp, val);
    }

    void lazySetNext(UnsafeNode<T> val) {
        UNSAFE.putOrderedObject(this, nextOffset, val);
    }

    boolean casNext(UnsafeNode<T> cmp, UnsafeNode<T> val) {
        return UNSAFE.compareAndSwapObject(this, nextOffset, cmp, val);
    }
}



  然后写了个Test方法进行测试, 先测试用UNSAFE.putObject(...)方法对volatile成员变量进行赋值。我高高兴兴地Run Test Case,结果立马报错......

have_no_choice.png

        // Unsafe赋值性能测试
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        System.out.println(sdf.format(new Date()));

        for (int i = 0; i < Integer.MAX_VALUE; i++) {
            UnsafeNode<Integer> node = new UnsafeNode<>(i);
        }

        System.out.println(sdf.format(new Date()));
Unsafe_Fail.png



  好吧认为我不安全......正如Unsafe的类注释中写的:

Although the class and all methods are public, use of this class is limited because only trusted code can obtain instances of it.

  我只能用反射去搞了:

        Field theUnsafeInstance = Unsafe.class.getDeclaredField("theUnsafe");
        theUnsafeInstance.setAccessible(true);
        sun.misc.Unsafe UNSAFE = (Unsafe) theUnsafeInstance.get(Unsafe.class);

测试Node,UnfaseNode代码改成如下:

class UnsafeNode<T> {
    volatile T item;
    volatile UnsafeNode<T> next;
    static final sun.misc.Unsafe UNSAFE;
    static final long itemOffset;
    static final long nextOffset;

    static {
        try {
            //获取 Unsafe 内部的私有的实例化单例对象
            Field field = Unsafe.class.getDeclaredField("theUnsafe");
            //无视权限
            field.setAccessible(true);
            UNSAFE = (Unsafe) field.get(null);
            Class<?> klass = UnsafeNode.class;
            itemOffset = UNSAFE.objectFieldOffset
                    (klass.getDeclaredField("item"));
            nextOffset = UNSAFE.objectFieldOffset
                    (klass.getDeclaredField("next"));
        } catch (Exception e) {
            throw new Error(e);
        }
    }

    UnsafeNode(T item) {
        UNSAFE.putObject(this, itemOffset, item);
    }

    boolean casItem(T cmp, T val) {
        return UNSAFE.compareAndSwapObject(this, itemOffset, cmp, val);
    }

    void lazySetNext(UnsafeNode<T> val) {
        UNSAFE.putOrderedObject(this, nextOffset, val);
    }

    boolean casNext(UnsafeNode<T> cmp, UnsafeNode<T> val) {
        return UNSAFE.compareAndSwapObject(this, nextOffset, cmp, val);
    }
}



用Test Case再次跑: 哈哈成功了,用了不到1s完成。😄

Unsafe_Success.png




  我再用普通赋值方法去完成volatile变量的赋值,新的数据结构Node类如下:

class NormalNode<T> {

    volatile T item;
    volatile NormalNode<T> next;

    NormalNode(T it) {
        item = it;
        next = null;
    }

    //忽略其他...
}



  然后改下Test Case测试方法:

         // 普通volitile赋值性能测试
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        System.out.println(sdf.format(new Date()));

        for (int i = 0; i < Integer.MAX_VALUE; i++) {
            NormalNode<Integer> node = new NormalNode<>(i);
        }

        System.out.println(sdf.format(new Date()));



结果相同功能竟然用了花了3s!性能相差3倍以上!

No_Unsafe_Success.png


结论:

  一个普通的volatile成员变量赋值,JDK都考虑地很周到,推荐大家有空多翻翻JDK各个模块,能学好很多,别嘲笑我才发现这个性能点......😂。
下次具体和大家说说Magic Happens的源头。 sun.misc.Unsafe类,做详解Unsafe文章。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,110评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,443评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,474评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,881评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,902评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,698评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,418评论 3 419
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,332评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,796评论 1 316
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,968评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,110评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,792评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,455评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,003评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,130评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,348评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,047评论 2 355