1. 程式人生 > >Java: synchronized詳解,靜態同步方法,普通同步方法,同步程式碼塊

Java: synchronized詳解,靜態同步方法,普通同步方法,同步程式碼塊

對程式碼進行同步控制我們可以選擇同步方法,也可以選擇同步塊,這兩種方式各有優缺點,至於具體選擇什麼方式,就見仁見智了,同步塊不僅可以更加精確的控制物件鎖,也就是控制鎖的作用域,何謂鎖的作用域?鎖的作用域就是從鎖被獲取到其被釋放的時間。而且可以選擇要獲取哪個物件的物件鎖。但是如果在使用同步塊機制時,如果使用過多的鎖也會容易引起死鎖問題,同時獲取和釋放所也有代價,而同步方法,它們所擁有的鎖就是該方法所屬的類的物件鎖,換句話說,也就是this物件,而且鎖的作用域也是整個方法,這可能導致其鎖的作用域可能太大,也有可能引起死鎖,同時因為可能包含了不需要進行同步的程式碼塊在內,也會降低程式的執行效率。而不管是同步方法還是同步塊,我們都不應該在他們的程式碼塊內包含無限迴圈,如果程式碼內部要是有了無限迴圈,那麼這個同步方法或者同步塊在獲取鎖以後因為程式碼會一直不停的迴圈著執行下去,也就沒有機會釋放它所獲取的鎖,而其它等待這把鎖的執行緒就永遠無法獲取這把鎖,這就造成了一種死鎖現象。

詳細解說一下同步方法的鎖,同步方法分為靜態同步方法與非靜態同步方法。

所有的非靜態同步方法用的都是同一把鎖——例項物件本身,也就是說如果一個例項物件的非靜態同步方法獲取鎖後,該例項物件的其他非靜態同步方法必須等待獲取鎖的方法釋放鎖後才能獲取鎖,可是別的例項物件的非靜態同步方法因為跟該例項物件的非靜態同步方法用的是不同的鎖,所以毋須等待該例項物件已獲取鎖的非靜態同步方法釋放鎖就可以獲取他們自己的鎖。

而所有的靜態同步方法用的也是同一把鎖——類物件本身,這兩把鎖是兩個不同的物件,所以靜態同步方法與非靜態同步方法之間是不會有競態條件的。但是一旦一個靜態同步方法獲取鎖後,其他的靜態同步方法都必須等待該方法釋放鎖後才能獲取鎖,而不管是同一個例項物件的靜態同步方法之間,還是不同的例項物件的靜態同步方法之間,只要它們同一個類的例項物件!

而對於同步塊,由於其鎖是可以選擇的,所以只有使用同一把鎖的同步塊之間才有著競態條件,這就得具體情況具體分析了,但這裡有個需要注意的地方,同步塊的鎖是可以選擇的,但是不是可以任意選擇的!!!!這裡必須要注意一個物理物件和一個引用物件的例項變數之間的區別!使用一個引用物件的例項變數作為鎖並不是一個好的選擇,因為同步塊在執行過程中可能會改變它的值,其中就包括將其設定為null,而對一個null物件加鎖會產生異常,並且對不同的物件加鎖也違背了同步的初衷!這看起來是很清楚的,但是一個經常發生的錯誤就是選用了錯誤的鎖物件,因此必須注意:同步是基於實際物件而不是物件引用的!多個變數可以引用同一個物件,變數也可以改變其值從而指向其他的物件,因此,當選擇一個物件鎖時,我們要根據實際物件而不是其引用來考慮!作為一個原則,不要選擇一個可能會在鎖的作用域中改變值的例項變數作為鎖物件!!!!

一、同步問題提出

執行緒的同步是為了防止多個執行緒訪問一個數據物件時,對資料造成的破壞。
例如:兩個執行緒ThreadA、ThreadB都操作同一個物件Foo物件,並修改Foo物件上的資料。

public class Foo { 
    private int x = 100; 

    public int getX() { 
        return x; 
    } 

    public int fix(int y) { 
        x = x - y; 
        return x; 
    } 
}

public class MyRunnable implements Runnable { 
    private Foo foo = new Foo(); 

    public static void main(String[] args) { 
        MyRunnable r = new MyRunnable(); 
        Thread ta = new Thread(r, "Thread-A"); 
        Thread tb = new Thread(r, "Thread-B"); 
        ta.start(); 
        tb.start(); 
    } 

    public void run() { 
        for (int i = 0; i < 3; i++) { 
            this.fix(30); 
            try { 
                Thread.sleep(1); 
            } catch (InterruptedException e) { 
                e.printStackTrace(); 
            } 
            System.out.println(Thread.currentThread().getName() + " : 當前foo物件的x值= " + foo.getX()); 
        } 
    } 

    public int fix(int y) { 
        return foo.fix(y); 
    } 
}

執行結果:

Thread-A : 當前foo物件的x值= 40 
Thread-B : 當前foo物件的x值= 40 
Thread-B : 當前foo物件的x值= -20 
Thread-A : 當前foo物件的x值= -50 
Thread-A : 當前foo物件的x值= -80 
Thread-B : 當前foo物件的x值= -80 

Process finished with exit code 0

從結果發現,這樣的輸出值明顯是不合理的。原因是兩個執行緒不加控制的訪問Foo物件並修改其資料所致。
如果要保持結果的合理性,只需要達到一個目的,就是將對Foo的訪問加以限制,每次只能有一個執行緒在訪問。這樣就能保證Foo物件中資料的合理性了。

在具體的Java程式碼中需要完成一下兩個操作:
把競爭訪問的資源類Foo變數x標識為private;
同步哪些修改變數的程式碼,使用synchronized關鍵字同步方法或程式碼。

二、同步和鎖定

1、鎖的原理

Java中每個物件都有一個內建鎖

當程式執行到非靜態的synchronized同步方法上時,自動獲得與正在執行程式碼類的當前例項(this例項)有關的鎖。獲得一個物件的鎖也稱為獲取鎖、鎖定物件、在物件上鎖定或在物件上同步。

當程式執行到synchronized同步方法或程式碼塊時才該物件鎖才起作用。

一個物件只有一個鎖。所以,如果一個執行緒獲得該鎖,就沒有其他執行緒可以獲得鎖,直到第一個執行緒釋放(或返回)鎖。這也意味著任何其他執行緒都不能進入該物件上的synchronized方法或程式碼塊,直到該鎖被釋放。

釋放鎖是指持鎖執行緒退出了synchronized同步方法或程式碼塊。

關於鎖和同步,有以下幾個要點:
1)、只能同步方法,而不能同步變數和類;
2)、每個物件只有一個鎖;當提到同步時,應該清楚在什麼上同步?也就是說,在哪個物件上同步?
3)、不必同步類中所有的方法,類可以同時擁有同步和非同步方法。
4)、如果兩個執行緒要執行一個類中的synchronized方法,並且兩個執行緒使用相同的例項來呼叫方法,那麼一次只能有一個執行緒能夠執行方法,另一個需要等待,直到鎖被釋放。也就是說:如果一個執行緒在物件上獲得一個鎖,就沒有任何其他執行緒可以進入(該物件的)類中的任何一個同步方法。
5)、如果執行緒擁有同步和非同步方法,則非同步方法可以被多個執行緒自由訪問而不受鎖的限制。
6)、執行緒睡眠時,它所持的任何鎖都不會釋放。
7)、執行緒可以獲得多個重進入(synchronized )鎖。比如,在一個物件的同步方法裡面呼叫另外一個物件的同步方法,則獲取了兩個物件的同步鎖。
8)、同步損害併發性,應該儘可能縮小同步範圍。同步不但可以同步整個方法,還可以同步方法中一部分程式碼塊。
9)、在使用同步程式碼塊時候,應該指定在哪個物件上同步,也就是說要獲取哪個物件的鎖。例如:

public int fix(int y) {
    synchronized (this) {
        x = x - y;
    }
    return x;
}
  當然,同步方法也可以改寫為非同步方法,但功能完全一樣的,例如:

public synchronized int getX() {
    return x++;
}
與

public int getX() {
    synchronized (this) {
        return x;
    }
}

效果是完全一樣的。

三、靜態方法同步

要同步靜態方法,需要一個用於整個類物件的鎖,這個物件是就是這個類(XXX.class)。
例如:

public static synchronized int setName(String name){
      Xxx.name = name;
}
等價於

public static int setName(String name){
      synchronized(Xxx.class){
            Xxx.name = name;
      }
}

四、如果執行緒不能不能獲得鎖會怎麼樣

如果執行緒試圖進入同步方法,而其鎖已經被佔用,則執行緒在該物件上被阻塞。實質上,執行緒進入該物件的的一種池中,必須在哪裡等待,直到其鎖被釋放,該執行緒再次變為可執行或執行為止。

當考慮阻塞時,一定要注意哪個物件正被用於鎖定:
1、呼叫同一個物件中非靜態同步方法的執行緒將彼此阻塞。如果是不同物件,則每個執行緒有自己的物件的鎖,執行緒間彼此互不干預。

2、呼叫同一個類中的靜態同步方法的執行緒將彼此阻塞,它們都是鎖定在相同的Class物件上。

3、靜態同步方法和非靜態同步方法將永遠不會彼此阻塞,因為靜態方法鎖定在Class物件上,非靜態方法鎖定在該類的物件上。

4、對於同步程式碼塊,要看清楚什麼物件已經用於鎖定(synchronized後面括號的內容)。在同一個物件上進行同步的執行緒將彼此阻塞,在不同物件上鎖定的執行緒將永遠不會彼此阻塞。

五、何時需要同步

在多個執行緒同時訪問互斥(可交換)資料時,應該同步以保護資料,確保兩個執行緒不會同時修改更改它。

對於非靜態欄位中可更改的資料,通常使用非靜態方法訪問。
對於靜態欄位中可更改的資料,通常使用靜態方法訪問。

如果需要在非靜態方法中使用靜態欄位,或者在靜態欄位中呼叫非靜態方法,問題將變得非常複雜。已經超出SJCP考試範圍了。

六、執行緒安全類

當一個類已經很好的同步以保護它的資料時,這個類就稱為“執行緒安全的”。
即使是執行緒安全類,也應該特別小心,因為操作的執行緒是間仍然不一定安全。

舉個形象的例子,比如一個集合是執行緒安全的,有兩個執行緒在操作同一個集合物件,當第一個執行緒查詢集合非空後,刪除集合中所有元素的時候。第二個執行緒也來執行與第一個執行緒相同的操作,也許在第一個執行緒查詢後,第二個執行緒也查詢出集合非空,但是當第一個執行清除後,第二個再執行刪除顯然是不對的,因為此時集合已經為空了。
看個程式碼:

public class NameList { 
    private List nameList = Collections.synchronizedList(new LinkedList()); 

    public void add(String name) { 
        nameList.add(name); 
    } 

    public String removeFirst() { 
        if (nameList.size() > 0) { 
            return (String) nameList.remove(0); 
        } else { 
            return null; 
        } 
    } 
}

public class Test { 
    public static void main(String[] args) { 
        final NameList nl = new NameList(); 
        nl.add("aaa"); 
        class NameDropper extends Thread{ 
            public void run(){ 
                String name = nl.removeFirst(); 
                System.out.println(name); 
            } 
        } 

        Thread t1 = new NameDropper(); 
        Thread t2 = new NameDropper(); 
        t1.start(); 
        t2.start(); 
    } 
}

雖然集合物件
private List nameList = Collections.synchronizedList(new LinkedList());是同步的,但是程式還不是執行緒安全的。出現這種事件的原因是,上例中一個執行緒操作列表過程中無法阻止另外一個執行緒對列表的其他操作。

解決上面問題的辦法是,在操作集合物件的NameList上面做一個同步。改寫後的程式碼如下:

public class NameList { 
    private List nameList = Collections.synchronizedList(new LinkedList()); 

    public synchronized void add(String name) { 
        nameList.add(name); 
    } 

    public synchronized String removeFirst() { 
        if (nameList.size() > 0) { 
            return (String) nameList.remove(0); 
        } else { 
            return null; 
        } 
    } 
}

這樣,當一個執行緒訪問其中一個同步方法時,其他執行緒只有等待。

七、執行緒死鎖

死鎖對Java程式來說,是很複雜的,也很難發現問題。當兩個執行緒被阻塞,每個執行緒在等待另一個執行緒時就發生死鎖。

還是看一個比較直觀的死鎖例子:

public class DeadlockRisk { 
    private static class Resource { 
        public int value; 
    } 

    private Resource resourceA = new Resource(); 
    private Resource resourceB = new Resource(); 

    public int read() { 
        synchronized (resourceA) { 
            synchronized (resourceB) { 
                return resourceB.value + resourceA.value; 
            } 
        } 
    } 

    public void write(int a, int b) { 
        synchronized (resourceB) { 
            synchronized (resourceA) { 
                resourceA.value = a; 
                resourceB.value = b; 
            } 
        } 
    } 
}

假設read()方法由一個執行緒啟動,write()方法由另外一個執行緒啟動。讀執行緒將擁有resourceA鎖,寫執行緒將擁有resourceB鎖,兩者都堅持等待的話就出現死鎖。

實際上,上面這個例子發生死鎖的概率很小。因為在程式碼內的某個點,CPU必須從讀執行緒切換到寫執行緒,所以,死鎖基本上不能發生。

但是,無論程式碼中發生死鎖的概率有多小,一旦發生死鎖,程式就死掉。有一些設計方法能幫助避免死鎖,包括始終按照預定義的順序獲取鎖這一策略。已經超出SCJP的考試範圍。

八、執行緒同步小結

1、執行緒同步的目的是為了保護多個執行緒反問一個資源時對資源的破壞。
2、執行緒同步方法是通過鎖來實現,每個物件都有切僅有一個鎖,這個鎖與一個特定的物件關聯,執行緒一旦獲取了物件鎖,其他訪問該物件的執行緒就無法再訪問該物件的其他同步方法。
3、對於靜態同步方法,鎖是針對這個類的,鎖物件是該類的Class物件。靜態和非靜態方法的鎖互不干預。一個執行緒獲得鎖,當在一個同步方法中訪問另外物件上的同步方法時,會獲取這兩個物件鎖。
4、對於同步,要時刻清醒在哪個物件上同步,這是關鍵。
5、編寫執行緒安全的類,需要時刻注意對多個執行緒競爭訪問資源的邏輯和安全做出正確的判斷,對“原子”操作做出分析,並保證原子操作期間別的執行緒無法訪問競爭資源。
6、當多個執行緒等待一個物件鎖時,沒有獲取到鎖的執行緒將發生阻塞。
7、死鎖是執行緒間相互等待鎖鎖造成的,在實際中發生的概率非常的小。真讓你寫個死鎖程式,不一定好使,呵呵。但是,一旦程式發生死鎖,程式將死掉。

package com.etrip.concurrent.executor;   

import java.util.Collections;   
import java.util.HashMap;   
import java.util.Iterator;   
import java.util.Map;   
import java.util.Map.Entry;   
import java.util.Set;   
/**  
 * 非靜態同步方法,靜態同步方法,同步語句塊的使用  
 *   
 *   
 * 進行多執行緒程式設計,同步控制是非常重要的,而同步控制就涉及到了鎖。   

       對程式碼進行同步控制我們可以選擇同步方法,也可以選擇同步塊,這兩種方式各有優缺點,至於具體選擇什麼方式,就見仁見智了,同步塊不僅可以更加精確的控制物件鎖,也就是控制鎖的作用域,何謂鎖的作用域?鎖的作用域就是從鎖被獲取到其被釋放的時間。而且可以選擇要獲取哪個物件的物件鎖。但是如果在使用同步塊機制時,如果使用過多的鎖也會容易引起死鎖問題,同時獲取和釋放所也有代價,而同步方法,它們所擁有的鎖就是該方法所屬的類的物件鎖,換句話說,也就是this物件,而且鎖的作用域也是整個方法,這可能導致其鎖的作用域可能太大,也有可能引起死鎖,同時因為可能包含了不需要進行同步的程式碼塊在內,也會降低程式的執行效率。而不管是同步方法還是同步塊,我們都不應該在他們的程式碼塊內包含無限迴圈,如果程式碼內部要是有了無限迴圈,那麼這個同步方法或者同步塊在獲取鎖以後因為程式碼會一直不停的迴圈著執行下去,也就沒有機會釋放它所獲取的鎖,而其它等待這把鎖的執行緒就永遠無法獲取這把鎖,這就造成了一種死鎖現象。   
 *   
 * @author longgangbai  
 */  
public class StaticInstanceLock {   


    private   int count;   
    private  static  StaticInstanceLock  instance=null;   
    private StaticInstanceLock(){   
    }   
    /**  
     * 靜態方法的鎖  
     *   
     * @return  
     */  
    public static synchronized StaticInstanceLock getInstance(){   
        if(instance==null){   
            instance=new  StaticInstanceLock();   
        }   
        return instance;   
    }   

    /**  
     * 非靜態方法的鎖  
     * @return  
     */  
    public synchronized int getCount(){   
        return count;   
    }   

    public synchronized  void setCount(int count){   
        this.count=count;   
    }   
    /**  
     * 同步語句塊的使用  
     *   
     */  
    public void synmethod(){   
              //HashMap為非安全性Map   
              HashMap<String,String> hashmap = new HashMap<String,String>();   
              hashmap.put("ZH","中國");   
              hashmap.put("EN","英國");   
              hashmap.put("AM","美國");   
              hashmap.put("FR","法國");   

              //建立一個同步的物件Map   
              Map<String,String> m = Collections.synchronizedMap(hashmap);   
              Set<String> s = m.keySet();  // Needn't be in synchronized block   
              //這裡同步的物件均為需要使用同步的物件如Map而非Set   
              synchronized(m) {  // Synchronizing on m, not s!   
                  Iterator<String> i = s.iterator(); // Must be in synchronized block   
                  while (i.hasNext()){   
                      foo(i.next());   
                  }   
              }   
    }   

    public void foo(String entry){   
        System.out.println("StaticInstanceLock ="+entry);   
    }   

    public static void main(String[] args) {   

        StaticInstanceLock instance=StaticInstanceLock.getInstance();   
        instance.setCount(7);   
        int count = instance.getCount();   
        instance.synmethod();   
    }   

}  

相關推薦

Java: synchronized靜態同步方法普通同步方法同步程式碼

對程式碼進行同步控制我們可以選擇同步方法,也可以選擇同步塊,這兩種方式各有優缺點,至於具體選擇什麼方式,就見仁見智了,同步塊不僅可以更加精確的控制物件鎖,也就是控制鎖的作用域,何謂鎖的作用域?鎖的作用域就是從鎖被獲取到其被釋放的時間。而且可以選擇要獲取哪個物

同步鎖定Java synchronized(C#翻譯)

? lock (new { }.GetType()){} synchronized有兩種用法,一種是在方法定義時使用,多執行緒狀態下,這個方法只能同時被同一個執行緒執行;另一種就是你問到的這種情況,用於鎖定程式碼段,也就是說,{ }括

Java synchronized

ret 內置 etc 優先權 one lan string 作用域 靜態成員函數 轉自:http://www.cnblogs.com/devinzhang/archive/2011/12/14/2287675.html 第一篇: 使用synchronized 在編寫一個

java synchronized(轉載)

正文: java synchronized詳解 記下來,很重要。 Java語言的關鍵字,當它用來修飾一個方法或者一個程式碼塊的時候,能夠保證在同一時刻最多隻有一個執行緒執行該段程式碼。 一、當兩個併發執行緒訪問同一個物件object中的這個sync

Java同步關鍵字synchronized

前言 多執行緒程式設計可以極大地提高了效率,但也會帶來執行緒安全問題。比如說多個執行緒向資料庫插入資料,就可能會導致資料庫中資料重複。 什麼時候會引發執行緒安全問題 首先我需要了解什麼是臨界資源?有這樣一種資源,在某一時刻只能被一個執行緒所使用,這種資源可以是各

Java併發——執行緒同步Volatile與Synchronized

0. 前言 轉載請註明出處:http://blog.csdn.net/seu_calvin/article/details/52370068 面試時很可能遇到這樣一個問題:使用volatile修飾int型變數i,多個執行緒同時進行i++操作,這樣可以實現執行緒安全嗎?提到

Java BigDecimal提供了豐富的四舍五入規則

字節 equals mat hashcode 字符 plain move man gnu java.math.BigDecimal類提供用於算術,刻度操作,舍入,比較,哈希算法和格式轉換操作。 toString()方法提供BigDecimal的規範表示。它使用戶可以完全控制

Java BigDecimal提供了豐富的四捨五入規則

原文地址:https://www.cnblogs.com/qynprime/p/8028397.html java.math.BigDecimal類提供用於算術,刻度操作,舍入,比較,雜湊演算法和格式轉換操作。 toString()方法提供BigDecimal的規範表示。它使使用者可以完全控

java執行緒 - synchronized

synchronized : 代表同步的修飾符 目錄 1.特性: 2.互斥性demo: 3.重入性講解 1.特性: 互斥性(排他性) : 同一時刻,只允許一個執行緒進行訪問,會一直等待執行緒釋放鎖,容易造成死鎖。 重入性: 程式或者子程式可以在任意時刻被中斷然

Java BigDecimal使用 Java BigDecimal BigDecimal的用法(保留兩位小數,四捨五入,數字格式化科學計數法轉數字數字裡的逗號處理)

1.引言  float和double型別的主要設計目標是為了科學計算和工程計算。他們執行二進位制浮點運算,這是為了在廣域數值範圍上提供較為精確的快速近似計算而精心設計的。然而,它們沒有提供完全精確的結果,所以不應該被用於要求精確結果的場合。但是,商業計算往往要求結果精確,這時候BigDecimal

C# 反射:定義、建立物件、呼叫例項方法靜態方法

    1、反射的定義及功能介紹:審查元資料並收集關於它的型別資訊的能力。元資料(編譯以後的最基本資料單元)就是一大堆的表,當編譯程式集或者模組時,編譯器會建立一個類定義表,一個欄位定義表,和一個方法定義表等。可能這些說的比較抽象。我再從另一個角度來說:反射是.Net中

JAVA多執行緒synchronized

Java語言的關鍵字,當它用來修飾一個方法或者一個程式碼塊的時候,能夠保證在同一時刻最多隻有一個執行緒執行該段程式碼。 當兩個併發執行緒訪問同一個物件object中的這個synchronized(this)同步程式碼塊時,一個時間內只能有一個執行緒得到執行。另一個執行緒必須等待當前執行緒執行完這個程

Java註解自定義註解利用反射解析註解

概要 這篇文章將會帶領你瞭解Java註解,註解的使用,註解的解析,利用反射解析執行時註解,相信有一定Java基礎的小夥伴一定會接觸大量的註解,Spring , Hibernate , MyBatis等著名的框架也有很多關於註解方面的應用,對於註解的使用小夥伴們

【夯實基礎】java關鍵字synchronized

尊重版權:http://www.cnblogs.com/GnagWang/archive/2011/02/27/1966606.html Java語言的關鍵字,當它用來修飾一個方法或者一個程式碼塊的時候,能夠保證在同一時刻最多隻有一個執行緒執行該段程式碼。      一、當兩個併發執行緒訪問同一個物件obj

Java集合3:Iteratorfail-fast機制與比較器

而不是 維護 統一 做了 count end 代碼 continue 來替 Java集合詳解3:Iterator,fail-fast機制與比較器 今天我們來探索一下LIterator,fail-fast機制與比較器的源碼。 具體代碼在我的GitHub中可以找到 https:

Java併發synchronized

今天和大家一起學習下併發程式設計,先舉一個簡單的生活例子,我們去醫院或者銀行排隊叫號,那每個工作人員之間如何保證不會叫重號呢? public class TicketDemo extends Thread { private int index = 1; private st

java架構之路(多執行緒)synchronized以及鎖的膨脹升級過程

  上幾次部落格,我們把volatile基本都說完了,剩下的還有我們的synchronized,還有我們的AQS,這次部落格我來說一下synchronized的使用和原理。   synchronized是jvm內部的一把隱式鎖,一切的加鎖和解鎖過程是由jvm虛擬機器來控制的,不需要我們認為的干預,我們大致從瞭

Java多執行緒之synchronized

## 目錄 - synchronized簡介 - 同步的原理 - 物件頭與鎖的實現 - 鎖的優化與升級 - Monitor Record - 鎖的對比 ## synchronized簡介 `synchronized`關鍵字,一般稱之為“同步鎖”或者重量級鎖(JAVA SE 1.6之後引入了`偏向鎖`和`輕

java final

inline mon 技術 但是 common src strac 都是 機制 final關鍵字 可用於聲明屬性、方法、和類 final類: final 類 不可被繼承,沒有子類,包括其中的方法默認都是final方法; final 類的方法不可被重寫,但是其中沒有被f

Java IO

amr mst 數據丟失 網上 ria break idc png 字符串 學習Java的同學註意了!!! 學習過程中遇到什麽問題或者想獲取學習資源的話,歡迎加入Java學習交流群,群號碼:618528494 我們一起學Java! 初學Java,一直搞不懂Java裏