多線程【Thread、線程創建】

来源:http://www.cnblogs.com/caigq/archive/2017/06/23/7049119.html
-Advertisement-
Play Games

主線程:執行主方法的線程,就叫做主線程 單線程程式:程式從mani開始從上到下依次運行 程式從main方法開始運行,JVM運行main方法,會找操作系統 開闢一條通向cpu的執行路徑,cpu可以通過這條路徑來執行main方法 這條路徑有一個名字叫主(main)線程 創建線程方式一繼承Thread類 ...


主線程:執行主方法的線程,就叫做主線程

單線程程式:程式從mani開始從上到下依次運行

程式從main方法開始運行,JVM運行main方法,會找操作系統
開闢一條通向cpu的執行路徑,cpu可以通過這條路徑來執行main方法
這條路徑有一個名字叫主(main)線程

創建線程方式一繼承Thread類
實現步驟:
1.創建Thread類的子類
2.重寫Thread類中的run方法,設置線程的任務
3.創建Thread類的子類對象
4.調用Thread類中的start方法開啟一個新的線程,執行run方法
使該線程開始執行;Java 虛擬機調用該線程的 run 方法。
結果是兩個線程併發地運行;當前線程(main線程)和另一個線程(執行 run 方法的線程)。
多次啟動一個線程是非法的。特別是當線程已經結束執行後,不能再重新啟動。

列印的結果出現了隨機性:
開啟兩個線程,對於cpu就選擇權利
喜歡誰,就執行誰,所以就出現了隨機性結果

 1 public class MyThread extends Thread{
 2     /*
 3      * 2.重寫Thread類中的run方法,設置線程的任務
 4      * 開啟這個線程要乾什麼事情
 5      */
 6     @Override
 7     public void run() {
 8         for (int i = 0; i < 50; i++) {
 9             System.out.println("run..."+i);
10         }
11     }
12 }
13     public static void main(String[] args) {
14         //3.創建Thread類的子類對象
15         MyThread mt = new MyThread();
16         //mt.run();//不會開啟線程,還是單線程程式
17         //4.調用Thread類中的start方法開啟一個新的線程,執行run方法
18         mt.start();
19         
20         new MyThread().start();
21         
22         for (int i = 0; i < 50; i++) {
23             System.out.println("main..."+i);
24         }
25     }

線程的名稱:
主線程:"main"
開啟的其它線程的名稱:"Thread-0","Thread-1"....

獲取線程的名稱
1.Thread類中的方法getName
String getName() 返回該線程的名稱。
2.Thread類中的靜態方法,獲取當前正在執行的線程
static Thread currentThread() 返回對當前正在執行的線程對象的引用。
設置線程的名稱:
1.Thread類中的方法setName(String name)
void setName(String name) 改變線程名稱,使之與參數 name 相同。
2.子類添加帶參構造,調用父類Thread類的帶參構造方法,傳遞線程的名稱,讓父類給線程起名字(讓父親給兒子起名字)
Thread(String name) 分配新的 Thread 對象。


創建線程方式—實現Runnable介面

實現步驟:
1.創建Runnable介面的實現類
2.重寫Runnable介面中的run方法,設置線程任務
3.創建Runnable介面的實現類對象
4.創建Thread類對象,構造方法中傳入Runnable介面的實現類
Thread(Runnable target) 分配新的 Thread 對象。
5.調用Thread類中的方法start,開啟線程執行run方法


實現Runnable的好處
1.避免了類繼承Thread類之後,無法繼承其它的類(單繼承的局限性)
2.把設置線程任務,和開啟線程進行解耦,增強了擴展性
實現類的作用:就是設置線程任務
Thread類的作用:開啟線程
好處:傳遞不同的實現類,實現類重寫的方法不一樣,可以調用不同的方法

線程的匿名內部類使用

匿名:沒有名字
內部類:寫在其他類內部的類(成員位置:成員內部類,局部位置(方法中):局部內部類)

匿名內部類的格式:
new 父類/介面(){
重寫父類/介面中的方法;
};

多線程的父類:
Thread
Runnable

1  new Thread(){
2             //重寫run方法,設置線程任務
3             @Override
4             public void run() {
5                 for (int i = 0; i < 20; i++) {
6                     System.out.println(Thread.currentThread().getName()+":"+i);
7                 }
8             }
9         }

以上一堆代碼就是一個創建子類重寫父類方法的過程
相當於: new MyThread().start();

程式出現了線程安全問題:賣了重覆的票和不存在的票

解決方案:
第一種方式:可以使用同步代碼塊

synchronized(鎖對象){
產生安全問題的代碼;
訪問了共用數據的代碼;
}

註意:
必須要保證多個線程使用的是同一個鎖對象
//在成員位置上創建一個鎖對象(保證唯一)

 1 Object obj = new Object();
 2     
 3     @Override
 4     public void run() {
 5         //讓賣票重覆執行
 6         while(true){
 7         
 8              * 同步代碼塊
 9              * 程式會頻繁的判斷鎖,獲取鎖,釋放鎖,所以會降低速度
10              
11             synchronized (obj) {
12                 if(ticket>0){
13                     //為了提高安全問題的概率,讓程式睡眠
14                     try {
15                         Thread.sleep(10);
16                     } catch (InterruptedException e) {
17                         e.printStackTrace();
18                     }
19                     //賣票ticket--
20                     System.out.println(Thread.currentThread().getName()+"...賣第"+ticket--+"張票");
21                 }
22             }
23         }
24     }

程式出現了線程安全問題:賣了重覆的票和不存在的票

解決方案:
第二種方式:同步方法

使用步驟:
1.把可能出現安全問題的代碼抽取到一個方法中
2.把方法增加一個關鍵字synchronized
修飾符 synchronized 返回值類型 方法名(參數){
可能出現安全問題的代碼;
訪問了共用數據的代碼;
}

同步方法使用的鎖對象是什麼?
使用的就是本類對象new RunnableImpl()-->叫this

靜態的同步方法,使用時什麼鎖對象?
使用的是本類對象的class屬性(反射)
RunnableImpl.class

 1 *@Override
 2     public void run() {
 3         //讓賣票重覆執行
 4         while(true){
 5             payTicket2();
 6         }
 7     }
 8     
 9     
10      * 靜態的同步方法
11      
12     public static synchronized void payTicket2(){
13         synchronized (RunnableImpl.class) {
14             if(ticket>0){
15                 //為了提高安全問題的概率,讓程式睡眠
16                 try {
17                     Thread.sleep(10);
18                 } catch (InterruptedException e) {
19                     e.printStackTrace();
20                 }
21                 //賣票ticket--
22                 System.out.println(Thread.currentThread().getName()+"...賣第"+ticket--+"張票");
23             }
24         }
25     }
26     
27     
28     
29      * 抽取出一個同步方法
30      * 快捷鍵:alt+shift+m
31      
32     public ynchronized void payTicket1() {
33         synchronized (this) {
34             //System.out.println(this);//cn.itcsat.demo10.RunnableImpl@67064
35             if(ticket>0){
36                 //為了提高安全問題的概率,讓程式睡眠
37                 try {
38                     Thread.sleep(10);
39                 } catch (InterruptedException e) {
40                     e.printStackTrace();
41                 }
42                 //賣票ticket--
43                 System.out.println(Thread.currentThread().getName()+"...賣第"+ticket--+"張票");
44             }
45         }
46     }

程式出現了線程安全問題:賣了重覆的票和不存在的票
*
* 解決方案:
* 第三種方式:使用Lock介面,JDK1.5之後出現的
*
* java.util.concurrent.locks.Lock介面
* 方法:
* void lock() 獲取鎖。
* void unlock() 釋放鎖。
* 介面的實現類:ReentrantLock
*
* 實現步驟:
* 1.在成員位置創建一個Lock介面的實現類對象ReentrantLock
* 2.在可能出現線程安全問題的代碼前,調用lock方法獲取鎖
* 3.在可能出現線程安全問題的代碼後,調用unlock方法釋放鎖
*
*1.在成員位置創建一個Lock介面的實現類對象ReentrantLock
Lock l = new ReentrantLock();

 1 @Override
 2     public void run() {
 3         //讓賣票重覆執行
 4         while(true){
 5             //2.在可能出現線程安全問題的代碼前,調用lock方法獲取鎖
 6             l.lock();
 7                 if(ticket>0){
 8                     //為了提高安全問題的概率,讓程式睡眠
 9                     try {
10                         Thread.sleep(10);
11                         //賣票ticket--
12                         System.out.println(Thread.currentThread().getName()+"...賣第"+ticket--+"張票");
13                     } catch (InterruptedException e) {
14                         e.printStackTrace();
15                     }finally {
16                         //3.在可能出現線程安全問題的代碼後,調用unlock方法釋放鎖
17                         l.unlock();    
18                     }
19                 }
20         }

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 學習單片機過程中遇到兩個問題: 1、寫好串口代碼,手邊沒有硬體調試; 2、串口收發出現奇怪的問題,無法定位; 經過百度,百度,再百度後,終於找到瞭解決方法,那就是:串口收發模擬。不多說,總結如下: 實驗需要的軟體:VSPD(Virtual Serial Port Driver)、串口調試助手、Kei ...
  • 1.看似針對同樣一段查詢表ef達式,重覆執行卻沒有被記錄下來。其實這是正常情況,因為ef並沒有重覆去執行 相同sql查詢。 2.MiniProfiler結合MVC過濾器進行 攔截記錄Sql,示例代碼: 3.以上的miniprofiler並不能 攔截到 sql語句查詢,需要使用 minprofiler ...
  • 一、WCF簡介 1、什麼是WCF? WCF的全稱是:Windows Communication Foundation。從本質上來說,它是一套軟體開發包,是微軟公司推出的符合SOA思想的技術框架。 2、WCF中包括哪些東西? WCF為程式員提供了豐富的功能,其中包括:托管、服務實例管理、非同步、安全、事 ...
  • <div style="float: left;border-radius:70%; height: 80px; overflow:hidden;"> <img class="mui-icon-image" src="../../img/002 劉運忠.jpg" style="height: 80p ...
  • <div id="segmented" class="mui-segmented-control"> <a class="mui-control-item mui-active" href="#item1mobile"> 考勤記錄 </a> <a class="mui-control-item" h ...
  • //論文要用到其改進演算法,在此先demo測試一下using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; namespace ... ...
  • 閉包(closure)是函數式編程的重要的語法結構。閉包也是一種組織代碼的結構,它同樣提高了代碼的可重覆使用性。 如果在一個內嵌函數里,對在外部函數內(但不是在全局作用域)的變數進行引用,那麼內嵌函數就被認為是閉包(closure)。 定義在外部函數內但由內部函數引用或者使用的變數稱為自由變數。 總 ...
  • 用python3.4讀取Excel中的數據並繪圖,一個簡單的實現。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...