摘要:為什么線程同步當(dāng)我們有多個線程要對同一個資源進(jìn)行操作的時候,比如說文件,這時候就不能讓多個線程同時操作這個文件了。程序完善要解決上面出現(xiàn)的問題,就要用到我們最開始說的線程同步了。通過這個關(guān)鍵字,我們就能在執(zhí)行方法的時候,實現(xiàn)線程同步了。
為什么線程同步
當(dāng)我們有多個線程要對同一個資源進(jìn)行操作的時候,比如說文件,這時候就不能讓多個線程同時操作這個文件了。因為當(dāng)文件共享的時候,多個操作就會產(chǎn)生沖突。相信使用過git進(jìn)行團(tuán)隊開發(fā)的人就會有比價深刻的體會。
這時候就需要引入線程的同步機制。
所謂同步,是指一個一個執(zhí)行,也就是排隊,幾個線程排隊執(zhí)行,讓線程之間有一個先來后到的關(guān)系。這樣,就不會產(chǎn)生爭搶的現(xiàn)象,也就不會出現(xiàn)沖突。
問題描述一個有兩個線程的程序,第一個線程計算2~1000之間的偶數(shù)及個數(shù),第二個線程計算1000~2000之間的偶數(shù)及個數(shù)。
初步實現(xiàn)首先,我們根據(jù)題目來設(shè)計一下實現(xiàn)方式。
1.定義兩個線程。這里要定義自己的線程類,然后去設(shè)置計算偶數(shù)的開始和結(jié)束
2.實現(xiàn)計算所給范圍內(nèi)的偶數(shù)和偶數(shù)個數(shù)
3.執(zhí)行線程
可以看到實現(xiàn)方式很簡單。初步實現(xiàn)如下:
1.定義自己的線程類:public class MyTread extends Thread { private int begin; // 范圍開始 private int end; // 范圍結(jié)束 public MyTread(int begin, int end) { this.begin = begin; this.end = end; } }2.實現(xiàn)計算偶數(shù)以及偶數(shù)個數(shù)
@Override public void run() { this.getEven(begin, end); } // 獲取偶數(shù)以及個數(shù) private void getEven(int begin, int end) { int count = 0; System.out.println(begin + "~" + end + "之間的偶數(shù)為:"); if (begin % 2 != 0) { begin += 1; } for (; begin <= end; begin += 2) { System.out.print(begin + " "); count++; } System.out.println(); System.out.println("偶數(shù)個數(shù)為:" + count); }3.初始化線程,運行
public static void main(String[] args) { MyTread tread1 = new MyTread(2, 500); MyTread tread2 = new MyTread(500, 1000); tread1.start(); tread2.start(); }
注:這里為了展示后面的效果,所以將數(shù)字范圍減小。
看一下結(jié)果:
好像并不是我們想要的效果。我們想要這樣的效果:
我們想要每一部分的偶數(shù)和偶數(shù)個數(shù)都在一起,但是我們實現(xiàn)后的效果確實兩部分混亂出現(xiàn)的。
程序完善要解決上面出現(xiàn)的問題,就要用到我們最開始說的線程同步了。
根據(jù)我們開始的介紹,發(fā)現(xiàn)實現(xiàn)同步的一個條件就是有一個共享的資源。那么我們的代碼里什么是這個共享的資源呢?貌似沒有。所以先創(chuàng)建一個。
1.建立共享資源我們這里的共享資源完全就可以是同一個對象,所以我們就另外建立一個類,用來示例共享資源。
public class Even {}2.然后在共享類中建立計算偶數(shù)的方法:
public class Even { /** * 獲取某個范圍內(nèi)偶數(shù)以及個數(shù) * @param begin 統(tǒng)計開始 * @param end 統(tǒng)計結(jié)束 */ public synchronized void getEven(int begin, int end) { int count = 0; System.out.println(begin + "~" + end + "之間的偶數(shù)為:"); if (begin % 2 != 0) { begin += 1; } for (; begin <= end; begin += 2) { System.out.print(begin + " "); count++; } System.out.println(); System.out.println("偶數(shù)個數(shù)為:" + count); } }
細(xì)心的人會發(fā)現(xiàn),這個方法跟我們最開始寫的方法有點不同,它多了一個關(guān)鍵字:synchronized。通過這個關(guān)鍵字,我們就能在執(zhí)行方法的時候,實現(xiàn)線程同步了。
3.最后在線程中調(diào)用共享方法public class MyTread extends Thread { Even even; @Override public void run() { even.getEven(begin, end); } }
這時候,我們再來執(zhí)行一下:
實現(xiàn)了我們想要的效果。
最后,附上完整代碼:
/** * 偶數(shù)類 */ public class Even { /** * 獲取某個范圍內(nèi)偶數(shù)以及個數(shù) * @param begin 統(tǒng)計開始 * @param end 統(tǒng)計結(jié)束 */ public synchronized void getEven(int begin, int end) { int count = 0; System.out.println(begin + "~" + end + "之間的偶數(shù)為:"); if (begin % 2 != 0) { begin += 1; } for (; begin <= end; begin += 2) { System.out.print(begin + " "); count++; } System.out.println(); System.out.println("偶數(shù)個數(shù)為:" + count); } } public class MyTread extends Thread { Even even; private int begin; // 范圍開始 private int end; // 范圍結(jié)束 public MyTread(Even even, int begin, int end) { this.even = even; this.begin = begin; this.end = end; } @Override public void run() { even.getEven(begin, end); } public static void main(String[] args) { Even even = new Even(); MyTread tread1 = new MyTread(even, 2, 500); MyTread tread2 = new MyTread(even, 500, 1000); tread1.start(); tread2.start(); } }
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/71764.html
摘要:的關(guān)鍵字中的塊使用關(guān)鍵字進(jìn)行標(biāo)記。由于每個類只有一個類對象存在于中,因此全局同時只有一個線程能夠進(jìn)入到同一個類的靜態(tài)同步方法中。同步代碼塊使這種期望成為可能。注意同步代碼塊如何在括號中接受一個對象。相同的實例被傳入兩個不同的線程實例中。 Java的synchronized塊標(biāo)記一個方法或一個代碼塊為同步的。synchronized塊能用于防止出現(xiàn)競態(tài)條件。 Java的synchroni...
摘要:多線程和并發(fā)問題是技術(shù)面試中面試官比較喜歡問的問題之一。線程可以被稱為輕量級進(jìn)程。一個守護(hù)線程是在后臺執(zhí)行并且不會阻止終止的線程。其他的線程狀態(tài)還有,和。上下文切換是多任務(wù)操作系統(tǒng)和多線程環(huán)境的基本特征。 多線程和并發(fā)問題是 Java 技術(shù)面試中面試官比較喜歡問的問題之一。在這里,從面試的角度列出了大部分重要的問題,但是你仍然應(yīng)該牢固的掌握J(rèn)ava多線程基礎(chǔ)知識來對應(yīng)日后碰到的問題。(...
摘要:線程可以被稱為輕量級進(jìn)程。一個守護(hù)線程是在后臺執(zhí)行并且不會阻止終止的線程。其他的線程狀態(tài)還有,和。上下文切換是多任務(wù)操作系統(tǒng)和多線程環(huán)境的基本特征。在的線程中并沒有可供任何對象使用的鎖和同步器。 原文:Java Multi-Threading and Concurrency Interview Questions with Answers 翻譯:并發(fā)編程網(wǎng) - 鄭旭東 校對:方騰飛 多...
摘要:死亡狀態(tài)線程退出有可能是正常執(zhí)行完成也有可能遇見異常退出。類有新建與死亡狀態(tài)返回其余狀態(tài)返回判斷線程是否存活。線程因某些原因進(jìn)入阻塞狀態(tài)。執(zhí)行同步代碼塊的過程中執(zhí)行了當(dāng)前線程放棄開始睡眠進(jìn)入就緒狀態(tài)但是不會釋放鎖。 【java內(nèi)存模型簡介 JVM中存在一個主存區(qū)(Main Memory或Java Heap Memory),Java中所有變量都是存在主存中的,對于所有線程進(jìn)行共享,而每個...
閱讀 1107·2021-11-24 10:24
閱讀 2594·2021-11-22 13:54
閱讀 1000·2021-09-24 09:55
閱讀 3603·2019-08-30 15:54
閱讀 1318·2019-08-30 15:44
閱讀 1096·2019-08-30 14:23
閱讀 3203·2019-08-29 13:45
閱讀 1284·2019-08-29 11:19