在早期的操作系統(tǒng)中,執(zhí)行任務(wù)被抽象為進(jìn)程(Process)。其中,進(jìn)程是操作系統(tǒng)運(yùn)行和調(diào)度的基本單元。
隨著計算機(jī)技術(shù)的不斷發(fā)展,由于進(jìn)程開銷資源較大,以進(jìn)程為調(diào)度單位的方式逐漸產(chǎn)生弊端。因此,計算機(jī)先進(jìn)工作者(科學(xué)家)們在進(jìn)程的基礎(chǔ)上,提出了線程(Thead)的概念。
線程是進(jìn)程中的運(yùn)行單位,可以把線程看作輕量級的進(jìn)程。計算機(jī)CPU會按照某種策略為每一個線程分配一定的時間片去執(zhí)行。
進(jìn)程是指程序的一次動態(tài)執(zhí)行過程,計算機(jī)中正在執(zhí)行的程序就是進(jìn)程,每一個程序都對對應(yīng)著各自的一個進(jìn)程。
一個進(jìn)程包含了從代碼加載完畢到執(zhí)行完成的一個完成過程,是操作系統(tǒng)中資源分配的最小單位。
線程是比進(jìn)程更小的執(zhí)行單元,是計算機(jī)CPU調(diào)度和分配的基本單位。
每一個進(jìn)程都會至少包含一個線程,而一個線程只屬于一個進(jìn)程。
每一個進(jìn)程都有自己的資源,一個進(jìn)程內(nèi)的所有線程都共享這個進(jìn)程所包含的資源。
每一個線程可以對所屬進(jìn)程的所有資源進(jìn)行調(diào)度和運(yùn)算,其中,線程可以是操作系統(tǒng)內(nèi)核來控制調(diào)度,也可以是由用戶程序來控制調(diào)度。
基本定義
現(xiàn)代計算機(jī),從組成部分上來看,大體可以分為硬件和軟件兩個部分。硬件是基礎(chǔ),而軟件是運(yùn)行在硬件之上的程序。
其中,軟件可以分為操作系統(tǒng)和應(yīng)用程序:
操作系統(tǒng)(Operation System):專注于對硬件的支持和交互管理并提供一個運(yùn)行環(huán)境給應(yīng)用程序使用
應(yīng)用程序(Application Program):能實(shí)現(xiàn)若干功能且運(yùn)行在操作系統(tǒng)中的軟件
由于線程可以由操作系統(tǒng)內(nèi)核和用戶程序來控制調(diào)度,因此按照操作系統(tǒng)和應(yīng)用程序兩個層次來分類。
線程可以主要分為內(nèi)核線程和 用戶線程(應(yīng)用線程)兩類,其中:
內(nèi)核線程(Kernel Thread):由操作系統(tǒng)內(nèi)核支持和管理的線程,內(nèi)核線程的創(chuàng)建,啟動,同步,銷毀,切換等均由操作系統(tǒng)完成。
用戶(應(yīng)用線程,Applciation Thread)線程(User Thread) :用戶(應(yīng)用)線程的管理工作在用戶(應(yīng)用)空間完成,它完全建立在用戶(應(yīng)用)空間的線程庫上,由內(nèi)核支持但不由內(nèi)核管理,內(nèi)核也無法感知用戶線程的存在。用戶(應(yīng)用)線程的創(chuàng)建,啟動,同步,銷毀,切換等均在在用戶(應(yīng)用)空間完成,不用切換到內(nèi)核。
從Java領(lǐng)域來看,Java語言編譯后的字節(jié)碼(Byte Code) 運(yùn)行在JVM (Java 虛擬機(jī))上,其中JVM其實(shí)是一個進(jìn)程,所以Java屬于應(yīng)用程序?qū)印?/p>
我們都知道,Java的線程類為:java.lang.Thread,當(dāng)任務(wù)不能在當(dāng)前線程中執(zhí)行時,我們就會去創(chuàng)建一個Thread對象。
我們在Java層通過new 關(guān)鍵字創(chuàng)建一個Thread對象,然后調(diào)用start()方法啟動該線程,那么從線程的角度來看,主要可以分為:
Java應(yīng)用程序?qū)泳€程(Java Application Thread ):主要是Java語言編程的程序創(chuàng)建的Thread線程對象,屬于用戶空間
Java虛擬機(jī)層線程(Java JVM Thread ):主要是Java虛擬機(jī)中包含且支持和管理的線程,屬于用戶空間,
操作系統(tǒng)層線程(OS Thread):根據(jù)操作系統(tǒng)的實(shí)際情況而定的抽象表示,主要是看操作系統(tǒng)和庫是否支持和管理的線程,一般Linux主要通過pthread庫來實(shí)現(xiàn),早期版本不支持。
其中,在Hotspot JVM 中的 Java 線程與原生操作系統(tǒng)線程有直接的映射關(guān)系。當(dāng)線程本地存儲、緩沖區(qū)分配、同步對象、棧、程序計數(shù)器等準(zhǔn)備好以后,就會創(chuàng)建一個操作系統(tǒng)原生線程。
Java 線程結(jié)束,原生線程隨之被回收。操作系統(tǒng)負(fù)責(zé)調(diào)度所有線程,并把它們分配到任何可用的 CPU 上。
當(dāng)原生線程初始化完畢,就會調(diào)用 Java 線程的 run() 方法。當(dāng)線程結(jié)束時,會釋放原生線程和 Java 線程的所有資源。
一般在Hotspot JVM 后臺運(yùn)行的系統(tǒng)線程主要有下面幾方面:
虛擬機(jī)線程(VM thread):這個線程等待 JVM 到達(dá)安全點(diǎn)操作出現(xiàn)。這些操作必須要在獨(dú)立的線程里執(zhí)行,因?yàn)楫?dāng)堆修改無法進(jìn)行時,線程都需要 JVM 位于安全點(diǎn)。這些操作的類型有:stop-theworld
垃圾回收、線程棧 dump、線程暫停、線程偏向鎖(biased locking)解除。
周期性任務(wù)線程: 這線程負(fù)責(zé)定時器事件(也就是中斷),用來調(diào)度周期性操作的執(zhí)行。
GC 線程: 這些線程支持 JVM 中不同的垃圾回收活動。
編譯器線程: 這些線程在運(yùn)行時將字節(jié)碼動態(tài)編譯成本地平臺相關(guān)的機(jī)器碼。
信號分發(fā)線程: 這個線程接收發(fā)送到 JVM 的信號并調(diào)用適當(dāng)?shù)?JVM 方法處理。
由此可見,Java層到內(nèi)層層的線程創(chuàng)建的大致流程:java.lang.Thread(Java應(yīng)用程序?qū)?—>Java Thread(JVM 層)->OS Thread(操作系統(tǒng)層)->pthread(根據(jù)操作系統(tǒng)的情況而定)->內(nèi)核線程(Kernel Thread)。
更多關(guān)于“java培訓(xùn)”的問題,歡迎咨詢千鋒教育在線名師。千鋒教育多年辦學(xué),課程大綱緊跟企業(yè)需求,更科學(xué)更嚴(yán)謹(jǐn),每年培養(yǎng)泛IT人才近2萬人。不論你是零基礎(chǔ)還是想提升,都可以找到適合的班型,千鋒教育隨時歡迎你來試聽。