亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

java中synchronized關(guān)鍵字的3種寫法實(shí)例

 更新時(shí)間:2021年11月28日 12:41:13   作者:我想月薪過(guò)萬(wàn)  
synchronized是Java中的關(guān)鍵字,是一種同步鎖,下面這篇文章主要給大家介紹了關(guān)于java中synchronized關(guān)鍵字的3種寫法,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下

預(yù)備知識(shí)

首先,我們得知道在java中存在三種變量:

  • 實(shí)例變量 ==》 存在于堆中
  • 靜態(tài)變量 ==》 存在于方法區(qū)中
  • 局部變量 ==》 存在于棧中

然后,我們得明白,合適會(huì)發(fā)生高并發(fā)不安全

  • 條件1:多線程并發(fā)。
  • 條件2:有共享數(shù)據(jù)。
  • 條件3:共享數(shù)據(jù)有修改的行為。

具體不安全案例請(qǐng)參考 如下這篇文章:java線程安全問(wèn)題詳解

在上面這篇文章銀行取錢案例中,我們解決線程安全問(wèn)題的方法是加了一個(gè) synchronized 關(guān)鍵字。下面我們就詳細(xì)介紹一下?synchronized 的三種寫法,分別解決什么問(wèn)題?。?!

寫法一:修飾代碼塊

package ThreadSafa;
 
public class Test {
    public static void main(String[] args) {
        TestAccount ta1 = new TestAccount();
        ta1.setNum(10);
 
        //共用一個(gè)賬戶對(duì)象
        TestThread t1 = new TestThread(ta1);
        TestThread t2 = new TestThread(ta1);
        t1.start();
        t2.start();
    }
}
 
class TestThread extends Thread {
 
    private TestAccount mAccount;
 
    public TestThread(TestAccount mAccount) {
        this.mAccount = mAccount;
    }
 
    @Override
    public void run() {
        mAccount.updateNum(1);
    }
}
 
class TestAccount {
    private double num;
 
    public double getNum() {
        return num;
    }
 
    public void setNum(double num) {
        this.num = num;
    }
 
    public void updateNum(int n) {
        synchronized (this) {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            setNum(getNum() - n);
        }
        System.out.println(getNum());
    }
}

運(yùn)行結(jié)果

?寫法二:修飾方法

package ThreadSafa;
 
public class Test {
    public static void main(String[] args) {
        TestAccount ta1 = new TestAccount();
        ta1.setNum(10);
 
        TestThread t1 = new TestThread(ta1);
        TestThread t2 = new TestThread(ta1);
        t1.start();
        t2.start();
    }
}
 
class TestThread extends Thread {
 
    private TestAccount mAccount;
 
    public TestThread(TestAccount mAccount) {
        this.mAccount = mAccount;
    }
 
    @Override
    public void run() {
        mAccount.updateNum(1);
    }
}
 
class TestAccount {
    private double num;
 
    public double getNum() {
        return num;
    }
 
    public void setNum(double num) {
        this.num = num;
    }
 
    public synchronized void updateNum(int n) {
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        setNum(getNum() - n);
        System.out.println(getNum());
    }
}

運(yùn)行結(jié)果

總結(jié)

可以看到 ,前面這兩種寫法其實(shí)是等價(jià)的,什么意思呢?就是當(dāng)你用?synchronized 修飾共享對(duì)象 this 的時(shí)候 你就可以吧?synchronized 提到方法前面,但是我們一般不會(huì)這么干,因?yàn)閿U(kuò)大?synchronized 修飾的代碼范圍會(huì)使代碼運(yùn)行效率降低。

同時(shí),前面兩種方法都是為了解決 實(shí)例變量 線程安全問(wèn)題而誕生的,對(duì)于靜態(tài)變量我們?cè)趺刺幚砟???qǐng)看寫法三:

寫法三:修飾靜態(tài)方法

package ThreadSafa;
 
public class Test {
    public static void main(String[] args) {
        TestAccount ta1 = new TestAccount();
        TestAccount ta2 = new TestAccount();
 
        TestThread t1 = new TestThread(ta1);
        TestThread t2 = new TestThread(ta2);
        t1.start();
        t2.start();
    }
}
 
class TestThread extends Thread {
 
    private TestAccount mAccount;
 
    public TestThread(TestAccount mAccount) {
        this.mAccount = mAccount;
    }
 
    @Override
    public void run() {
        mAccount.updateCount(1);
    }
}
 
class TestAccount {
    private double num;
    public static double count = 10;
 
    public double getNum() {
        return num;
    }
 
    public void setNum(double num) {
        this.num = num;
    }
 
    public synchronized void updateNum(int n) {
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        setNum(getNum() - n);
        System.out.println(getNum());
    }
 
    public synchronized static void updateCount(int n) {
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        count -= n;
        System.out.println(count);
    }
}

運(yùn)行結(jié)果展示

可以看到,在靜態(tài)方法上加 synchronized?之后,他鎖的是這個(gè)類,盡管兩個(gè)賬戶對(duì)象不一樣,但是,加了?synchronized 會(huì)保證他們排隊(duì)執(zhí)行,也就是保證了線程安全。

synchronized原理

Synchronized是通過(guò)對(duì)象內(nèi)部的一個(gè)叫做監(jiān)視器鎖(monitor)來(lái)實(shí)現(xiàn)的。但是監(jiān)視器鎖本質(zhì)又是依賴于底層的操作系統(tǒng)的互斥鎖(Mutex Lock)來(lái)實(shí)現(xiàn)的。而操作系統(tǒng)實(shí)現(xiàn)線程之間的切換這就需要從用戶態(tài)轉(zhuǎn)換到核心態(tài),這個(gè)成本非常高,狀態(tài)之間的轉(zhuǎn)換需要相對(duì)比較長(zhǎng)的時(shí)間,這就是為什么Synchronized效率低的原因。這種依賴于操作系統(tǒng)互斥鎖(Mutex Lock)所實(shí)現(xiàn)的鎖我們稱之為“重量級(jí)鎖”。

1. monitor鎖定過(guò)程

當(dāng)monitor被占用時(shí)就會(huì)處于鎖定狀態(tài),線程執(zhí)行monitorenter指令時(shí)嘗試獲取monitor的所有權(quán),過(guò)程如下:

a、如果monitor的進(jìn)入數(shù)為0,則該線程進(jìn)入monitor,然后將進(jìn)入數(shù)設(shè)置為1,該線程即為monitor的所有者。

b、如果線程已經(jīng)占有該monitor,只是重新進(jìn)入,則進(jìn)入monitor的進(jìn)入數(shù)加1.

c、如果其他線程已經(jīng)占用了monitor,則該線程進(jìn)入阻塞狀態(tài),直到monitor的進(jìn)入數(shù)為0,再重新嘗試獲取monitor的所有權(quán)。

2. synchronized鎖

Java SE1.6對(duì)Synchronized進(jìn)行了各種優(yōu)化之后,它并不那么重了。在不同的場(chǎng)景中引入不同的鎖優(yōu)化。

1.偏向鎖:適用于鎖沒(méi)有競(jìng)爭(zhēng)的情況,假設(shè)共享變量只有一個(gè)線程訪問(wèn)。如果有其他線程競(jìng)爭(zhēng)鎖,鎖則會(huì)膨脹成為輕量級(jí)鎖。

2.輕量級(jí)鎖:適用于鎖有多個(gè)線程競(jìng)爭(zhēng),但是在一個(gè)同步方法塊周期中鎖不存在競(jìng)爭(zhēng),如果在同步周期內(nèi)有其他線程競(jìng)爭(zhēng)鎖,鎖會(huì)膨脹為重量級(jí)鎖。

3.重量級(jí)鎖:競(jìng)爭(zhēng)激烈的情況下使用重量級(jí)鎖。

偏向鎖和輕量級(jí)鎖之所以會(huì)在性能上比重量級(jí)鎖是因?yàn)楹茫举|(zhì)上是因?yàn)槠蜴i和輕量級(jí)鎖僅僅使用了CAS。

3. synchronized鎖優(yōu)化

盡量采用輕量級(jí)鎖和偏向鎖等對(duì)Synchronized的優(yōu)化,但是這兩種鎖也不是完全沒(méi)缺點(diǎn)的,比如競(jìng)爭(zhēng)比較激烈的時(shí)候,不但無(wú)法提升效率,反而會(huì)降低效率,因?yàn)槎嗔艘粋€(gè)鎖升級(jí)的過(guò)程,這個(gè)時(shí)候就需要通過(guò)-XX:-UseBiasedLocking來(lái)禁用偏向鎖。

總結(jié)

局部變量 =》 存在于棧中 =》 線程之間不能共享 =》 所以數(shù)據(jù)永遠(yuǎn)是安全的

實(shí)例變量 =》 存在于堆中 =》 線程之間能共享 =》 采用寫法一和寫法二保證線程安全

靜態(tài)變量 =》 存在于方法區(qū) =》 線程之間能共享 =》 采用方寫法三保證線程安全

相關(guān)文章

  • Spring AOP定義Before增加實(shí)戰(zhàn)案例詳解

    Spring AOP定義Before增加實(shí)戰(zhàn)案例詳解

    這篇文章主要介紹了Spring AOP定義Before增加,結(jié)合實(shí)例形式詳細(xì)分析了Spring面向切面AOP定義Before增加相關(guān)定義與使用技巧,需要的朋友可以參考下
    2020-01-01
  • 詳解Lombok快速上手(安裝、使用與注解參數(shù))

    詳解Lombok快速上手(安裝、使用與注解參數(shù))

    這篇文章主要介紹了詳解Lombok快速上手(安裝、使用與注解參數(shù)) ,這里整理了一些日常編碼中能遇到的所有關(guān)于它的使用詳解,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-12-12
  • El表達(dá)式使用問(wèn)題javax.el.ELException:Failed to parse the expression的解決方式

    El表達(dá)式使用問(wèn)題javax.el.ELException:Failed to parse the expression

    今天小編就為大家分享一篇關(guān)于Jsp El表達(dá)式使用問(wèn)題javax.el.ELException:Failed to parse the expression的解決方式,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2018-12-12
  • 基于spring boot排除掃描類的三種方式小結(jié)

    基于spring boot排除掃描類的三種方式小結(jié)

    這篇文章主要介紹了spring boot排除掃描類的三種方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-08-08
  • Java實(shí)現(xiàn)鼠標(biāo)拖放功能的方法

    Java實(shí)現(xiàn)鼠標(biāo)拖放功能的方法

    這篇文章主要介紹了Java實(shí)現(xiàn)鼠標(biāo)拖放功能的方法,很實(shí)用的功能,需要的朋友可以參考下
    2014-07-07
  • SpringBoot中配置log4j2日志詳解

    SpringBoot中配置log4j2日志詳解

    這篇文章主要介紹了SpringBoot中配置log4j2日志詳解,Apache Log4j2 是對(duì)原先的 Log4j 項(xiàng)目的升級(jí)版本,參考了 logback 的一些優(yōu)秀的設(shè)計(jì),并且修復(fù)了一些問(wèn)題,因此帶來(lái)了一些重大的提升,需要的朋友可以參考下
    2023-11-11
  • Java關(guān)鍵字之this用法詳解

    Java關(guān)鍵字之this用法詳解

    這篇文章將為大家詳細(xì)介紹一下Java關(guān)鍵字this的用法,文中有相關(guān)的代碼示例,希望對(duì)大家的學(xué)習(xí)或工作有一定的幫助,感興趣的同學(xué)可以參考下
    2023-05-05
  • 使用java獲取md5值的兩種方法

    使用java獲取md5值的兩種方法

    本篇文章是對(duì)使用java獲取md5值的兩種方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
    2013-06-06
  • SpringBoot ApplicationContextAware拓展接口使用詳解

    SpringBoot ApplicationContextAware拓展接口使用詳解

    當(dāng)一個(gè)類實(shí)現(xiàn)了這個(gè)接口(ApplicationContextAware)之后,這個(gè)類就可以方便獲得ApplicationContext中的所有bean。換句話說(shuō),就是這個(gè)類可以直接獲取spring配置文件中,所有有引用到的bean對(duì)象
    2023-04-04
  • java 實(shí)現(xiàn)計(jì)數(shù)排序和桶排序?qū)嵗a

    java 實(shí)現(xiàn)計(jì)數(shù)排序和桶排序?qū)嵗a

    這篇文章主要介紹了java 實(shí)現(xiàn)計(jì)數(shù)排序和桶排序?qū)嵗a的相關(guān)資料,需要的朋友可以參考下
    2017-02-02

最新評(píng)論