0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

名詞解釋:阻塞是什么意思?

開關(guān)電源芯片 ? 來源:低并發(fā)編程 ? 作者:閃客sun ? 2021-07-21 09:47 ? 次閱讀

前言:很多詞匯,不論對科班生還是非科班生,如果不知道底層原理,就永遠(yuǎn)是一個魔法詞匯。這些魔法詞匯一多,就會導(dǎo)致暈頭轉(zhuǎn)向。所以開個新系列,降妖除魔,就是要斬殺這些如妖魔鬼怪般的魔法詞匯。

問兩個問題

阻塞,是我們程序員口中常常提到的詞。

這個詞,既熟悉,又陌生,熟悉到一提到它就倍感親切,但一具體解釋,就迷迷糊糊。

這個函數(shù)是阻塞的么?

public void function() {

while(true){}

}

如果你說不出來,那你再看看這個函數(shù)是阻塞的么?

public void function() {

Thread.sleep(2000);

}

為了搞清楚這個問題,我們就來一起追蹤一下阻塞的本質(zhì),消滅阻塞這個魔法詞匯。

從一段 Java 代碼開始

寫一段很簡單的 java 代碼

import java.util.Scanner;

public class Zuse {

public static void main(String[] args) {

Scanner scanner = new Scanner(System.in);

String line = scanner.nextLine();

System.out.println(line);

}

}

運行這段代碼發(fā)現(xiàn),程序?qū)白枞痹?scanner.nextLine() 這一行代碼,直到用戶輸入并且按下了回車鍵,程序才會繼續(xù)往下走,打印我們輸入的內(nèi)容,并且結(jié)束。

我們跟蹤一下這一行代碼的源碼,九曲十八彎之后,終于跟蹤到了一個不能再往下跟蹤的 native 代碼。

private native int readBytes(byte b[], int off, int len) throws IOException;

當(dāng)然我們可以通過 openJDK 源碼繼續(xù)查下去,但我有點懶,怕翻車,這里用另一個巧妙的辦法。

由于我們知道這個代碼一定最終會觸發(fā)一次 linux 的 IO 操作相關(guān)的系統(tǒng)調(diào)用,所以我們用 strace 命令直接將其找到。

strace -ff -e trace=desc java Zuse

我們看到程序阻塞在了這里。

read(0,

當(dāng)我們輸入一個字符串 “hello” 并按下回車后,這個系統(tǒng)調(diào)用函數(shù)被補全。

read(0, “hello

”, 8192)

OK大功告成,觸發(fā) linux 的系統(tǒng)調(diào)用就是 read()

這樣,我們成功通過 strace 命令,直接跨越到了 linux 內(nèi)核里,中間的調(diào)用過程,就不用瞎操心了。

來到 linux 內(nèi)核

linux 的系統(tǒng)調(diào)用會注冊到系統(tǒng)調(diào)用表(sys_call_table)中,通常是在前綴加一個 sys_。

fn_ptr sys_call_table[] = { sys_setup, sys_exit, sys_fork, sys_read,

sys_write, sys_open, sys_close, sys_waitpid, sys_creat, sys_link,

sys_unlink, sys_execve, sys_chdir, sys_time, sys_mknod, sys_chmod,

sys_chown, sys_break, sys_stat, sys_lseek, sys_getpid, sys_mount,

sys_umount, sys_setuid, sys_getuid, sys_stime, sys_ptrace, sys_alarm,

sys_fstat, sys_pause, sys_utime, sys_stty, sys_gtty, sys_access,

sys_nice, sys_ftime, sys_sync, sys_kill, sys_rename, sys_mkdir,

sys_rmdir, sys_dup, sys_pipe, sys_times, sys_prof, sys_brk, sys_setgid,

sys_getgid, sys_signal, sys_geteuid, sys_getegid, sys_acct, sys_phys,

sys_lock, sys_ioctl, sys_fcntl, sys_mpx, sys_setpgid, sys_ulimit,

sys_uname, sys_umask, sys_chroot, sys_ustat, sys_dup2, sys_getppid,

sys_getpgrp, sys_setsid, sys_sigaction, sys_sgetmask, sys_ssetmask,

sys_setreuid, sys_setregid

};

所以我們就定位到 sys_read 函數(shù),這個函數(shù)在 linux 內(nèi)核源碼的 read_write.c 文件中。

int sys_read (unsigned int fd, char *buf, int count)

{

。。。

if (S_ISCHR (inode-》i_mode))

return rw_char (。。。);

if (S_ISBLK (inode-》i_mode))

return block_read (。。。);

。。。

}

我們讀取的是標(biāo)準(zhǔn)輸入,屬于字符型文件,走第一個分支。

之后,要經(jīng)過非常非常多的調(diào)用棧,我感覺是 linux 當(dāng)中最繁瑣的歷程了,這個過程在我腦子里還是一片漿糊。具體可以看飛哥的《read一個字節(jié)實際發(fā)生了什么》,一行一行源碼給你分析清楚,不過是以讀取磁盤為例,和這個讀取終端設(shè)備一樣也要經(jīng)歷文件系統(tǒng)的層層折磨。

由于我們只想知道阻塞的本質(zhì),所以,忽略中間這一大坨。

跟到最后,發(fā)現(xiàn)一句關(guān)鍵代碼,讓我提起了精神。

if (EMPTY (tty-》secondary)) {

sleep_if_empty (&tty-》secondary);

}

再往里跟

static void sleep_if_empty (struct tty_queue *queue) {

// 關(guān)中斷

cli ();

// 只要隊列為空

while (EMPTY (*queue))

// 可中斷睡眠

interruptible_sleep_on (&queue-》proc_list);

// 開中斷

sti ();

}

繼續(xù)往里跟

// 將當(dāng)前任務(wù)置為可中斷的等待狀態(tài)void interruptible_sleep_on (struct task_struct **p) {

。。。

current-》state = TASK_INTERRUPTIBLE;

schedule ();

。。。

}

OK,整個流程簡單描述就是,只要用戶不輸入,字符隊列就為空,此時將調(diào)用一個 interruptible_sleep_on 函數(shù),將線程狀態(tài)變?yōu)榭芍袛嗟牡却隣顟B(tài),同時調(diào)用 schedule() 函數(shù),強制進行一次進程調(diào)度。

從進程調(diào)度看阻塞的本質(zhì)

關(guān)于進程是怎么調(diào)度的,可以看《上帝視角看進程調(diào)度》。

我這里簡單挑出重點,說明一下 schedule 也就是進程調(diào)度的過程,以 linux-0.11 為例。

很簡答,這個函數(shù)就做了三件事:

1. 拿到剩余時間片(counter的值)最大且在 runnable 狀態(tài)(state = 0)的進程號 next。

2. 如果所有 runnable 進程時間片都為 0,則將所有進程(注意不僅僅是 runnable 的進程)的 counter 重新賦值(counter = counter/2 + priority),然后再次執(zhí)行步驟 1。3. 最后拿到了一個進程號 next,調(diào)用了 switch_to(next) 這個方法,就切換到了這個進程去執(zhí)行了。

我們只看第一條就好了,進程調(diào)度機制在選擇下一個要調(diào)度的進程時,會跳過不是 RUNNABLE 狀態(tài)的進程。

而我們剛剛將當(dāng)前任務(wù)設(shè)置為 TASK_INTERRUPTIBLE,就是告訴進程調(diào)度算法,下次不要調(diào)度我,相當(dāng)于放棄了 CPU 的執(zhí)行權(quán),相當(dāng)于將當(dāng)前進程掛起。

而底層的這一個操作,直接導(dǎo)致上層看來,像是停在了那一行不走一樣,就是這一行。

import java.util.Scanner;

public class Zuse {public static void main(String[] args) {

Scanner scanner = new Scanner(System.in);

String line = scanner.nextLine();

System.out.println(line);

}

}

這就是阻塞的本質(zhì)。

再看喚醒的本質(zhì)就簡單了

有阻塞就有喚醒,當(dāng)我們按下鍵盤時,會觸發(fā)鍵盤中斷,會進入鍵盤中斷處理函數(shù),keyboard_interrupt。

這個函數(shù)是提前注冊在中斷向量表里的。

再次經(jīng)過九曲十八彎的跟蹤后,發(fā)現(xiàn)這樣一句代碼。

wake_up(&tty-》secondary.proc_list);

跟進去。

void wake_up(struct task_struct **p)

{

if (p && *p) {

(**p).state = TASK_RUNNABLE;

*p = NULL;

}

}

一目了然,將進程的狀態(tài)改為 RUNNABLE,一會進程調(diào)度時,就可以參與了。

這就是阻塞后,喚醒的本質(zhì)。

總結(jié)

所以,Java 代碼中的一行 readline 會導(dǎo)致阻塞,實際上就是運行到了這段代碼。

interruptible_sleep_on (&tty-》secondary-》proc_list);

而鍵盤輸入后會將其喚醒,實際上就是運行到了這段代碼。

wake_up(&tty-》secondary.proc_list);

這兩段代碼里,其實就是通過改寫 state 值去玩的,剩下的交給調(diào)度算法。

// 阻塞

current-》state = TASK_INTERRUPTIBLE;

// 喚醒

(**p).state = TASK_RUNNABLE;

所以開篇兩個問題,你可以回答了么?

這個函數(shù)是阻塞的么?

public void function() {

while(true){}

}

這個函數(shù)是阻塞的么?

public void function() {

Thread.sleep(2000);

}

答案都是否定的,因為這兩個都沒有讓出 CPU 資源。(筆誤,sleep是讓出CPU資源的)

而阻塞的本質(zhì),是將進程掛起,不再參與進程調(diào)度。

而掛起的本質(zhì),其實就是將進程的 state 賦值為非 RUNNABLE,這樣調(diào)度機制的代碼中,就不會把它作為下一個獲得 CPU 運行機會的可選項了。

怎么樣,阻塞這個妖魔,除了么?

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 阻塞
    +關(guān)注

    關(guān)注

    0

    文章

    24

    瀏覽量

    8073

原文標(biāo)題:究竟什么是阻塞?

文章出處:【微信號:gh_3980db2283cd,微信公眾號:開關(guān)電源芯片】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    socket阻塞和非阻塞的區(qū)別是什么

    在計算機編程中,socket 是一種通信端點,用于在網(wǎng)絡(luò)中進行數(shù)據(jù)傳輸。Socket 可以是阻塞的或非阻塞的,這兩種模式在處理數(shù)據(jù)傳輸時有不同的行為。 阻塞模式(Blocking Mode) 在
    的頭像 發(fā)表于 08-16 11:13 ?434次閱讀

    半導(dǎo)體知識百科:不可不知的50大專業(yè)名詞

    在半導(dǎo)體行業(yè)中,掌握專業(yè)名詞對于從業(yè)者來說至關(guān)重要。這些名詞不僅是行業(yè)交流的基礎(chǔ),更是理解和掌握相關(guān)技術(shù)、工藝及產(chǎn)品的關(guān)鍵。以下是半導(dǎo)體人必須知道的50個專業(yè)名詞解釋。
    的頭像 發(fā)表于 08-15 11:02 ?664次閱讀
    半導(dǎo)體知識百科:不可不知的50大專業(yè)<b class='flag-5'>名詞</b>

    晶圓制造工藝流程及一些常用名詞解釋

    共讀好書 晶圓制造工藝流程 1、 表面清洗 2、 初次氧化 3、 CVD(Chemical Vapor deposition) 法沉積一層 Si3N4 (Hot CVD 或 LPCVD) 。 (1)常壓 CVD (Normal Pressure CVD) (2)低壓 CVD (Low Pressure CVD) (3)熱 CVD (Hot CVD)/(thermal CVD) (4)電漿增強 CVD (Plasma Enhanced CVD) (5)MOCVD (Metal Organic (6)外延生長法 (LPE) 4、 涂敷光刻膠 (1)光刻膠的涂敷 (2)預(yù)烘 (3)曝光 (4)顯影 (5)后烘 (6)腐蝕 (7)光刻膠的去除 5、 此處用干法氧化法將氮化硅去除 6 、離子布植將硼離子 (B+3) 透過 SiO2 膜注入襯底,形成 P 型阱 7、
    的頭像 發(fā)表于 07-30 08:43 ?1079次閱讀
    晶圓制造工藝流程及一些常用<b class='flag-5'>名詞解釋</b>

    半導(dǎo)體行業(yè)常用術(shù)語大全:30個專業(yè)名詞詳解

    在半導(dǎo)體行業(yè)中,了解和掌握專業(yè)名詞是至關(guān)重要的。這些名詞不僅有助于我們更好地理解半導(dǎo)體的制造過程,還能提升我們在行業(yè)中的溝通效率。以下是半導(dǎo)體人必須知道的30個專業(yè)名詞解釋
    的頭像 發(fā)表于 06-14 10:47 ?8589次閱讀
    半導(dǎo)體行業(yè)常用術(shù)語大全:30個專業(yè)<b class='flag-5'>名詞</b>詳解

    科普 | 電能質(zhì)量產(chǎn)品相關(guān)名詞解釋

    電能質(zhì)量產(chǎn)品涉及多個電壓等級和參數(shù),如額定電壓、供電電壓和用電電壓等。此外,文章還討論了電力系統(tǒng)頻率、供電電壓偏差、電壓快速變化、電壓不平衡、閃變、電壓暫降、暫升、中斷、瞬態(tài)過電壓以及間諧波和電網(wǎng)信號電壓等電能質(zhì)量問題及其原因。
    的頭像 發(fā)表于 05-30 17:09 ?247次閱讀
    科普 | 電能質(zhì)量產(chǎn)品相關(guān)<b class='flag-5'>名詞解釋</b>

    模塊電源相關(guān)名詞解釋 穩(wěn)壓精度負(fù)載調(diào)整率動態(tài)負(fù)載等

    BOSHIDA 模塊電源相關(guān)名詞解釋 穩(wěn)壓精度負(fù)載調(diào)整率動態(tài)負(fù)載等 16、穩(wěn)壓精度: 指電源在標(biāo)稱輸入電壓和標(biāo)稱輸出負(fù)載的條件下,實測輸出電壓與輸出電壓標(biāo)稱值之差同輸出電壓標(biāo)稱值的百分比。 17
    的頭像 發(fā)表于 04-09 09:10 ?468次閱讀

    模塊電源性能參數(shù)名詞解釋 BOSHIDA規(guī)格書

    模塊電源
    穩(wěn)控自動化
    發(fā)布于 :2024年04月08日 09:56:48

    模塊電源性能參數(shù)名詞解釋

    模塊電源性能參數(shù)名詞解釋 BOSHIDA規(guī)格書 在電源模塊技術(shù)手冊中,有很多特性參數(shù),下面給新手解釋下各名詞。 1、輸入電壓范圍: 指電源在規(guī)定的工作條件下所指定輸入端子間的電壓標(biāo)稱值。 2、輸入
    的頭像 發(fā)表于 04-07 09:12 ?389次閱讀
    模塊電源性能參數(shù)<b class='flag-5'>名詞解釋</b>

    什么是阻塞和非阻塞?

    什么是阻塞和非阻塞?我們就用管道的讀寫來舉例子。
    的頭像 發(fā)表于 03-25 10:04 ?426次閱讀

    verilog同步和異步的區(qū)別 verilog阻塞賦值和非阻塞賦值的區(qū)別

    Verilog是一種硬件描述語言,用于設(shè)計和模擬數(shù)字電路。在Verilog中,同步和異步是用來描述數(shù)據(jù)傳輸和信號處理的兩種不同方式,而阻塞賦值和非阻塞賦值是兩種不同的賦值方式。本文將詳細(xì)解釋
    的頭像 發(fā)表于 02-22 15:33 ?1328次閱讀

    常用AI名詞解釋

    AGI:Artificial General Intelligence (通用人工智能):是指具備與人類同等或超越人類的智能,能夠表現(xiàn)出正常人類所具有的所有智能行為。又被稱為強人工智能。
    的頭像 發(fā)表于 12-21 15:40 ?1085次閱讀

    什么是CMOS圖像傳感器?CMOS圖像傳感器的基本名詞解釋

    Sensor,作為攝像頭模組最重要的一部分,其總價值占比超過50%,在攝像頭模組及相關(guān)行業(yè),提到“Sensor”這個詞,一般代指圖像傳感器。目前,除了一些特殊領(lǐng)域,CMOS圖像傳感器占據(jù)絕大部分市場。本期,我們將簡單介紹CMOS圖像傳感器的一些基礎(chǔ)概念和名詞概念。
    的頭像 發(fā)表于 11-30 16:19 ?5025次閱讀
    什么是CMOS圖像傳感器?CMOS圖像傳感器的基本<b class='flag-5'>名詞解釋</b>

    什么事件會使執(zhí)行變成阻塞

    執(zhí)行阻塞是指當(dāng)一個進程無法繼續(xù)執(zhí)行時被掛起的狀態(tài)。這可以由多種事件引起,下面詳細(xì)介紹了一些常見的情況: I/O操作阻塞:在進行文件讀寫、網(wǎng)絡(luò)通信、數(shù)據(jù)庫訪問等I/O操作時,如果所需的數(shù)據(jù)還沒有
    的頭像 發(fā)表于 11-17 14:08 ?788次閱讀

    阻塞態(tài)可以直接到運行態(tài)嗎

    阻塞態(tài)即是指進程或線程在等待某種事件或資源時暫時停止執(zhí)行的狀態(tài)。在計算機系統(tǒng)中,由于各種原因,進程或線程可能會進入阻塞態(tài),等待著能夠繼續(xù)執(zhí)行的條件成熟。 在絕大多數(shù)情況下,阻塞態(tài)到運行態(tài)并不是直接
    的頭像 發(fā)表于 11-17 11:43 ?2076次閱讀

    阻塞的的connect()函數(shù)如何編寫

    由于網(wǎng)絡(luò)編程涉及很多細(xì)節(jié)和技巧,一直想寫篇文章來總結(jié)下這方面的心得與經(jīng)驗,希望對來者有一點幫助,那就善莫大焉了。 一、非阻塞的的connect()函數(shù)如何編寫 我們知道用connect()函數(shù)默認(rèn)
    的頭像 發(fā)表于 11-11 16:23 ?1330次閱讀
    非<b class='flag-5'>阻塞</b>的的connect()函數(shù)如何編寫