java多線程編程之使用Synchronized塊同步變量
下面的代碼演示了如何同步特定的類方法:
package mythread;
public class SyncThread extends Thread
{
private static String sync = "";
private String methodType = "";
private static void method(String s)
{
synchronized (sync)
{
sync = s;
System.out.println(s);
while (true);
}
}
public void method1()
{
method("method1");
}
public static void staticMethod1()
{
method("staticMethod1");
}
public void run()
{
if (methodType.equals("static"))
staticMethod1();
else if (methodType.equals("nonstatic"))
method1();
}
public SyncThread(String methodType)
{
this.methodType = methodType;
}
public static void main(String[] args) throws Exception
{
SyncThread sample1 = new SyncThread("nonstatic");
SyncThread sample2 = new SyncThread("static");
sample1.start();
sample2.start();
}
}
運行結(jié)果如下:
method1
staticMethod1
看到上面的運行結(jié)果很多讀者可能感到驚奇。在上面的代碼中method1和staticMethod1方法使用了靜態(tài)字符串變量sync進行同步。這兩個方法只能有一個同時執(zhí)行,而這兩個方法都會執(zhí)行014行的無限循環(huán)語句。因此,輸出結(jié)果只能是method1和staticMethod1其中之一。但這個程序?qū)⑦@兩個字符串都輸出了。
出現(xiàn)這種結(jié)果的愿意很簡單,我們看一下012行就知道了。原來在這一行將sync的值改變了。在這里要說一下Java中的String類型。String類型和Java中其他的復雜類型不同。在使用String型變量時,只要給這個變量賦一次值,Java就會創(chuàng)建個新的String類型的實例。如下面的代碼所示:
String s = "hello";
System.out.println(s.hashCode());
s = "world";
System.out.println(s.hashCode());
在上面的代碼中。第一個s和再次賦值后的s的hashCode的值是不一樣的。由于創(chuàng)建String類的實例并不需要使用new,因此,在同步String類型的變量時要注意不要給這個變量賦值,否則會使變量無法同步。
由于在012行已經(jīng)為sync創(chuàng)建了一個新的實例,假設method1先執(zhí)行,當method1方法執(zhí)行了013行的代碼后,sync的值就已經(jīng)不是最初那個值了,而method1方法鎖定的仍然是sync變量最初的那個值。而在這時,staticMethod1正好執(zhí)行到synchronized(sync),在staticMethod1方法中要鎖定的這個sync和method1方法鎖定的sync已經(jīng)不是一個了,因此,這兩個方法的同步性已經(jīng)被破壞了。
解決以上問題的方法當然是將012行去掉。在本例中加上這行,只是為了說明使用類變量來同步方法時如果在synchronized塊中將同步變量的值改變,就會破壞方法之間的同步。為了徹底避免這種情況發(fā)生,在定義同步變量時可以使用final關鍵字。如將上面的程序中的005行可改成如下形式:
private final static String sync = "";
使用final關鍵字后,sync只能在定義時為其賦值,并且以后不能再修改。如果在程序的其他地方給sync賦了值,程序就無法編譯通過。在Eclipse等開發(fā)工具中,會直接在錯誤的地方給出提示。
我們可以從兩個角度來理解synchronized塊。如果從類方法的角度來理解,可以通過類變量來同步相應的方法。如果從類變量的角度來理解,可以使用synchronized塊來保證某個類變量同時只能被一個方法訪問。不管從哪個角度來理解,它們的實質(zhì)都是一樣的,就是利用類變量來獲得同步鎖,通過同步鎖的互斥性來實現(xiàn)同步。
注意:在使用synchronized塊時應注意,synchronized塊只能使用對象作為它的參數(shù)。如果是簡單類型的變量(如int、char、boolean等),不能使用synchronized來同步。
- Java 中 synchronized的用法詳解(四種用法)
- 深入理解java中的synchronized關鍵字
- 詳解Java中synchronized關鍵字的死鎖和內(nèi)存占用問題
- Java 多線程同步 鎖機制與synchronized深入解析
- Java synchronized鎖升級jol過程詳解
- java中synchronized(同步代碼塊和同步方法)詳解及區(qū)別
- Java 同步鎖(synchronized)詳解及實例
- java多線程編程之使用Synchronized關鍵字同步類方法
- 深入理解java內(nèi)置鎖(synchronized)和顯式鎖(ReentrantLock)
- Java中關鍵字synchronized的使用方法詳解
相關文章
RestTemplate發(fā)送get和post請求,下載文件的實例
這篇文章主要介紹了RestTemplate發(fā)送get和post請求,下載文件的實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09Java 中 synchronized的用法詳解(四種用法)
Java語言的關鍵字,當它用來修飾一個方法或者一個代碼塊的時候,能夠保證在同一時刻最多只有一個線程執(zhí)行該段代碼。本文給大家介紹java中 synchronized的用法,對本文感興趣的朋友一起看看吧2015-11-11Java簡單統(tǒng)計字符串中漢字,英文字母及數(shù)字數(shù)量的方法
這篇文章主要介紹了Java簡單統(tǒng)計字符串中漢字,英文字母及數(shù)字數(shù)量的方法,涉及java針對字符串的遍歷、編碼轉(zhuǎn)換、判斷等相關操作技巧,需要的朋友可以參考下2017-06-06SpringBoot如何讀取war包jar包和Resource資源
這篇文章主要介紹了SpringBoot如何讀取war包jar包和Resource資源,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-01-01Elasticsearch寫入瓶頸導致skywalking大盤空白
這篇文章主要為大家介紹了Elasticsearch寫入瓶頸導致skywalking大盤空白的解決方案,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步2022-02-02