自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Java多線程編程基礎之線程對象

開發(fā) 后端
多線程是這樣一種機制,它允許在程序中并發(fā)執(zhí)行多個指令流,每個指令流都稱為一個線程,彼此間互相獨立。本文主要介紹JAVA多線程的一些基本知識,一起來看。

在進入java平臺的線程對象之前,基于基礎篇(一)的一些問題,我先插入兩個基本概念。

線程的并發(fā)與并行

在單CPU系統(tǒng)中,系統(tǒng)調度在某一時刻只能讓一個線程運行,雖然這種調試機制有多種形式(大多數(shù)是時間片輪巡為主),但無論如何,要通過不斷切換需要運行的線程讓其運行的方式就叫并發(fā)(concurrent)。而在多CPU系統(tǒng)中,可以讓兩個以上的線程同時運行,這種可以同時讓兩個以上線程同時運行的方式叫做并行(parallel)。

在上面包括以后的所有論述中,請各位朋友諒解,我無法用最準確的詞語來定義儲如并發(fā)和并行這類術語,但我以我的經(jīng)驗能通俗地告訴大家它是怎么一回事,如果您看到我說的一些"標準"文檔上說的不一樣,只要意思一致,那您就不要挑刺了。

JAVA線程對象

現(xiàn)在我們來開始考察JAVA中線程對象。

在JAVA中,要開始一個線程,有兩種方式。一是直接調用Thread實例的start()方法,二是

將Runable實例傳給一個Thread實例然后調用它的start()方法。

在前面已經(jīng)說過,線程對象和線程是兩個完全不同的概念。這里我們再次深入一下,生成一個線程的實例,并不代表啟動了線程。而啟動線程是說在某個線程對象上啟動了該實例對應的線程,當該線程結束后,并不會就立即消失。

對于從很多書籍上可以看到的基礎知識我就不用多說了。既然是基礎知識,我也著重于從普通文檔上讀不到的內容。所以本節(jié)我重點要說的是兩種線程對象產(chǎn)生線程方式的區(qū)別。

 

  1. class MyThread extends Thread{  
  2.  public int x = 0;  
  3.  public void run(){  
  4. for(int i=0;i<100;i++){  
  5.   try{  
  6. Thread.sleep(10);  
  7.   }catch(Exception e){}  
  8.   System.out.println(x++);  
  9. }  
  10.  }  

 

如果我們生成MyThread的一個實例,然后調用它的start()方法,那么就產(chǎn)生了這個實例對應的線程:

 

  1. public class Test {  
  2.  public static void main(String[] args) throws Exception{  
  3. MyThread mt = new MyThread();  
  4. mt.start();  
  5.  }  

 

不用說,最終會打印出0到99,現(xiàn)在我們稍微玩一點花樣:

 

  1. public class Test {  
  2.  public static void main(String[] args) throws Exception{  
  3. MyThread mt = new MyThread();  
  4. mt.start();  
  5. System.out.println(101);  
  6.  }  

 

也不用說,在基礎篇(一)中我們知道由于單CPU的原因,一般會先打印101,然后打印0到99。不過我們可以控制線程讓它按我們的意思來運行:

 

  1. public class Test {  
  2.  public static void main(String[] args) throws Exception{  
  3. MyThread mt = new MyThread();  
  4. mt.start();  
  5. mt.join();  
  6. System.out.println(101);  
  7.  }  

 

好了,我們終于看到,mt實例對應的線程(假如我有時說mt線程請你不要怪我,不過我盡量不這么說)。在運行完成后,主線程才打印101。因為我們讓當前線程(這里是主線程)等待mt線程的運行結束。"在線程對象a上調用join()方法,就是讓當前正在執(zhí)行的線程等待線程對象a對應的線程運行完成后才繼續(xù)運行。" 請大家一定要深刻理解并熟記這句話,而我這里引出這個知識點的目的是為了讓你繼續(xù)看下面的例子:

 

  1. public class Test {  
  2.  public static void main(String[] args) throws Exception{  
  3. MyThread mt = new MyThread();  
  4. mt.start();  
  5. mt.join();  
  6. Thread.sleep(3000);  
  7. mt.start();  
  8.  }  

 

當線程對象mt運行完成后,我們讓主線程休息一下,然后我們再次在這個線程對象上啟動線程。結果我們看到:

 

  1. Exception in thread "main" java.lang.IllegalThreadStateException 

 

也就是這種線程對象一時運行一次完成后,它就再也不能運行第二次了。我們可以看一下它有具體實現(xiàn):

 

  1. public synchronized void start() {  
  2.  if (started)  
  3. throw new IllegalThreadStateException();  
  4. started = true;  
  5. group.add(this);  
  6. start0();  
  7.  } 

 

一個Thread的實例一旦調用start()方法,這個實例的started標記就標記為true,事實中不管這個線程后來有沒有執(zhí)行到底,只要調用了一次start()就再也沒有機會運行了,這意味著:

通過Thread實例的start(),一個Thread的實例只能產(chǎn)生一個線程

那么如果要在一個實例上產(chǎn)生多個線程(也就是我們常說的線程池),我們應該如何做呢?這就是Runnable接口給我們帶來的偉大的功能。

 

  1. class R implements Runnable{  
  2.  private int x = 0;  
  3.  public void run(){  
  4. for(int i=0;i<100;i++){  
  5.   try{  
  6. Thread.sleep(10);  
  7.   }catch(Exception e){}  
  8.   System.out.println(x++);  
  9. }  
  10.  }  

 

正如它的名字一樣,Runnable的實例是可運行的,但它自己并不能直接運行,它需要被Thread對象來包裝才行運行:

 

  1. public class Test {  
  2.  public static void main(String[] args) throws Exception{  
  3. new Thread(new R()).start();  
  4.  }  

 

當然這個結果和mt.start()沒有什么區(qū)別。但如果我們把一個Runnable實例給Thread對象多次包裝,我們就可以看到它們實際是在同一實例上啟動線程:

 

  1. public class Test {  
  2.  public static void main(String[] args) throws Exception{  
  3. R r = new R();  
  4. for(int i=0;i<10;i++)  
  5.   new Thread(r).start();  
  6.  }  

 

x是實例對象,但結果是x被加到了999,說明這10個線程是在同一個r對象上運行的。請大家注意,因為這個例子是在單CPU上運行的,所以沒有對多個線程同時操作共同的對象進行同步。這里是為了說明的方便而簡化了同步,而真正的環(huán)境中你無法預知程序會在什么環(huán)境下運行,所以一定要考慮同步。

到這里我們做一個完整的例子來說明線程產(chǎn)生的方式不同而生成的線程的區(qū)別:

 

  1. package debug;  
  2. import java.io.*;  
  3. import java.lang.Thread;  
  4. class MyThread extends Thread{  
  5.  public int x = 0;  
  6.  public void run(){  
  7. System.out.println(++x);  
  8.  }  
  9. }  
  10. class R implements Runnable{  
  11.  private int x = 0;  
  12.  public void run(){  
  13. System.out.println(++x);  
  14.  }  
  15. }  
  16. public class Test {  
  17.  public static void main(String[] args) throws Exception{  
  18. for(int i=0;i<10;i++){  
  19.   Thread t = new MyThread();  
  20.   t.start();  
  21. }  
  22. Thread.sleep(10000);//讓上面的線程運行完成  
  23. R r = new R();  
  24. for(int i=0;i<10;i++){  
  25.   Thread t = new Thread(r);  
  26.   t.start();  
  27. }  
  28.  }  

 

上面10個線程對象產(chǎn)生的10個線程運行時打印了10次1。下面10個線程對象產(chǎn)生的10個線程運行時打印了1到10。我們把下面的10個線程稱為同一實例(Runnable實例)的多個線程。

【編輯推薦】

  1. Java多線程程序設計詳細解析
  2. Java多線程遞歸如何彌補管理漏洞
  3. Java多線程程序四步輕松學成
  4. Java多線程求和詳細學習筆記
  5. Java多線程進程應對同一程序運行資源
責任編輯:于鐵 來源: ITWAKA.COM
相關推薦

2009-03-12 10:52:43

Java線程多線程

2023-06-07 13:49:00

多線程編程C#

2013-07-16 10:12:14

iOS多線程多線程概念多線程入門

2023-06-08 08:21:08

多線程編程線程間通信

2023-06-09 07:59:37

多線程編程鎖機制

2023-06-13 13:39:00

多線程異步編程

2009-02-24 08:36:51

多線程線程池網(wǎng)絡服務器

2009-07-29 16:42:35

Java多線程編程

2011-07-22 14:55:20

多線程

2017-03-08 14:18:37

Linux多線程編程

2018-10-25 15:55:44

Java多線程鎖優(yōu)化

2021-03-01 11:20:13

網(wǎng)絡安全多線程代碼

2013-05-23 15:59:00

線程池

2022-05-26 08:31:41

線程Java線程與進程

2023-06-05 07:56:10

線程分配處理器

2023-06-06 08:17:52

多線程編程Thread類

2023-05-12 14:14:00

Java線程中斷

2013-07-16 10:57:34

iOS多線程多線程概念多線程入門

2013-07-16 12:13:27

iOS多線程多線程概念GCD

2023-04-02 17:53:10

多線程編程自測
點贊
收藏

51CTO技術棧公眾號