欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談Java鎖機制

 更新時間:2021年09月30日 16:10:13   作者:隨身電源  
在多線程環(huán)境下,程序往往會出現(xiàn)一些線程安全問題,為此,Java提供了一些線程的同步機制來解決安全問題,比如:synchronized鎖和Lock鎖都能解決線程安全問題。下面小編就來詳細(xì)介紹該知識點,需要的朋友可以參考一下

1、悲觀鎖和樂觀鎖

我們可以將鎖大體分為兩類:

  • 悲觀鎖
  • 樂觀鎖

顧名思義,悲觀鎖總是假設(shè)最壞的情況,每次獲取數(shù)據(jù)的時候都認(rèn)為別的線程會修改,所以每次在拿數(shù)據(jù)的時候都會上鎖,這樣其它線程想要修改這個數(shù)據(jù)的時候都會被阻塞直到獲取鎖。比如MySQL數(shù)據(jù)庫中的表鎖、行鎖、讀鎖、寫鎖等,Java中的synchronizedReentrantLock等。

而樂觀鎖總是假設(shè)最好的情況,每次獲取數(shù)據(jù)的時候都認(rèn)為別的線程不會修改,所以并不會上鎖,但是在修改數(shù)據(jù)的時候需要判斷一下在此期間有沒有別的線程修改過數(shù)據(jù),如果沒有修改過則正常修改,如果修改過則這次修改就是失敗的。常見的樂觀鎖有版本號控制、CAS算法等。

2、悲觀鎖應(yīng)用

案例如下:

public class LockDemo {

    static int count = 0;

    public static void main(String[] args) throws InterruptedException {
        List<Thread> threadList = new ArrayList<>();
        for (int i = 0; i < 50; i++) {
            Thread thread = new Thread(() -> {
                for (int j = 0; j < 1000; ++j) {
                    count++;
                }
            });
            thread.start();
            threadList.add(thread);
        }
        // 等待所有線程執(zhí)行完畢
        for (Thread thread : threadList) {
            thread.join();
        }
        System.out.println(count);
    }
}

在該程序中一共開啟了50個線程,并在線程中對共享變量count進行++操作,所以如果不發(fā)生線程安全問題,最終的結(jié)果應(yīng)該是50000,但該程序中一定存在線程安全問題,運行結(jié)果為:

48634

若想解決線程安全問題,可以使用synchronized關(guān)鍵字:

public class LockDemo {

    static int count = 0;

    public static void main(String[] args) throws InterruptedException {
        List<Thread> threadList = new ArrayList<>();
        for (int i = 0; i < 50; i++) {
            Thread thread = new Thread(() -> {
                // 使用synchronized關(guān)鍵字解決線程安全問題
                synchronized (LockDemo.class) {
                    for (int j = 0; j < 1000; ++j) {
                        count++;
                    }
                }
            });
            thread.start();
            threadList.add(thread);
        }
        for (Thread thread : threadList) {
            thread.join();
        }
        System.out.println(count);
    }
}

將修改count變量的操作使用synchronized關(guān)鍵字包裹起來,這樣當(dāng)某個線程在進行++操作時,別的線程是無法同時進行++的,只能等待前一個線程執(zhí)行完1000次后才能繼續(xù)執(zhí)行,這樣便能保證最終的結(jié)果為50000。

使用ReentrantLock也能夠解決線程安全問題:

public class LockDemo {

    static int count = 0;

    public static void main(String[] args) throws InterruptedException {
        List<Thread> threadList = new ArrayList<>();
        Lock lock = new ReentrantLock();
        for (int i = 0; i < 50; i++) {
            Thread thread = new Thread(() -> {
                // 使用ReentrantLock關(guān)鍵字解決線程安全問題
                lock.lock();
                try {
                    for (int j = 0; j < 1000; ++j) {
                        count++;
                    }
                } finally {
                    lock.unlock();
                }
            });
            thread.start();
            threadList.add(thread);
        }
        for (Thread thread : threadList) {
            thread.join();
        }
        System.out.println(count);
    }
}

這兩種鎖機制都是悲觀鎖的具體實現(xiàn),不管其它線程是否會同時修改,它都直接上鎖,保證了原子操作。

3、樂觀鎖應(yīng)用

由于線程的調(diào)度是極其耗費操作系統(tǒng)資源的,所以,我們應(yīng)該盡量避免線程在不斷阻塞和喚醒中切換,由此產(chǎn)生了樂觀鎖。

在數(shù)據(jù)庫表中,我們往往會設(shè)置一個version字段,這就是樂觀鎖的體現(xiàn),假設(shè)某個數(shù)據(jù)表的數(shù)據(jù)內(nèi)容如下:

+----+------+----------+ ------- +
| id | name | password | version |
+----+------+----------+ ------- +
|  1 | zs   | 123456   |    1    |
+----+------+----------+ ------- +

它是如何避免線程安全問題的呢?

假設(shè)此時有兩個線程A、B想要修改這條數(shù)據(jù),它們會執(zhí)行如下的sql語句:

select version from e_user where name = 'zs';

update e_user set password = 'admin',version = version + 1 where name = 'zs' and version = 1;

首先兩個線程均查詢出zs用戶的版本號為1,然后線程A先執(zhí)行了更新操作,此時將用戶的密碼修改為了admin,并將版本號加1,接著線程B執(zhí)行更新操作,此時版本號已經(jīng)為2了,所以更新肯定是失敗的,由此,線程B就失敗了,它只能重新去獲取版本號再進行更新,這就是樂觀鎖,我們并沒有對程序和數(shù)據(jù)庫進行任何的加鎖操作,但它仍然能夠保證線程安全。

4、CAS

仍然以最開始做加法的程序為例,在Java中,我們還可以采用一種特殊的方式來實現(xiàn)它:

public class LockDemo {

    static AtomicInteger count = new AtomicInteger(0);

    public static void main(String[] args) throws InterruptedException {
        List<Thread> threadList = new ArrayList<>();
        for (int i = 0; i < 50; i++) {
            Thread thread = new Thread(() -> {
                for (int j = 0; j < 1000; ++j) {
                    // 使用AtomicInteger解決線程安全問題
                    count.incrementAndGet();
                }
            });
            thread.start();
            threadList.add(thread);
        }
        for (Thread thread : threadList) {
            thread.join();
        }
        System.out.println(count);
    }
}

為何使用AtomicInteger類就能夠解決線程安全問題呢?

我們來查看一下源碼:

public final int incrementAndGet() {
    return unsafe.getAndAddInt(this, valueOffset, 1) + 1;
}

當(dāng)count調(diào)用incrementAndGet()方法時,實際上調(diào)用的是UnSafe類的getAndAddInt()方法:

public final int getAndAddInt(Object var1, long var2, int var4) {
    int var5;
    do {
        var5 = this.getIntVolatile(var1, var2);
    } while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));

    return var5;
}

getAndAddInt()方法中有一個循環(huán),關(guān)鍵的代碼就在這里,我們假設(shè)線程A此時進入了該方法,此時var1即為AtomicInteger對象(初始值為0),var2的值為12(這是一個內(nèi)存偏移量,我們可以不用關(guān)心),var4的值為1(準(zhǔn)備對count進行加1操作)。

首先通過AtomicInteger對象和內(nèi)存偏移量即可得到主存中的數(shù)據(jù)值:

var5 = this.getIntVolatile(var1, var2);

獲取到var5的值為0,然后程序會進行判斷:

!this.compareAndSwapInt(var1, var2, var5, var5 + var4)

compareAndSwapInt()是一個本地方法,它的作用是比較并交換,即:判斷var1的值與主存中取出的var5的值是否相同,此時肯定是相同的,所以會將var5+var4的值賦值給var1,并返回true,對true取反為false,所以循環(huán)就結(jié)束了,最終方法返回1。

這是一切正常的運行流程,然而當(dāng)發(fā)生并發(fā)時,處理情況就不太一樣了,假設(shè)此時線程A執(zhí)行到了getAndAddInt()方法:

public final int getAndAddInt(Object var1, long var2, int var4) {
    int var5;
    do {
        var5 = this.getIntVolatile(var1, var2);
    } while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));

    return var5;
}

線程A此時獲取到var1的值為0(var1即為共享變量AtomicInteger),當(dāng)線程A正準(zhǔn)備執(zhí)行下去時,線程B搶先執(zhí)行了,線程B此時獲取到var1的值為0,var5的值為0,比較成功,此時var1的值就變?yōu)?;這時候輪到線程A執(zhí)行了,它獲取var5的值為1,此時var1的值不等于var5的值,此次加1操作就會失敗,并重新進入循環(huán),此時var1的值已經(jīng)發(fā)生了變化,此時重新獲取var5的值也為1,比較成功,所以將var1的值加1變?yōu)?,若是在獲取var5之前別的線程又修改了主存中var1的值,則本次操作又會失敗,程序重新進入循環(huán)。

這就是利用自旋的方式來實現(xiàn)一個樂觀鎖,因為它沒有加鎖,所以省下了線程調(diào)度的資源,但也要避免程序一直自旋的情況發(fā)生。

5、手寫一個自旋鎖

public class LockDemo {

    private AtomicReference<Thread> atomicReference = new AtomicReference<>();

    public void lock() {
        // 獲取當(dāng)前線程對象
        Thread thread = Thread.currentThread();
        // 自旋等待
        while (!atomicReference.compareAndSet(null, thread)) {
        }
    }

    public void unlock() {
        // 獲取當(dāng)前線程對象
        Thread thread = Thread.currentThread();
        atomicReference.compareAndSet(thread, null);
    }

    static int count = 0;

    public static void main(String[] args) throws InterruptedException {
        LockDemo lockDemo = new LockDemo();
        List<Thread> threadList = new ArrayList<>();
        for (int i = 0; i < 50; i++) {
            Thread thread = new Thread(() -> {
                lockDemo.lock();
                for (int j = 0; j < 1000; j++) {
                    count++;
                }
                lockDemo.unlock();
            });
            thread.start();
            threadList.add(thread);
        }
        // 等待線程執(zhí)行完畢
        for (Thread thread : threadList) {
            thread.join();
        }
        System.out.println(count);
    }
}

使用CAS的原理可以輕松地實現(xiàn)一個自旋鎖,首先,AtomicReference中的初始值一定為null,所以第一個線程在調(diào)用lock()方法后會成功將當(dāng)前線程的對象放入AtomicReference,此時若是別的線程調(diào)用lock()方法,會因為該線程對象與AtomicReference中的對象不同而陷入循環(huán)的等待中,直到第一個線程執(zhí)行完++操作,調(diào)用了unlock()方法,該線程才會將AtomicReference值置為null,此時別的線程就可以跳出循環(huán)了。

通過CAS機制,我們能夠在不添加鎖的情況下模擬出加鎖的效果,但它的缺點也是顯而易見的:

  • 循環(huán)等待占用CPU資源
  • 只能保證一個變量的原子操作
  • 會產(chǎn)生ABA問題

到此這篇關(guān)于淺談Java鎖機制的文章就介紹到這了,更多相關(guān)Java鎖機制內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • feign post參數(shù)對象不加@RequestBody的使用說明

    feign post參數(shù)對象不加@RequestBody的使用說明

    這篇文章主要介紹了feign post參數(shù)對象不加@RequestBody的使用說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • SpringCloud Alibaba使用Seata處理分布式事務(wù)的技巧

    SpringCloud Alibaba使用Seata處理分布式事務(wù)的技巧

    在傳統(tǒng)的單體項目中,我們使用@Transactional注解就能實現(xiàn)基本的ACID事務(wù)了,隨著微服務(wù)架構(gòu)的引入,需要對數(shù)據(jù)庫進行分庫分表,每個服務(wù)擁有自己的數(shù)據(jù)庫,這樣傳統(tǒng)的事務(wù)就不起作用了,那么我們?nèi)绾伪WC多個服務(wù)中數(shù)據(jù)的一致性呢?跟隨小編一起通過本文了解下吧
    2021-06-06
  • SpringBoot項目中的視圖解析器問題(兩種)

    SpringBoot項目中的視圖解析器問題(兩種)

    SpringBoot官網(wǎng)推薦使用HTML視圖解析器,但是根據(jù)個人的具體業(yè)務(wù)也有可能使用到JSP視圖解析器,所以本文介紹了兩種視圖解析器,感興趣的可以了解下
    2020-06-06
  • Java多線程編程綜合案例詳解

    Java多線程編程綜合案例詳解

    這篇文章將通過三個案例帶大家了解一下Java中的多線程編程,文中的示例代碼介紹詳細(xì),對我們的學(xué)習(xí)或工作有一定的價值,感興趣的小伙伴可以了解一下
    2022-07-07
  • Spring中@Scope注解用法解析

    Spring中@Scope注解用法解析

    這篇文章主要介紹了Spring中@Scope注解用法解析,@Scope注解主要作用是調(diào)節(jié)Ioc容器中的作用域,在Spring IoC容器中主要有以下五種作用域,需要的朋友可以參考下
    2023-11-11
  • idea2023創(chuàng)建JavaWeb教程之右鍵沒有Servlet的問題解決

    idea2023創(chuàng)建JavaWeb教程之右鍵沒有Servlet的問題解決

    最近在寫一個javaweb項目,但是在IDEA中創(chuàng)建好項目后,在搭建結(jié)構(gòu)的時候創(chuàng)建servlet文件去沒有選項,所以這里給大家總結(jié)下,這篇文章主要給大家介紹了關(guān)于idea2023創(chuàng)建JavaWeb教程之右鍵沒有Servlet問題的解決方法,需要的朋友可以參考下
    2023-10-10
  • 在SpringBoot: SpringBoot里面創(chuàng)建導(dǎo)出Excel的接口教程

    在SpringBoot: SpringBoot里面創(chuàng)建導(dǎo)出Excel的接口教程

    這篇文章主要介紹了在SpringBoot: SpringBoot里面創(chuàng)建導(dǎo)出Excel的接口教程,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-10-10
  • 詳解Java模擬棧的實現(xiàn)以及Stack類的介紹

    詳解Java模擬棧的實現(xiàn)以及Stack類的介紹

    棧是一種數(shù)據(jù)結(jié)構(gòu),它按照后進先出的原則來存儲和訪問數(shù)據(jù)。Stack是一個類,表示棧數(shù)據(jù)結(jié)構(gòu)的實現(xiàn)。本文就來和大家介紹一下Java模擬棧的實現(xiàn)以及Stack類的使用,需要的可以參考一下
    2023-04-04
  • SpringBoot集成kaptcha驗證碼

    SpringBoot集成kaptcha驗證碼

    這篇文章主要為大家詳細(xì)介紹了SpringBoot集成kaptcha驗證碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-07-07
  • IDEA如何切換JDK版本

    IDEA如何切換JDK版本

    本文主要介紹了IDEA如何切換JDK版本,JDK版本之間的關(guān)系是一個向后兼容的關(guān)系,所以我們需要切換JDK的版本號,具有一定的參考價值,感興趣的可以了解一下
    2024-01-01

最新評論