Java多线程初学者指南(12):使用Synchronized块同步变量
生活随笔
收集整理的這篇文章主要介紹了
Java多线程初学者指南(12):使用Synchronized块同步变量
小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.
我們可以通過(guò)synchronized塊來(lái)同步特定的靜態(tài)或非靜態(tài)方法。要想實(shí)現(xiàn)這種需求必須為這些特性的方法定義一個(gè)類變量,然后將這些方法的代碼用synchronized塊括起來(lái),并將這個(gè)類變量作為參數(shù)傳入synchronized塊。下面的代碼演示了如何同步特定的類方法:
??001??package mythread;
??002??
??003??public?class SyncThread extends?Thread
??004??{
??005??????private?static?String?sync?=?"";
??006??????private?String?methodType?=?"";
??007??
??008??????private?static?void?method(String?s)
??009??????{
??010??????????synchronized?(sync)
??011??????????{
??012??????????????sync?=?s;
??013??????????????System.out.println(s);
??014??????????????while?(true);
??015??????????}
??016??????}
??017??????public?void?method1()
??018??????{
??019??????????method("method1");
??020??????}
??021??????public?static?void?staticMethod1()
??022??????{
??023??????????method("staticMethod1");
??024??????}
??025??????public?void?run()
??026??????{
??027??????????if?(methodType.equals("static"))
??028??????????????staticMethod1();
??029??????????else?if?(methodType.equals("nonstatic"))
??030??????????????method1();
??031??????}
??032??????public SyncThread(String?methodType)
??033??????{
??034??????????this.methodType?=?methodType;
??035??????}
??036??????public?static?void?main(String[]?args)?throws?Exception
??037??????{
??038????????? SyncThread sample1?=?new SyncThread("nonstatic");
??039????????? SyncThread sample2?=?new SyncThread("static");
??040??????????sample1.start();
??041??????????sample2.start();
??042??????}
??043??}
運(yùn)行結(jié)果如下:
method1
staticMethod1
看到上面的運(yùn)行結(jié)果很多讀者可能感到驚奇。在上面的代碼中method1和staticMethod1方法使用了靜態(tài)字符串變量sync進(jìn)行同步。這兩個(gè)方法只能有一個(gè)同時(shí)執(zhí)行,而這兩個(gè)方法都會(huì)執(zhí)行014行的無(wú)限循環(huán)語(yǔ)句。因此,輸出結(jié)果只能是method1和staticMethod1其中之一。但這個(gè)程序?qū)⑦@兩個(gè)字符串都輸出了。 出現(xiàn)這種結(jié)果的愿意很簡(jiǎn)單,我們看一下012行就知道了。原來(lái)在這一行將sync的值改變了。在這里要說(shuō)一下Java中的String類型。String類型和Java中其他的復(fù)雜類型不同。在使用String型變量時(shí),只要給這個(gè)變量賦一次值,Java就會(huì)創(chuàng)建個(gè)新的String類型的實(shí)例。如下面的代碼所示:
String?s?=?"hello";
System.out.println(s.hashCode());
s?=?"world";
System.out.println(s.hashCode());?
在上面的代碼中。第一個(gè)s和再次賦值后的s的hashCode的值是不一樣的。由于創(chuàng)建String類的實(shí)例并不需要使用new,因此,在同步String類型的變量時(shí)要注意不要給這個(gè)變量賦值,否則會(huì)使變量無(wú)法同步。 由于在012行已經(jīng)為sync創(chuàng)建了一個(gè)新的實(shí)例,假設(shè)method1先執(zhí)行,當(dāng)method1方法執(zhí)行了013行的代碼后,sync的值就已經(jīng)不是最初那個(gè)值了,而method1方法鎖定的仍然是sync變量最初的那個(gè)值。而在這時(shí),staticMethod1正好執(zhí)行到synchronized(sync),在staticMethod1方法中要鎖定的這個(gè)sync和method1方法鎖定的sync已經(jīng)不是一個(gè)了,因此,這兩個(gè)方法的同步性已經(jīng)被破壞了。 解決以上問(wèn)題的方法當(dāng)然是將012行去掉。在本例中加上這行,只是為了說(shuō)明使用類變量來(lái)同步方法時(shí)如果在synchronized塊中將同步變量的值改變,就會(huì)破壞方法之間的同步。為了徹底避免這種情況發(fā)生,在定義同步變量時(shí)可以使用final關(guān)鍵字。如將上面的程序中的005行可改成如下形式:
private?final?static?String?sync?=?""; ??? 使用final關(guān)鍵字后,sync只能在定義時(shí)為其賦值,并且以后不能再修改。如果在程序的其他地方給sync賦了值,程序就無(wú)法編譯通過(guò)。在Eclipse等開(kāi)發(fā)工具中,會(huì)直接在錯(cuò)誤的地方給出提示。 ??? 我們可以從兩個(gè)角度來(lái)理解synchronized塊。如果從類方法的角度來(lái)理解,可以通過(guò)類變量來(lái)同步相應(yīng)的方法。如果從類變量的角度來(lái)理解,可以使用synchronized塊來(lái)保證某個(gè)類變量同時(shí)只能被一個(gè)方法訪問(wèn)。不管從哪個(gè)角度來(lái)理解,它們的實(shí)質(zhì)都是一樣的,就是利用類變量來(lái)獲得同步鎖,通過(guò)同步鎖的互斥性來(lái)實(shí)現(xiàn)同步。
注意:在使用synchronized塊時(shí)應(yīng)注意,synchronized塊只能使用對(duì)象作為它的參數(shù)。如果是簡(jiǎn)單類型的變量(如int、char、boolean等),不能使用synchronized來(lái)同步。 國(guó)內(nèi)最棒的Google Android技術(shù)社區(qū)(eoeandroid),歡迎訪問(wèn)!
《銀河系列原創(chuàng)教程》發(fā)布
《Java Web開(kāi)發(fā)速學(xué)寶典》出版,歡迎定購(gòu)
??001??package mythread;
??002??
??003??public?class SyncThread extends?Thread
??004??{
??005??????private?static?String?sync?=?"";
??006??????private?String?methodType?=?"";
??007??
??008??????private?static?void?method(String?s)
??009??????{
??010??????????synchronized?(sync)
??011??????????{
??012??????????????sync?=?s;
??013??????????????System.out.println(s);
??014??????????????while?(true);
??015??????????}
??016??????}
??017??????public?void?method1()
??018??????{
??019??????????method("method1");
??020??????}
??021??????public?static?void?staticMethod1()
??022??????{
??023??????????method("staticMethod1");
??024??????}
??025??????public?void?run()
??026??????{
??027??????????if?(methodType.equals("static"))
??028??????????????staticMethod1();
??029??????????else?if?(methodType.equals("nonstatic"))
??030??????????????method1();
??031??????}
??032??????public SyncThread(String?methodType)
??033??????{
??034??????????this.methodType?=?methodType;
??035??????}
??036??????public?static?void?main(String[]?args)?throws?Exception
??037??????{
??038????????? SyncThread sample1?=?new SyncThread("nonstatic");
??039????????? SyncThread sample2?=?new SyncThread("static");
??040??????????sample1.start();
??041??????????sample2.start();
??042??????}
??043??}
運(yùn)行結(jié)果如下:
method1
staticMethod1
看到上面的運(yùn)行結(jié)果很多讀者可能感到驚奇。在上面的代碼中method1和staticMethod1方法使用了靜態(tài)字符串變量sync進(jìn)行同步。這兩個(gè)方法只能有一個(gè)同時(shí)執(zhí)行,而這兩個(gè)方法都會(huì)執(zhí)行014行的無(wú)限循環(huán)語(yǔ)句。因此,輸出結(jié)果只能是method1和staticMethod1其中之一。但這個(gè)程序?qū)⑦@兩個(gè)字符串都輸出了。 出現(xiàn)這種結(jié)果的愿意很簡(jiǎn)單,我們看一下012行就知道了。原來(lái)在這一行將sync的值改變了。在這里要說(shuō)一下Java中的String類型。String類型和Java中其他的復(fù)雜類型不同。在使用String型變量時(shí),只要給這個(gè)變量賦一次值,Java就會(huì)創(chuàng)建個(gè)新的String類型的實(shí)例。如下面的代碼所示:
String?s?=?"hello";
System.out.println(s.hashCode());
s?=?"world";
System.out.println(s.hashCode());?
在上面的代碼中。第一個(gè)s和再次賦值后的s的hashCode的值是不一樣的。由于創(chuàng)建String類的實(shí)例并不需要使用new,因此,在同步String類型的變量時(shí)要注意不要給這個(gè)變量賦值,否則會(huì)使變量無(wú)法同步。 由于在012行已經(jīng)為sync創(chuàng)建了一個(gè)新的實(shí)例,假設(shè)method1先執(zhí)行,當(dāng)method1方法執(zhí)行了013行的代碼后,sync的值就已經(jīng)不是最初那個(gè)值了,而method1方法鎖定的仍然是sync變量最初的那個(gè)值。而在這時(shí),staticMethod1正好執(zhí)行到synchronized(sync),在staticMethod1方法中要鎖定的這個(gè)sync和method1方法鎖定的sync已經(jīng)不是一個(gè)了,因此,這兩個(gè)方法的同步性已經(jīng)被破壞了。 解決以上問(wèn)題的方法當(dāng)然是將012行去掉。在本例中加上這行,只是為了說(shuō)明使用類變量來(lái)同步方法時(shí)如果在synchronized塊中將同步變量的值改變,就會(huì)破壞方法之間的同步。為了徹底避免這種情況發(fā)生,在定義同步變量時(shí)可以使用final關(guān)鍵字。如將上面的程序中的005行可改成如下形式:
private?final?static?String?sync?=?""; ??? 使用final關(guān)鍵字后,sync只能在定義時(shí)為其賦值,并且以后不能再修改。如果在程序的其他地方給sync賦了值,程序就無(wú)法編譯通過(guò)。在Eclipse等開(kāi)發(fā)工具中,會(huì)直接在錯(cuò)誤的地方給出提示。 ??? 我們可以從兩個(gè)角度來(lái)理解synchronized塊。如果從類方法的角度來(lái)理解,可以通過(guò)類變量來(lái)同步相應(yīng)的方法。如果從類變量的角度來(lái)理解,可以使用synchronized塊來(lái)保證某個(gè)類變量同時(shí)只能被一個(gè)方法訪問(wèn)。不管從哪個(gè)角度來(lái)理解,它們的實(shí)質(zhì)都是一樣的,就是利用類變量來(lái)獲得同步鎖,通過(guò)同步鎖的互斥性來(lái)實(shí)現(xiàn)同步。
注意:在使用synchronized塊時(shí)應(yīng)注意,synchronized塊只能使用對(duì)象作為它的參數(shù)。如果是簡(jiǎn)單類型的變量(如int、char、boolean等),不能使用synchronized來(lái)同步。 國(guó)內(nèi)最棒的Google Android技術(shù)社區(qū)(eoeandroid),歡迎訪問(wèn)!
《銀河系列原創(chuàng)教程》發(fā)布
《Java Web開(kāi)發(fā)速學(xué)寶典》出版,歡迎定購(gòu)
總結(jié)
以上是生活随笔為你收集整理的Java多线程初学者指南(12):使用Synchronized块同步变量的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 疲态
- 下一篇: Keyword-Driven Testi