欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

如何實(shí)現(xiàn)一個(gè)自旋分布式鎖

如何實(shí)現(xiàn)一個(gè)自旋分布式鎖,很多新手對(duì)此不是很清楚,為了幫助大家解決這個(gè)難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來(lái)學(xué)習(xí)下,希望你能有所收獲。

成都創(chuàng)新互聯(lián)-專業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性價(jià)比新蔡網(wǎng)站開(kāi)發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫(kù),直接使用。一站式新蔡網(wǎng)站制作公司更省心,省錢,快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋新蔡地區(qū)。費(fèi)用合理售后完善,十年實(shí)體公司更值得信賴。

AQS的全稱為AbstractQueuedSynchronizer,抽象隊(duì)列同步器

在ReentrantLock類中,我們來(lái)看一下加鎖是怎么來(lái)實(shí)現(xiàn)的。

private final Sync sync;
public void lock() {sync.lock();}

這個(gè)sync就是一個(gè)AQS的子類,并且是一個(gè)抽象類

abstract static class Sync extends AbstractQueuedSynchronizer

它的lock()方法是一個(gè)抽象方法

abstract void lock();

具體實(shí)現(xiàn)sync的是兩個(gè)子類,公平鎖類

static final class FairSync extends Sync

和非公平鎖類

static final class NonfairSync extends Sync

這里我們主要以非公平鎖來(lái)說(shuō)明,因?yàn)槲覀兤匠S玫拇蟛糠侄际欠枪芥i,在非公平鎖中,lock()方法的實(shí)現(xiàn)如下

final void lock() {
    //AQS的內(nèi)部方法,無(wú)鎖競(jìng)爭(zhēng)AQS中state的狀態(tài),state的初始值為0,獲得鎖的將0變?yōu)?if (compareAndSetState(0, 1))
        //競(jìng)爭(zhēng)到state為1的將當(dāng)前線程設(shè)為AQS的獨(dú)家主線程setExclusiveOwnerThread(Thread.currentThread());    else        acquire(1);}

在AbstractQueuedSynchronizer類中

private static final long stateOffset;

在靜態(tài)代碼塊中,我們可以看到這個(gè)stateOffset取的就是state,并且這個(gè)state是多線程可見(jiàn)的volatile。

stateOffset = unsafe.objectFieldOffset
    (AbstractQueuedSynchronizer.class.getDeclaredField("state"));
private volatile int state;
protected final boolean compareAndSetState(int expect, int update) {// See below for intrinsics setup to support this    return unsafe.compareAndSwapInt(this, stateOffset, expect, update);}
private transient Thread exclusiveOwnerThread;
protected final void setExclusiveOwnerThread(Thread thread) {exclusiveOwnerThread = thread;}

這里unsafe.compareAndSwapInt()是用C來(lái)實(shí)現(xiàn)的,我們可以用java來(lái)模擬該方法

@Slf4j@Getterpublic class GetState {private AtomicReference<Integer> state = new AtomicReference<>(0);    private boolean lockState() {while (true) {if (state.compareAndSet(0,1)) {return true;            }
        }
    }private void unlockState() {state.set(0);    }@AllArgsConstructor    private static class Task implements Runnable {private GetState getState;        @Override        public void run() {if (getState.lockState()) {log.info(Thread.currentThread().getName() + "獲取鎖");            }
        }
    }public static void main(String[] args) throws InterruptedException {
        ExecutorService service = Executors.newFixedThreadPool(16);        GetState state = new GetState();        for (int i = 0;i < 10;i++) {
            service.execute(new Task(state));        }while (state.getState().get() == 1) {
            Thread.sleep(1000);            state.unlockState();        }
        service.shutdown();    }
}

打印日志(每秒打印一條)

15:35:42.953 [pool-1-thread-1] INFO com.guanjian.websocket.tomic.GetState - pool-1-thread-1獲取鎖
15:35:43.953 [pool-1-thread-9] INFO com.guanjian.websocket.tomic.GetState - pool-1-thread-9獲取鎖
15:35:44.957 [pool-1-thread-5] INFO com.guanjian.websocket.tomic.GetState - pool-1-thread-5獲取鎖
15:35:45.962 [pool-1-thread-2] INFO com.guanjian.websocket.tomic.GetState - pool-1-thread-2獲取鎖
15:35:46.962 [pool-1-thread-7] INFO com.guanjian.websocket.tomic.GetState - pool-1-thread-7獲取鎖
15:35:47.962 [pool-1-thread-3] INFO com.guanjian.websocket.tomic.GetState - pool-1-thread-3獲取鎖
15:35:48.967 [pool-1-thread-8] INFO com.guanjian.websocket.tomic.GetState - pool-1-thread-8獲取鎖
15:35:49.969 [pool-1-thread-6] INFO com.guanjian.websocket.tomic.GetState - pool-1-thread-6獲取鎖
15:35:50.970 [pool-1-thread-4] INFO com.guanjian.websocket.tomic.GetState - pool-1-thread-4獲取鎖
15:35:51.971 [pool-1-thread-10] INFO com.guanjian.websocket.tomic.GetState - pool-1-thread-10獲取鎖

Process finished with exit code 0

現(xiàn)在我們可以來(lái)寫(xiě)一個(gè)支持自旋的分布式鎖了。

public class SpinDistributedLock {private volatile AtomicReference<Boolean> state = new AtomicReference<>(false);    public  boolean lock(redisService redisService,String key,String value,int expire) {while (true) {if (state.compareAndSet(false,                    RedisTool.tryGetDistributedLock(redisService,key,value,expire))) {if (state.get()) {return true;                }
            }
        }
    }public void unlock(RedisService redisService,String key,String value) {state.set(!RedisTool.releaseDistributedLock(redisService,key,value));    }
}

常規(guī)分布式鎖可以參考采用redis token,分布式鎖的接口冪等性實(shí)現(xiàn)

現(xiàn)在我們來(lái)進(jìn)行一個(gè)簡(jiǎn)單的測(cè)試,先不使用分布式鎖

我們?cè)趓edis中手動(dòng)設(shè)置一個(gè)鍵count,0

127.0.0.1:6379> set count 0
OK

我們的目的是累加這個(gè)count,但不能讓其超過(guò)10

@Servicepublic class NoDistributedTest {@Autowired    private RedisService redisService;    private class Task implements Runnable {@Override        public void run() {if (Integer.valueOf(redisService.get("count")) < 10) {redisService.incr("count");            }
        }
    }@PostConstruct    public void test() {
        ExecutorService service = Executors.newFixedThreadPool(16);        for (int i = 0;i < 100000;i++) {
            service.execute(new Task());        }
        service.shutdown();    }
}

我們啟動(dòng)兩個(gè)進(jìn)程,兩個(gè)進(jìn)程啟動(dòng)完成后,我們?cè)賮?lái)看一下該鍵的值。

127.0.0.1:6379> get count
"15"

這個(gè)時(shí)候我們可以看到,在沒(méi)有鎖的情況下,數(shù)量超過(guò)了10.

現(xiàn)在用分布式鎖來(lái)進(jìn)行測(cè)試。

將count鍵重新設(shè)為0

127.0.0.1:6379> set count 0
OK

@Slf4j@Servicepublic class DistributedTest {private SpinDistributedLock lock = new SpinDistributedLock();    @Autowired    private RedisService redisService;    private class Task implements Runnable {@Override        public void run() {try {lock.lock(redisService,"countlock","countlock",3);                log.info(Thread.currentThread().getName() + "進(jìn)入鎖");                if (Integer.valueOf(redisService.get("count")) < 10) {redisService.incr("count");                }
            } finally {lock.unlock(redisService,"countlock","countlock");                log.info(Thread.currentThread().getName() + "釋放鎖");            }
        }
    }@PostConstruct    public void test() {
        ExecutorService service = Executors.newFixedThreadPool(16);        for (int i = 0;i < 100000;i++) {
            service.execute(new Task());        }
        service.shutdown();    }
}

同樣啟動(dòng)兩個(gè)進(jìn)程或者更多進(jìn)程,啟動(dòng)完成后,我們來(lái)看一下count鍵的值

127.0.0.1:6379> get count
"10"

如何實(shí)現(xiàn)一個(gè)自旋分布式鎖

如何實(shí)現(xiàn)一個(gè)自旋分布式鎖

看完上述內(nèi)容是否對(duì)您有幫助呢?如果還想對(duì)相關(guān)知識(shí)有進(jìn)一步的了解或閱讀更多相關(guān)文章,請(qǐng)關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝您對(duì)創(chuàng)新互聯(lián)的支持。

網(wǎng)站名稱:如何實(shí)現(xiàn)一個(gè)自旋分布式鎖
URL分享:http://www.aaarwkj.com/article34/gdsgpe.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供移動(dòng)網(wǎng)站建設(shè)、網(wǎng)站設(shè)計(jì)、品牌網(wǎng)站制作、營(yíng)銷型網(wǎng)站建設(shè)、電子商務(wù)、外貿(mào)建站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

成都定制網(wǎng)站網(wǎng)頁(yè)設(shè)計(jì)
亚洲黄色成人在线观看| 五月婷婷丁香婷婷丁香| av真人青青小草一区二区欧美| 青青成线在人线免费啪| 午夜精品久久久免费视频| 中文字幕欧美人妻在线| 成人黄色三级免费网站| 国产一级性生活高清在线| 禁止18岁以下观看的视频| 亚洲av一区二区三区网站| 欧美精品青青久久久久久| 蜜桃视频中文字幕二区三区 | 亚洲精品有码中文字幕| 亚洲欧美中文日韩一区| 国产成人在线观看av| 四虎免费在线高清观看| 亚洲成av在线免费不卡| 欧美丝袜熟女日韩亚洲| 日韩精品一区二区三区夜色| 国模在线视频一区二区| 国产白浆一区二区视频| 99热这里只有精品中文有码| 亚洲av乱码久久精品蜜桃| 日本黄色免费在线观看网站| 日本理论午夜三级在线观看| 天堂社区人妻在线亚洲| 欧美亚洲精品一区二区三区| 日韩蜜桃av一二三四区| 91国产视频在线观看免费| 人妻少妇系列一区二区| 日韩精品在线免费观看了| 亚洲av日韩综合一区尤物| 日本在线一区二区不卡视频| 日本人妻伦理在线播放| 亚洲免费麻豆一区二区三区| 欧美日韩亚洲精品久久| 精品人妻二区中文字幕| 啄木乌欧美一区二区三| 黄色高清无遮挡在线观看| 亚洲最大成人综合福利网| 亚洲男人av天堂午夜在|