程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> 關於JAVA >> Java多線程編程中synchronized線程同步的教程

Java多線程編程中synchronized線程同步的教程

編輯:關於JAVA

Java多線程編程中synchronized線程同步的教程。本站提示廣大學習愛好者:(Java多線程編程中synchronized線程同步的教程)文章只能為提供參考,不一定能成為您想要的結果。以下是Java多線程編程中synchronized線程同步的教程正文


0.關於線程同步
(1)為何須要同步多線程?
線程的同步是指讓多個運轉的線程在一路優越地協作,到達讓多線程按請求公道地占用釋放資本。我們采取Java中的同步代碼塊和同步辦法到達如許的目標。好比如許的處理多線程無固定序履行的成績:

public class TwoThreadTest {
  public static void main(String[] args) {
    Thread th1= new MyThread1();
    Thread th2= new MyThread2();
    th1.start();
    th2.start();
  }
}

class MyThread2 extends Thread{
  @Override
  public void run() {
    for( int i=0;i<10;i++)
      System. out.println( "thread 1 counter:"+i);
  }
}

class MyThread1 extends Thread{
  @Override
  public void run() {
    for( int i=0;i<10;i++)
      System. out.println( "thread 2 counter:"+i);
  }  
}

這類狀況下多線程履行的成果是隨機地去隨意率性拔出履行,這完整取決於JVM關於線程的調劑,在許多請求定序履行的情形下,這類隨機履行的狀況明顯是不合請求的。

public class ThreadTest {
  public static void main(String[] args) {
    MyThread thread = new MyThread();
    Thread th1= new Thread(thread);
    Thread th2= new Thread(thread);
    th1.start();
    th2.start();
  }

}

class MyThread implements Runnable{
  @Override
  public synchronized void run() {
    for( int i=0;i<10;i++)
      System. out.println(Thread. currentThread().getName()+" counter:"+i);
  }
}

應用了同步辦法後我們便可以掌握線程獨有履行體對象,如許在履行的進程中便可以使得線程將履行體上的義務一次性履行完撤退退卻出鎖定狀況,JVM再調劑另外一個線程出去一次性運轉履行體內的義務。

(2)線程創立運轉的范式:
在之前我們也有本身的線程創立和運轉的編程范式,普通是界說一個履行類重寫run()辦法,然則這類方法將履行體和履行的義務放在了一路,從軟件工程的角度來看晦氣於解耦。一個線程的履行的意思是說線程經由過程履行對象履行了某個對象的某個義務,從這個角度來講,將義務的劃定者從履行類平分離出來可使很多線程編程的各個腳色了了出來,進而取得優越地解耦,以下就是線程創立和履行的編程范式:

public class FormalThreadClass {
  public static void main(String[] args) {
    Thread thread = new Thread( new MyRunnable());
    thread.start();
  }
}

class MyRunnable implements Runnable{
  MyTask myTask = new MyTask();
  @Override
  public void run() {
    myTask.doTask();
  }
}

class MyTask{
  public void doTask() {
    System. out.println( "This is real Tasking");
  }
}


1. synchronized道理
在java中,每個對象有且唯一一個同步鎖。這也意味著,同步鎖是依附於對象而存在。
當我們挪用某對象的synchronized辦法時,就獲得了該對象的同步鎖。例如,synchronized(obj)就獲得了“obj這個對象”的同步鎖。
分歧線程對同步鎖的拜訪是互斥的。也就是說,某時光點,對象的同步鎖只能被一個線程獲得到!經由過程同步鎖,我們就可以在多線程中,完成對“對象/辦法”的互斥拜訪。 例如,如今有兩個線程A和線程B,它們都邑拜訪“對象obj的同步鎖”。假定,在某一時辰,線程A獲得到“obj的同步鎖”並在履行一些操作;而此時,線程B也妄圖獲得“obj的同步鎖” —— 線程B會獲得掉敗,它必需期待,直到線程A釋放了“該對象的同步鎖”以後線程B能力獲得到“obj的同步鎖”從而才可以運轉。

2. synchronized根本規矩
我們將synchronized的根本規矩總結為上面3條,並經由過程實例對它們停止解釋。
第一條: 當一個線程拜訪“某對象”的“synchronized辦法”或許“synchronized代碼塊”時,其他線程對“該對象”的該“synchronized辦法”或許“synchronized代碼塊”的拜訪將被壅塞。
第二條: 當一個線程拜訪“某對象”的“synchronized辦法”或許“synchronized代碼塊”時,其他線程依然可以拜訪“該對象”的非同步代碼塊。
第三條: 當一個線程拜訪“某對象”的“synchronized辦法”或許“synchronized代碼塊”時,其他線程對“該對象”的其他的“synchronized辦法”或許“synchronized代碼塊”的拜訪將被壅塞。
(1)第一條:
當一個線程拜訪“某對象”的“synchronized辦法”或許“synchronized代碼塊”時,其他線程對“該對象”的該“synchronized辦法”或許“synchronized代碼塊”的拜訪將被壅塞。 上面是“synchronized代碼塊”對應的演示法式。

class MyRunable implements Runnable {

 @Override
 public void run() {
  synchronized(this) {
   try { 
    for (int i = 0; i < 5; i++) {
     Thread.sleep(100); // 休眠100ms
     System.out.println(Thread.currentThread().getName() + " loop " + i); 
    }
   } catch (InterruptedException ie) { 
   }
  } 
 }
}

public class Demo1_1 {

 public static void main(String[] args) { 
  Runnable demo = new MyRunable();  // 新建“Runnable對象”

  Thread t1 = new Thread(demo, "t1"); // 新建“線程t1”, t1是基於demo這個Runnable對象
  Thread t2 = new Thread(demo, "t2"); // 新建“線程t2”, t2是基於demo這個Runnable對象
  t1.start();       // 啟動“線程t1”
  t2.start();       // 啟動“線程t2” 
 } 
}

運轉成果:

t1 loop 0
t1 loop 1
t1 loop 2
t1 loop 3
t1 loop 4
t2 loop 0
t2 loop 1
t2 loop 2
t2 loop 3
t2 loop 4

成果解釋:run()辦法中存在“synchronized(this)代碼塊”,並且t1和t2都是基於"demo這個Runnable對象"創立的線程。這就意味著,我們可以將synchronized(this)中的this看做是“demo這個Runnable對象”;是以,線程t1和t2同享“demo對象的同步鎖”。所以,當一個線程運轉的時刻,別的一個線程必需期待“運轉線程”釋放“demo的同步鎖”以後能力運轉。
假如你確認,你弄清晰這個成績了。那我們將下面的代碼停止修正,然後再運轉看看成果怎樣樣,看看你能否會含混。修正後的源碼以下:

class MyThread extends Thread {

 public MyThread(String name) {
  super(name);
 }

 @Override
 public void run() {
  synchronized(this) {
   try { 
    for (int i = 0; i < 5; i++) {
     Thread.sleep(100); // 休眠100ms
     System.out.println(Thread.currentThread().getName() + " loop " + i); 
    }
   } catch (InterruptedException ie) { 
   }
  } 
 }
}

public class Demo1_2 {

 public static void main(String[] args) { 
  Thread t1 = new MyThread("t1"); // 新建“線程t1”
  Thread t2 = new MyThread("t2"); // 新建“線程t2”
  t1.start();       // 啟動“線程t1”
  t2.start();       // 啟動“線程t2” 
 } 
}

代碼解釋:比擬Demo1_2 和 Demo1_1,我們發明,Demo1_2中的MyThread類是直接繼續於Thread,並且t1和t2都是MyThread子線程。
榮幸的是,在“Demo1_2的run()辦法”也挪用了synchronized(this),正如“Demo1_1的run()辦法”也挪用了synchronized(this)一樣!
那末,Demo1_2的履行流程是否是和Demo1_1一樣呢?運轉成果:

t1 loop 0
t2 loop 0
t1 loop 1
t2 loop 1
t1 loop 2
t2 loop 2
t1 loop 3
t2 loop 3
t1 loop 4
t2 loop 4

成果解釋:
假如這個成果一點也不令你覺得驚奇,那末我信任你對synchronized和this的熟悉曾經比擬深入了。不然的話,請持續浏覽這裡的剖析。
synchronized(this)中的this是指“以後的類對象”,即synchronized(this)地點的類對應確當前對象。它的感化是獲得“以後對象的同步鎖”。
關於Demo1_2中,synchronized(this)中的this代表的是MyThread對象,而t1和t2是兩個分歧的MyThread對象,是以t1和t2在履行synchronized(this)時,獲得的是分歧對象的同步鎖。關於Demo1_1對而言,synchronized(this)中的this代表的是MyRunable對象;t1和t2配合一個MyRunable對象,是以,一個線程獲得了對象的同步鎖,會形成別的一個線程期待。
(2)第二條:
當一個線程拜訪“某對象”的“synchronized辦法”或許“synchronized代碼塊”時,其他線程依然可以拜訪“該對象”的非同步代碼塊。
上面是“synchronized代碼塊”對應的演示法式。

class Count {

 // 含有synchronized同步塊的辦法
 public void synMethod() {
  synchronized(this) {
   try { 
    for (int i = 0; i < 5; i++) {
     Thread.sleep(100); // 休眠100ms
     System.out.println(Thread.currentThread().getName() + " synMethod loop " + i); 
    }
   } catch (InterruptedException ie) { 
   }
  } 
 }

 // 非同步的辦法
 public void nonSynMethod() {
  try { 
   for (int i = 0; i < 5; i++) {
    Thread.sleep(100);
    System.out.println(Thread.currentThread().getName() + " nonSynMethod loop " + i); 
   }
  } catch (InterruptedException ie) { 
  }
 }
}

public class Demo2 {

 public static void main(String[] args) { 
  final Count count = new Count();
  // 新建t1, t1會挪用“count對象”的synMethod()辦法
  Thread t1 = new Thread(
    new Runnable() {
     @Override
     public void run() {
      count.synMethod();
     }
    }, "t1");

  // 新建t2, t2會挪用“count對象”的nonSynMethod()辦法
  Thread t2 = new Thread(
    new Runnable() {
     @Override
     public void run() {
      count.nonSynMethod();
     }
    }, "t2"); 


  t1.start(); // 啟動t1
  t2.start(); // 啟動t2
 } 
}

運轉成果:

t1 synMethod loop 0
t2 nonSynMethod loop 0
t1 synMethod loop 1
t2 nonSynMethod loop 1
t1 synMethod loop 2
t2 nonSynMethod loop 2
t1 synMethod loop 3
t2 nonSynMethod loop 3
t1 synMethod loop 4
t2 nonSynMethod loop 4

成果解釋:
主線程中新建了兩個子線程t1和t2。t1會挪用count對象的synMethod()辦法,該辦法內含有同步塊;而t2則會挪用count對象的nonSynMethod()辦法,該辦法不是同步辦法。t1運轉時,固然挪用synchronized(this)獲得“count的同步鎖”;然則並沒有形成t2的壅塞,由於t2沒有效到“count”同步鎖。
(3)第三條:
當一個線程拜訪“某對象”的“synchronized辦法”或許“synchronized代碼塊”時,其他線程對“該對象”的其他的“synchronized辦法”或許“synchronized代碼塊”的拜訪將被壅塞。
我們將下面的例子中的nonSynMethod()辦法體的也用synchronized(this)潤飾。修正後的源碼以下:

class Count {

 // 含有synchronized同步塊的辦法
 public void synMethod() {
  synchronized(this) {
   try { 
    for (int i = 0; i < 5; i++) {
     Thread.sleep(100); // 休眠100ms
     System.out.println(Thread.currentThread().getName() + " synMethod loop " + i); 
    }
   } catch (InterruptedException ie) { 
   }
  } 
 }

 // 也包括synchronized同步塊的辦法
 public void nonSynMethod() {
  synchronized(this) {
   try { 
    for (int i = 0; i < 5; i++) {
     Thread.sleep(100);
     System.out.println(Thread.currentThread().getName() + " nonSynMethod loop " + i); 
    }
   } catch (InterruptedException ie) { 
   }
  }
 }
}

public class Demo3 {

 public static void main(String[] args) { 
  final Count count = new Count();
  // 新建t1, t1會挪用“count對象”的synMethod()辦法
  Thread t1 = new Thread(
    new Runnable() {
     @Override
     public void run() {
      count.synMethod();
     }
    }, "t1");

  // 新建t2, t2會挪用“count對象”的nonSynMethod()辦法
  Thread t2 = new Thread(
    new Runnable() {
     @Override
     public void run() {
      count.nonSynMethod();
     }
    }, "t2"); 


  t1.start(); // 啟動t1
  t2.start(); // 啟動t2
 } 
}

運轉成果:

t1 synMethod loop 0
t1 synMethod loop 1
t1 synMethod loop 2
t1 synMethod loop 3
t1 synMethod loop 4
t2 nonSynMethod loop 0
t2 nonSynMethod loop 1
t2 nonSynMethod loop 2
t2 nonSynMethod loop 3
t2 nonSynMethod loop 4

成果解釋:
主線程中新建了兩個子線程t1和t2。t1和t2運轉時都挪用synchronized(this),這個this是Count對象(count),而t1和t2共用count。是以,在t1運轉時,t2會被壅塞,期待t1運轉釋放“count對象的同步鎖”,t2能力運轉。

3. synchronized辦法 和 synchronized代碼塊
“synchronized辦法”是用synchronized潤飾辦法,而 “synchronized代碼塊”則是用synchronized潤飾代碼塊。
synchronized辦法示例

public synchronized void foo1() {
  System.out.println("synchronized methoed");
}
synchronized代碼塊
public void foo2() {
  synchronized (this) {
    System.out.println("synchronized methoed");
  }
}

synchronized代碼塊中的this是指以後對象。也能夠將this調換成其他對象,例如將this調換成obj,則foo2()在履行synchronized(obj)時就獲得的是obj的同步鎖。
synchronized代碼塊可以更准確的掌握抵觸限制拜訪區域,有時刻表示更高效力。上面經由過程一個示例來演示:

// Demo4.java的源碼
public class Demo4 {

  public synchronized void synMethod() {
    for(int i=0; i<1000000; i++)
      ;
  }

  public void synBlock() {
    synchronized( this ) {
      for(int i=0; i<1000000; i++)
        ;
    }
  }

  public static void main(String[] args) {
    Demo4 demo = new Demo4();

    long start, diff;
    start = System.currentTimeMillis();        // 獲得以後時光(millis)
    demo.synMethod();                // 挪用“synchronized辦法”
    diff = System.currentTimeMillis() - start;    // 獲得“時光差值”
    System.out.println("synMethod() : "+ diff);

    start = System.currentTimeMillis();        // 獲得以後時光(millis)
    demo.synBlock();                // 挪用“synchronized辦法塊”
    diff = System.currentTimeMillis() - start;    // 獲得“時光差值”
    System.out.println("synBlock() : "+ diff);
  }
}

(某一次)履行成果:

synMethod() : 11
synBlock() : 3

4. 實例鎖 和 全局鎖
實例鎖 -- 鎖在某一個實例對象上。假如該類是單例,那末該鎖也具有全局鎖的概念。
(1)實例鎖對應的就是synchronized症結字。
(2)全局鎖 -- 該鎖針對的是類,不管實例若干個對象,那末線程都同享該鎖。
全局鎖對應的就是static synchronized(或許是鎖在該類的class或許classloader對象上)。
關於“實例鎖”和“全局鎖”有一個很抽象的例子:

pulbic class Something {
  public synchronized void isSyncA(){}
  public synchronized void isSyncB(){}
  public static synchronized void cSyncA(){}
  public static synchronized void cSyncB(){}
}

假定,Something有兩個實例x和y。剖析上面4組表達式獲得的鎖的情形。
(1) x.isSyncA()與x.isSyncB()
(2) x.isSyncA()與y.isSyncA()
(3) x.cSyncA()與y.cSyncB()
(4) x.isSyncA()與Something.cSyncA()

(1) 不克不及被同時拜訪。
由於isSyncA()和isSyncB()都是拜訪統一個對象(對象x)的同步鎖!

// LockTest1.java的源碼
class Something {
  public synchronized void isSyncA(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : isSyncA");
      }
    }catch (InterruptedException ie) { 
    } 
  }
  public synchronized void isSyncB(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : isSyncB");
      }
    }catch (InterruptedException ie) { 
    } 
  }
}

public class LockTest1 {

  Something x = new Something();
  Something y = new Something();

  // 比擬(01) x.isSyncA()與x.isSyncB() 
  private void test1() {
    // 新建t11, t11會挪用 x.isSyncA()
    Thread t11 = new Thread(
        new Runnable() {
          @Override
          public void run() {
            x.isSyncA();
          }
        }, "t11");

    // 新建t12, t12會挪用 x.isSyncB()
    Thread t12 = new Thread(
        new Runnable() {
          @Override
          public void run() {
            x.isSyncB();
          }
        }, "t12"); 


    t11.start(); // 啟動t11
    t12.start(); // 啟動t12
  }

  public static void main(String[] args) {
    LockTest1 demo = new LockTest1();
    demo.test1();
  }
}

運轉成果:

t11 : isSyncA
t11 : isSyncA
t11 : isSyncA
t11 : isSyncA
t11 : isSyncA
t12 : isSyncB
t12 : isSyncB
t12 : isSyncB
t12 : isSyncB
t12 : isSyncB

(2) 可以同時被拜訪

由於拜訪的不是統一個對象的同步鎖,x.isSyncA()拜訪的是x的同步鎖,而y.isSyncA()拜訪的是y的同步鎖。

// LockTest2.java的源碼
class Something {
  public synchronized void isSyncA(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : isSyncA");
      }
    }catch (InterruptedException ie) { 
    } 
  }
  public synchronized void isSyncB(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : isSyncB");
      }
    }catch (InterruptedException ie) { 
    } 
  }
  public static synchronized void cSyncA(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : cSyncA");
      } 
    }catch (InterruptedException ie) { 
    } 
  }
  public static synchronized void cSyncB(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : cSyncB");
      } 
    }catch (InterruptedException ie) { 
    } 
  }
}

public class LockTest2 {

  Something x = new Something();
  Something y = new Something();

  // 比擬(02) x.isSyncA()與y.isSyncA()
  private void test2() {
    // 新建t21, t21會挪用 x.isSyncA()
    Thread t21 = new Thread(
        new Runnable() {
          @Override
          public void run() {
            x.isSyncA();
          }
        }, "t21");

    // 新建t22, t22會挪用 x.isSyncB()
    Thread t22 = new Thread(
        new Runnable() {
          @Override
          public void run() {
            y.isSyncA();
          }
        }, "t22"); 


    t21.start(); // 啟動t21
    t22.start(); // 啟動t22
  }

  public static void main(String[] args) {
    LockTest2 demo = new LockTest2();

    demo.test2();
  }
}

運轉成果:

t21 : isSyncA
t22 : isSyncA
t21 : isSyncA
t22 : isSyncA
t21 : isSyncA
t22 : isSyncA
t21 : isSyncA
t22 : isSyncA
t21 : isSyncA
t22 : isSyncA

(3) 不克不及被同時拜訪
由於cSyncA()和cSyncB()都是static類型,x.cSyncA()相當於Something.isSyncA(),y.cSyncB()相當於Something.isSyncB(),是以它們共用一個同步鎖,不克不及被同時反問。

// LockTest3.java的源碼
class Something {
  public synchronized void isSyncA(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : isSyncA");
      }
    }catch (InterruptedException ie) { 
    } 
  }
  public synchronized void isSyncB(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : isSyncB");
      }
    }catch (InterruptedException ie) { 
    } 
  }
  public static synchronized void cSyncA(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : cSyncA");
      } 
    }catch (InterruptedException ie) { 
    } 
  }
  public static synchronized void cSyncB(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : cSyncB");
      } 
    }catch (InterruptedException ie) { 
    } 
  }
}

public class LockTest3 {

  Something x = new Something();
  Something y = new Something();

  // 比擬(03) x.cSyncA()與y.cSyncB()
  private void test3() {
    // 新建t31, t31會挪用 x.isSyncA()
    Thread t31 = new Thread(
        new Runnable() {
          @Override
          public void run() {
            x.cSyncA();
          }
        }, "t31");

    // 新建t32, t32會挪用 x.isSyncB()
    Thread t32 = new Thread(
        new Runnable() {
          @Override
          public void run() {
            y.cSyncB();
          }
        }, "t32"); 


    t31.start(); // 啟動t31
    t32.start(); // 啟動t32
  }

  public static void main(String[] args) {
    LockTest3 demo = new LockTest3();

    demo.test3();
  }
}

運轉成果:

t31 : cSyncA
t31 : cSyncA
t31 : cSyncA
t31 : cSyncA
t31 : cSyncA
t32 : cSyncB
t32 : cSyncB
t32 : cSyncB
t32 : cSyncB
t32 : cSyncB

(4) 可以被同時拜訪
由於isSyncA()是實例辦法,x.isSyncA()應用的是對象x的鎖;而cSyncA()是靜態辦法,Something.cSyncA()可以懂得對應用的是“類的鎖”。是以,它們是可以被同時拜訪的。

// LockTest4.java的源碼
class Something {
  public synchronized void isSyncA(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : isSyncA");
      }
    }catch (InterruptedException ie) { 
    } 
  }
  public synchronized void isSyncB(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : isSyncB");
      }
    }catch (InterruptedException ie) { 
    } 
  }
  public static synchronized void cSyncA(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : cSyncA");
      } 
    }catch (InterruptedException ie) { 
    } 
  }
  public static synchronized void cSyncB(){
    try { 
      for (int i = 0; i < 5; i++) {
        Thread.sleep(100); // 休眠100ms
        System.out.println(Thread.currentThread().getName()+" : cSyncB");
      } 
    }catch (InterruptedException ie) { 
    } 
  }
}

public class LockTest4 {

  Something x = new Something();
  Something y = new Something();

  // 比擬(04) x.isSyncA()與Something.cSyncA()
  private void test4() {
    // 新建t41, t41會挪用 x.isSyncA()
    Thread t41 = new Thread(
        new Runnable() {
          @Override
          public void run() {
            x.isSyncA();
          }
        }, "t41");

    // 新建t42, t42會挪用 x.isSyncB()
    Thread t42 = new Thread(
        new Runnable() {
          @Override
          public void run() {
            Something.cSyncA();
          }
        }, "t42"); 


    t41.start(); // 啟動t41
    t42.start(); // 啟動t42
  }

  public static void main(String[] args) {
    LockTest4 demo = new LockTest4();

    demo.test4();
  }
}

運轉成果:

t41 : isSyncA
t42 : cSyncA
t41 : isSyncA
t42 : cSyncA
t41 : isSyncA
t42 : cSyncA
t41 : isSyncA
t42 : cSyncA
t41 : isSyncA
t42 : cSyncA

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved