CLH锁与笑傲江湖

CLH锁

所谓CLH锁是并行计算中自旋锁的实现机制之一。

自旋的含义是如果一个Atomic寄存器的值不符合要求,就一直进行循环。这与阻塞的区别是:阻塞会产生现场恢复的开销,而自旋锁省却了这一部分,同时自旋锁需要一直占用CPU。

它隶属于队列锁,它是对ALock的一种改进。

原始代码

import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.locks.Condition;

class QNode{
    boolean locked;
}

public class CLHLock implements java.util.concurrent.locks.Lock {
    AtomicReference<QNode> tail = new AtomicReference<QNode>(new QNode());
    ThreadLocal<QNode> myPred;
    ThreadLocal<QNode> myNode;

    public CLHLock() {
        tail = new AtomicReference<QNode>(new QNode());
        myNode = new ThreadLocal<QNode>() {
            protected QNode initialValue() {
                return new QNode();
            }
        };
        myPred = new ThreadLocal<QNode>() {
            protected QNode initialValue() {
                return null;
            }
        };
    }
    @Override
    public void lock() {
        QNode qnode = myNode.get();
        qnode.locked = true;
        QNode pred = tail.getAndSet(qnode);
        myPred.set(pred);
        while (pred.locked) {
        }
    }
    @Override
    public void unlock() {
        QNode qnode = myNode.get();
        qnode.locked = false;
        myNode.set(myPred.get());
    }
}

解析

这段代码我们初看可能看不懂。不要着急,让浩哥给你讲个故事吧。

在代码世界中有一个笑傲江湖的平行宇宙。在这个平行宇宙中有一个武林盟主——东方不败,他可以号令群雄,但是由于一些众所周知的原因,他整日无所事事,沉湎于酒肉男色,这就激起了武林中另一群野心阴谋家的注意。这群野心家有个共同的名字阴谋家,他们希望拿到那个号令武林的令牌来兴风作浪,实现自己的一些小阴谋。

有一天岳不群伪造了一颗令牌并在其上安装了一个小芯片以便自己以后寻找。他找到东方不败,用自己的假令牌移花接木换到了真令牌,开始了为非作歹的生涯。这个过程就是上锁。

在之后的某一天,岳不群想要退隐江湖,于是他通过gps找到了自己原来的那颗令牌。找到令牌的一刻他惊呆了,令牌居然不在东方不败手中,居然在林平之手中。事情的经过是这样的,林平之也想号令武林,因为阴谋家的阴谋总是类似的,他也找到东方不败换了令牌,但是他万万没有想到,换到的令牌居然也是假的,他大失所望,直到岳不群自投罗网,他抢走了岳不群的令牌,开启了自己为非作歹的时代。这个过程就是第一个线程解锁,并将锁转移到等待者的过程。

历史的发展大浪淘沙,无数阴谋家出现又退隐,有一天,直到最后一个阴谋家退隐,他将锁还给东方不败,这个武林又恢复了以往的平静。同时也在等待着下一轮的潮起潮落。

代码重构

import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;

public abstract class CLHLock implements Lock{
    //every yuebuqun knows the real king-dongfangbubai
    AtomicReference<Qnode> dongfangbubai;
    //I have a token which is a fake one when initialized.
    ThreadLocal<Qnode> myToken = new ThreadLocal<Qnode>(){
            protected Qnode initialValue(){
                return new Qnode();
            }
        };
    //It's a gps so I can find my token with it.
    ThreadLocal<Qnode> gps = new ThreadLocal<Qnode>(){
            protected Qnode initialValue(){
                return null;
            }
        };
    public CLHLock(){
        Qnode qnode = new Qnode();
        qnode.locked.set(false);
        this.dongfangbubai = new AtomicReference<Qnode>(qnode);
    }
    @Override
    public void lock() {
        //link my token with my gps.
        Qnode qnode = myToken.get();
        gps.set(qnode);
        //swap my bad node with the dongfangbubai(king)
        myToken.set(dongfangbubai.getAndSet(qnode));
        //spin until unlock
        while(myToken.get().locked.get()){;}
    }
    @Override
    public void unlock() {      
        //find my token
        //and swap the token with mine.
        gps.get().locked.set(false);
        myToken.get().locked.set(true);
    }

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

推荐阅读更多精彩内容