文章詳情頁
菜鳥初學(xué)Java的備忘錄(七)
瀏覽:3日期:2024-06-25 08:20:20
內(nèi)容: 我突然發(fā)現(xiàn)還有很多東西需要我弄明白,比如synchronized這個關(guān)鍵字的用法.因?yàn)樵谖易蛱爝M(jìn)行創(chuàng)建連接池套接字的研究的時候,發(fā)現(xiàn)假如我不弄清楚這個概念,根本就無法進(jìn)行下去,所以我決定將自己對Socket的興趣先冷卻一下,而回過頭來看synchronized.看了一上午的Think in Java,覺得還是卓有成效的,應(yīng)該立即寫下來加深印象.我感覺自己的大腦可重用性極低,總是需要生成新的記憶對象,從而耗費(fèi)許多重復(fù)勞動.所以象記錄,分析,總結(jié)這樣類似的工作應(yīng)該多多益善.要弄清synchronized的用法,首先要知道它是用來解決什么問題的.既然synchronized是同步的意思,那么它當(dāng)然就是來解決不同步的問題的.下面就舉一個不同步的例子來演示可能出現(xiàn)的問題.在這個例子當(dāng)中,我們會創(chuàng)建兩個線程類.一個叫TwoCounter,其工作是對兩個計數(shù)器變量同時進(jìn)行累加,從1開始,你馬上會想道,我們是要用它來實(shí)現(xiàn)一個同步.另一個對象叫Watcher,顧名思義,是用來做監(jiān)視工作的,它負(fù)責(zé)檢查TwoCounter線程中的兩個計數(shù)器的值是否相等,看起來這似乎是毫無意義的工作,因?yàn)榧热皇峭嚼奂拥?那么兩個計數(shù)器的值怎么可能不相等呢??但,事實(shí)情況不是這樣的.我們先來看程序.在看這個程序之前,最好先翻翻Think in Java的14.2.1,我的程序?qū)嶋H上是根據(jù)該節(jié)中給出的例子簡化的,其中的主類改作了Sharing2class TwoCounter extends Thread { private int count1 = 0, count2 = 0; private boolean started=false; public void start(){ if (!started) file://防止多次對一個線程調(diào)用Start方法 { started=true; super.start(); } } public void run() { while (true) { count1++;file://如果TwoCounter運(yùn)行到這個時候,CPU時間片被分配給了Watcher,那么這個時候Watcher讀出來的兩個計數(shù)器的值當(dāng)然會不一樣了,這個可能性是存在的。“這是由線程的本質(zhì)造成的——它們可在任何時候掛起(暫停)。所以在上述兩行的執(zhí)行時刻之間,有時會出現(xiàn)執(zhí)行暫停現(xiàn)象。同時,Watcher線程也正好跟隨著進(jìn)來,并正好在這個時候進(jìn)行比較,造成計數(shù)器出現(xiàn)不相等的情況.(Think in Java) count2++; System.out.println('Count1='+count1+',Count2='+count2); try { sleep(500); } catch (InterruptedException e){} } } public void synchTest() { Sharing2.incrementAccess(); if(count1 != count2) System.out.println('Unsynched');//一旦發(fā)現(xiàn)不同步,立即顯示 }}class Watcher extends Thread { private Sharing2 p; public Watcher(Sharing2 p) { this.p = p; start(); } public void run() { while(true) { p.s.synchTest(); try { sleep(500); } catch (InterruptedException e){} } }}public class Sharing2 { TwoCounter s; private static int accessCount = 0; public static void incrementAccess() { accessCount++; System.out.println('accessCount='+accessCount); } public static void main(String[] args) { Sharing2 aaa = new Sharing2(); aaa.s=new TwoCounter(); aaa.s.start();//打開TwoCounter線程 new Watcher(aaa);//打開Watcher線程 }} 上面的注釋講得很清楚了,有可能出現(xiàn)不同步的情況.但奇怪的是,我在運(yùn)行的時候,卻始終沒有遇到不同步的情況,那么只有一種情況,就是程序中count1++和count2++幾乎是同時進(jìn)行的,watcher線程插不進(jìn)來,但是為什么Think in Java上面的程序運(yùn)行之后就肯定有不同步的情況呢?兩個程序的原理是完全一樣的,唯一不同的是我的程序較為簡單,并且在命令行下運(yùn)行,未使用GUI.難道是因?yàn)槭褂肁pplet方式運(yùn)行或者以Windows主窗口的方式運(yùn)行開銷更大,使得watcher有機(jī)可趁嗎?于是我試著在count1++和count2++之間加了一條循環(huán)語句,人為的增大空隙,目的是為了讓watcher好插進(jìn)來,造成監(jiān)測出來的count1不等于count2的情況,實(shí)現(xiàn)不同步.修改后的程序是這樣的 ...... count1++; for(int i=0;i
標(biāo)簽:
Java
相關(guān)文章:
排行榜