java 两个线程同步_Java 多线程(二)—— 线程的同步
實現Runnable接口
public classTestThread2 {public static voidmain(String [] args){
Window window=newWindow();
Thread thread1=new Thread(window,"窗口一");
Thread thread2=new Thread(window,"窗口二");
Thread thread3=new Thread(window,"窗口三");
thread1.start();
thread2.start();
thread3.start();
}
}class Window implementsRunnable{int ticket=50;
@Overridepublic voidrun(){while (true){if(ticket > 0){try{
Thread.currentThread().sleep(100);//模擬賣票需要一定的時間
} catch(InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"售票,票號為:"+ticket--);
}else{break;
}
}
}
}
運行結果:
窗口二售票,票號為:13窗口三售票,票號為:12窗口一售票,票號為:11窗口二售票,票號為:10窗口一售票,票號為:10窗口三售票,票號為:10窗口三售票,票號為:9窗口一售票,票號為:8窗口二售票,票號為:7窗口三售票,票號為:6窗口一售票,票號為:5窗口二售票,票號為:4窗口三售票,票號為:3窗口一售票,票號為:2窗口二售票,票號為:1窗口三售票,票號為:0窗口一售票,票號為:-1
結果分析:這里出現了票數為0和負數還有重票的情況,這在現實生活中肯定是不存在的,那么為什么會出現這樣的情況呢?
當票號為10時:A線程、B線程、C線程同時進入到if(ticket > 0)的代碼塊中,A線程已經執行了打印輸出語句,但是還沒有做ticket--操作;
這時B線程就開始執行了打印操作,那么就會出現兩個線程打印票數一樣,即賣的是同一張票
當票號為1時:A線程、B線程,C線程同時進入到if(ticket > 0)的代碼塊中,A線程執行了打印語句,并且已經做完了ticket--操作,則此時ticket=0;
B線程再打印時就出現了0的情況,同理C線程打印就會出現-1的情況。
解決辦法:即我們不能同時讓超過兩個以上的線程進入到 if(ticket > 0)的代碼塊中,不然就會出現上述的錯誤。必須讓一個線程操作共享數據完畢以后,其他線程才有機會參與共享數據的操作。我們可以通過以下兩個辦法來解決:
1、使用 同步代碼塊
2、使用 同步方法
使用 同步代碼塊
synchronized(同步監視器){//需要被同步的代碼塊(即為操作共享數據的代碼)
}
同步監視器:由任意一個類的對象來充當,哪個線程獲取此監視器,誰就執行大括號里被同步的代碼。俗稱:鎖
要求:1、所有的線程必須公用同一把鎖!不能相對于線程是變化的對象;
2、并且只需鎖住操作共享數據的代碼,鎖多了或少了都不行;
實例:
1、實現的方式
public classTestWindow {public static voidmain(String [] args){
Window1 window=newWindow1();
Thread thread1=new Thread(window,"窗口一");
Thread thread2=new Thread(window,"窗口二");
Thread thread3=new Thread(window,"窗口三");
thread1.start();
thread2.start();
thread3.start();
}
}class Window1 implementsRunnable{int ticket=100;//共享數據
@Overridepublic voidrun(){while (true){synchronized (this){//this表示當前對象,此時表示創建的 window
if(ticket > 0){try{//模擬賣票需要一定的時間
Thread.sleep(100);
}catch(InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"售票,票號為:"+ticket--);
}
}
}
}
}
注意:在實現的方式中,考慮同步的話,可以使用this充當鎖,但在繼承的方式中,會創建多個對象,慎用this
2、繼承的方式
public classTestWindow1 {public static voidmain(String [] args){
Window2 window1=newWindow2();
Window2 window2=newWindow2();
window1.start();
window2.start();
}
}class Window2 extendsThread{static int ticket=100;//共享數據;注意聲明為 static,表示幾個窗口共享
static Object object=new Object();//用static 可以表示唯一
@Overridepublic voidrun(){while (true){//synchronized (this){//this表示當前對象,此時表示創建的 window1和window2
synchronized (object){//鎖必須是唯一,不能每個線程都使用自己的一把鎖
if(ticket > 0){try{
Thread.sleep(200);
}catch(InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"售票,票號為:"+ticket--);
}
}
}
}
}
注意:1、繼承的方式會創建多個實例,所以共享資源需要用static來修飾,表示共享
2、繼承的方式會創建多個實例,所以this?表示不同的實例對象,這里表示widow1和window2,所以不能使用this當鎖,此時可以定義一個?static?修飾的對象當鎖
使用 同步方法
語法:即用 ?synchronized ?關鍵字修飾方法
將操作共享數據的方法聲明為synchronized。即此方法為同步方法,能夠保證當其中一個線程執行此方法時,其他線程再外等待直至此線程執行完此方法。注意:同步方法的鎖:this
實例:
1、實現的方式
public classTestWindow2 {public static voidmain(String [] args){
Window3 window=newWindow3();
Thread thread1=new Thread(window,"窗口一");
Thread thread2=new Thread(window,"窗口二");
Thread thread3=new Thread(window,"窗口三");
thread1.start();
thread2.start();
thread3.start();
}
}class Window3 implementsRunnable{int ticket=100;//共享數據
@Overridepublic voidrun(){while (true){
show();
}
}public synchronized void show(){//this充當鎖,此時表示創建的 window;//如果用繼承的方式,使用同步方法,這里表示創建的 window1和window2,繼承的方式不要使用同步方法
if(ticket > 0){try{
Thread.currentThread().sleep(100);
}catch(InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"售票,票號為:"+ticket--);
}
}
}
注意:1、synchronized 的鎖為this,這里表示創建的對象實例window;
2、繼承的時候t this 表示創建的window1和window2,繼承的方式不要使用同步方法。
總結
以上是生活随笔為你收集整理的java 两个线程同步_Java 多线程(二)—— 线程的同步的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: [云炬创业管理笔记]第二章成为创业者讨论
- 下一篇: java 共享软件 保护_【Java并发