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

java多線程編程之使用Synchronized關鍵字同步類方法

 更新時間:2014年01月23日 15:07:32   作者:  
JAVA中要想解決“臟數(shù)據(jù)”的問題,最簡單的方法就是使用synchronized關鍵字來使run方法同步,看下面的代碼,只要在void和public之間加上synchronized關鍵字

復制代碼 代碼如下:

public synchronized void run()
{    
}

從上面的代碼可以看出,只要在void和public之間加上synchronized關鍵字,就可以使run方法同步,也就是說,對于同一個Java類的對象實例,run方法同時只能被一個線程調用,并當前的run執(zhí)行完后,才能被其他的線程調用。即使當前線程執(zhí)行到了run方法中的yield方法,也只是暫停了一下。由于其他線程無法執(zhí)行run方法,因此,最終還是會由當前的線程來繼續(xù)執(zhí)行。先看看下面的代碼:
sychronized關鍵字只和一個對象實例綁定
復制代碼 代碼如下:

class Test
  {
        public synchronized void method()
       {

       }
  }

  public class Sync implements Runnable
  {
       private Test test;
       public void run()
       {
            test.method();
       }
       public Sync(Test test)
       {
           this.test = test;
       }
       public static void main(String[] args) throws Exception
       {
           Test test1 =  new Test();
           Test test2 =  new Test();
           Sync sync1 = new Sync(test1);
           Sync sync2 = new Sync(test2);
           new Thread(sync1).start();
           new Thread(sync2).start();
       }
   }
 

在Test類中的method方法是同步的。但上面的代碼建立了兩個Test類的實例,因此,test1和test2的method方法是分別執(zhí)行的。要想讓method同步,必須在建立Sync類的實例時向它的構造方法中傳入同一個Test類的實例,如下面的代碼所示:
Sync sync1 = new Sync(test1);
不僅可以使用synchronized來同步非靜態(tài)方法,也可以使用synchronized來同步靜態(tài)方法。如可以按如下方式來定義method方法:
復制代碼 代碼如下:

class Test
{
    public static synchronized void method() {   }
}

建立Test類的對象實例如下:
Test test = new Test();
對于靜態(tài)方法來說,只要加上了synchronized關鍵字,這個方法就是同步的,無論是使用test.method(),還是使用Test.method()來調用method方法,method都是同步的,并不存在非靜態(tài)方法的多個實例的問題。
在23種設計模式中的單件(Singleton)模式如果按傳統(tǒng)的方法設計,也是線程不安全的,下面的代碼是一個線程不安全的單件模式。

復制代碼 代碼如下:

package test;

// 線程安全的Singleton模式
class Singleton
{
    private static Singleton sample;

    private Singleton()
    {
    }
    public static Singleton getInstance()
    {
        if (sample == null)
        {
            Thread.yield(); // 為了放大Singleton模式的線程不安全性
            sample = new Singleton();
        }
        return sample;
    }
}
public class MyThread extends Thread
{
    public void run()
    {
        Singleton singleton = Singleton.getInstance();
        System.out.println(singleton.hashCode());
    }
    public static void main(String[] args)
    {
        Thread threads[] = new Thread[5];
        for (int i = 0; i < threads.length; i++)
            threads[i] = new MyThread();
        for (int i = 0; i < threads.length; i++)
            threads[i].start();
    }
}


在上面的代碼調用yield方法是為了使單件模式的線程不安全性表現(xiàn)出來,如果將這行去掉,上面的實現(xiàn)仍然是線程不安全的,只是出現(xiàn)的可能性小得多。
程序的運行結果如下:
復制代碼 代碼如下:

25358555
26399554
7051261
29855319
5383406

上面的運行結果可能在不同的運行環(huán)境上有所有同,但一般這五行輸出不會完全相同。從這個輸出結果可以看出,通過getInstance方法得到的對象實例是五個,而不是我們期望的一個。這是因為當一個線程執(zhí)行了Thread.yield()后,就將CPU資源交給了另外一個線程。由于在線程之間切換時并未執(zhí)行到創(chuàng)建Singleton對象實例的語句,因此,這幾個線程都通過了if判斷,所以,就會產(chǎn)生了建立五個對象實例的情況(可能創(chuàng)建的是四個或三個對象實例,這取決于有多少個線程在創(chuàng)建Singleton對象之前通過了if判斷,每次運行時可能結果會不一樣)。
要想使上面的單件模式變成線程安全的,只要為getInstance加上synchronized關鍵字即可。代碼如下:
public static synchronized Singleton getInstance() {   }
當然,還有更簡單的方法,就是在定義Singleton變量時就建立Singleton對象,代碼如下:
private static final Singleton sample = new Singleton();
然后在getInstance方法中直接將sample返回即可。這種方式雖然簡單,但不知在getInstance方法中創(chuàng)建Singleton對象靈活。讀者可以根據(jù)具體的需求選擇使用不同的方法來實現(xiàn)單件模式。

在使用synchronized關鍵字時有以下四點需要注意:

1.  synchronized關鍵字不能繼承。
雖然可以使用synchronized來定義方法,但synchronized并不屬于方法定義的一部分,因此,synchronized關鍵字不能被繼承。如果在父類中的某個方法使用了synchronized關鍵字,而在子類中覆蓋了這個方法,在子類中的這個方法默認情況下并不是同步的,而必須顯式地在子類的這個方法中加上synchronized關鍵字才可以。當然,還可以在子類方法中調用父類中相應的方法,這樣雖然子類中的方法不是同步的,但子類調用了父類的同步方法,因此,子類的方法也就相當于同步了。這兩種方式的例子代碼如下:
在子類方法中加上synchronized關鍵字

復制代碼 代碼如下:

class Parent
{
    public synchronized void method() {   }
}
class Child extends Parent
{
    public synchronized void method() {   }
}

在子類方法中調用父類的同步方法

復制代碼 代碼如下:

class Parent
{
    public synchronized void method() {   }
}
class Child extends Parent
{
    public void method() { super.method();   }
}

2.  在定義接口方法時不能使用synchronized關鍵字。
3.  構造方法不能使用synchronized關鍵字,但可以使用下節(jié)要討論的synchronized塊來進行同步。
4.  synchronized可以自由放置。
在前面的例子中使用都是將synchronized關鍵字放在方法的返回類型前面。但這并不是synchronized可放置唯一位置。在非靜態(tài)方法中,synchronized還可以放在方法定義的最前面,在靜態(tài)方法中,synchronized可以放在static的前面,代碼如下:

復制代碼 代碼如下:

public synchronized void method();
synchronized public void method();
public static synchronized void method();
public synchronized static void method();
synchronized public static void method();

但要注意,synchronized不能放在方法返回類型的后面,如下面的代碼是錯誤的:

復制代碼 代碼如下:

public void synchronized method();
public static void synchronized method();

synchronized關鍵字只能用來同步方法,不能用來同步類變量,如下面的代碼也是錯誤的。

復制代碼 代碼如下:

public synchronized int n = 0;
public static synchronized int n = 0;

雖然使用synchronized關鍵字同步方法是最安全的同步方式,但大量使用synchronized關鍵字會造成不必要的資源消耗以及性能損失。雖然從表面上看synchronized鎖定的是一個方法,但實際上synchronized鎖定的是一個類。也就是說,如果在非靜態(tài)方法method1和method2定義時都使用了synchronized,在method1未執(zhí)行完之前,method2是不能執(zhí)行的。靜態(tài)方法和非靜態(tài)方法的情況類似。但靜態(tài)和非靜態(tài)方法不會互相影響??纯慈缦碌拇a:

復制代碼 代碼如下:

package test;

public class MyThread1 extends Thread
{
    public String methodName;

    public static void method(String s)
    {
        System.out.println(s);
        while (true)

    }
    public synchronized void method1()
    {
        method("非靜態(tài)的method1方法");
    }
    public synchronized void method2()
    {
        method("非靜態(tài)的method2方法");
    }
    public static synchronized void method3()
    {
        method("靜態(tài)的method3方法");
    }
    public static synchronized void method4()
    {
        method("靜態(tài)的method4方法");
    }
    public void run()
    {
        try
        {
            getClass().getMethod(methodName).invoke(this);
        }
        catch (Exception e)
        {
        }
    }
    public static void main(String[] args) throws Exception
    {
        MyThread1 myThread1 = new MyThread1();
        for (int i = 1; i <= 4; i++)
        {
            myThread1.methodName = "method" + String.valueOf(i);
            new Thread(myThread1).start();
            sleep(100);
        }
    }
}

運行結果如下:

復制代碼 代碼如下:

非靜態(tài)的method1方法
靜態(tài)的method3方法

從上面的運行結果可以看出,method2和method4在method1和method3未結束之前不能運行。因此,我們可以得出一個結論,如果在類中使用synchronized關鍵字來定義非靜態(tài)方法,那將影響這個中的所有使用synchronized關鍵字定義的非靜態(tài)方法。如果定義的是靜態(tài)方法,那么將影響類中所有使用synchronized關鍵字定義的靜態(tài)方法。這有點象數(shù)據(jù)表中的表鎖,當修改一條記錄時,系統(tǒng)就將整個表都鎖住了,因此,大量使用這種同步方式會使程序的性能大幅度下降。

相關文章

  • Java8 Map中新增的方法使用總結

    Java8 Map中新增的方法使用總結

    這篇文章主要介紹了Java8 Map中新增的方法使用的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2018-11-11
  • Java中的異常和處理機制實例詳解

    Java中的異常和處理機制實例詳解

    這篇文章主要介紹了Java中的異常和處理機制,結合實例形式詳細分析了Java異常與處理機制的相關概念、原理、用法及操作注意事項,需要的朋友可以參考下
    2019-05-05
  • JavaSwing后臺播放音樂mp3

    JavaSwing后臺播放音樂mp3

    這篇文章主要為大家詳細介紹了JavaSwing后臺播放音樂mp3,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-06-06
  • 詳解Java中l(wèi)ist,set,map的遍歷與增強for循環(huán)

    詳解Java中l(wèi)ist,set,map的遍歷與增強for循環(huán)

    這篇文章主要介紹了詳解Java中l(wèi)ist,set,map的遍歷與增強for循環(huán)的相關資料,需要的朋友可以參考下
    2017-02-02
  • 淺析我對 String、StringBuilder、StringBuffer 的理解

    淺析我對 String、StringBuilder、StringBuffer 的理解

    StringBuilder、StringBuffer 和 String 一樣,都是用于存儲字符串的。這篇文章談談小編對String、StringBuilder、StringBuffer 的理解,感興趣的朋友跟隨小編一起看看吧
    2020-05-05
  • Spring Boot項目利用Redis實現(xiàn)session管理實例

    Spring Boot項目利用Redis實現(xiàn)session管理實例

    本篇文章主要介紹了Spring Boot項目利用Redis實現(xiàn)session管理實例,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • Java封裝數(shù)組之改進為泛型數(shù)組操作詳解

    Java封裝數(shù)組之改進為泛型數(shù)組操作詳解

    這篇文章主要介紹了Java封裝數(shù)組之改進為泛型數(shù)組操作,結合實例形式詳細分析了Java封裝數(shù)組為泛型數(shù)組相關原理、操作技巧與注意事項,需要的朋友可以參考下
    2020-03-03
  • 教你用Springboot實現(xiàn)攔截器獲取header內容

    教你用Springboot實現(xiàn)攔截器獲取header內容

    項目中遇到一個需求,對接上游系統(tǒng)是涉及到需要增加請求頭,請求頭的信息是動態(tài)獲取的,需要動態(tài)從下游拿到之后轉給上游,文中非常詳細的介紹了該需求的實現(xiàn),需要的朋友可以參考下
    2021-05-05
  • HashSet工作原理_動力節(jié)點Java學院整理

    HashSet工作原理_動力節(jié)點Java學院整理

    HashSet 底層采用 HashMap 來保存所有元素,因此 HashSet 的實現(xiàn)比較簡單。接下來通過本文給大家介紹HashSet工作原理_動力節(jié)點Java學院整理,需要的朋友可以參考下
    2017-04-04
  • Springboot和Jpa實現(xiàn)學生CRUD操作代碼實例

    Springboot和Jpa實現(xiàn)學生CRUD操作代碼實例

    這篇文章主要介紹了Springboot和Jpa實現(xiàn)學生CRUD操作代碼實例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-03-03

最新評論