99精品伊人亚洲|最近国产中文炮友|九草在线视频支援|AV网站大全最新|美女黄片免费观看|国产精品资源视频|精彩无码视频一区|91大神在线后入|伊人终合在线播放|久草综合久久中文

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

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

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

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

開(kāi)關(guān)電源芯片 ? 來(lái)源:低并發(fā)編程 ? 作者:閃客sun ? 2021-07-21 09:47 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

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

問(wèn)兩個(gè)問(wèn)題

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

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

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

public void function() {

while(true){}

}

如果你說(shuō)不出來(lái),那你再看看這個(gè)函數(shù)是阻塞的么?

public void function() {

Thread.sleep(2000);

}

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

從一段 Java 代碼開(kāi)始

寫(xiě)一段很簡(jiǎn)單的 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);

}

}

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

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

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

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

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

strace -ff -e trace=desc java Zuse

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

read(0,

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

read(0, “hello

”, 8192)

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

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

來(lái)到 linux 內(nèi)核

linux 的系統(tǒng)調(diào)用會(huì)注冊(cè)到系統(tǒng)調(diào)用表(sys_call_table)中,通常是在前綴加一個(gè) 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ù),這個(gè)函數(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)輸入,屬于字符型文件,走第一個(gè)分支。

之后,要經(jīng)過(guò)非常非常多的調(diào)用棧,我感覺(jué)是 linux 當(dāng)中最繁瑣的歷程了,這個(gè)過(guò)程在我腦子里還是一片漿糊。具體可以看飛哥的《read一個(gè)字節(jié)實(shí)際發(fā)生了什么》,一行一行源碼給你分析清楚,不過(guò)是以讀取磁盤(pán)為例,和這個(gè)讀取終端設(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 ();

// 只要隊(duì)列為空

while (EMPTY (*queue))

// 可中斷睡眠

interruptible_sleep_on (&queue-》proc_list);

// 開(kāi)中斷

sti ();

}

繼續(xù)往里跟

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

。。。

current-》state = TASK_INTERRUPTIBLE;

schedule ();

。。。

}

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

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

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

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

很簡(jiǎn)答,這個(gè)函數(shù)就做了三件事:

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

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

我們只看第一條就好了,進(jìn)程調(diào)度機(jī)制在選擇下一個(gè)要調(diào)度的進(jìn)程時(shí),會(huì)跳過(guò)不是 RUNNABLE 狀態(tài)的進(jìn)程。

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

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

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ì)就簡(jiǎn)單了

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

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

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

wake_up(&tty-》secondary.proc_list);

跟進(jìn)去。

void wake_up(struct task_struct **p)

{

if (p && *p) {

(**p).state = TASK_RUNNABLE;

*p = NULL;

}

}

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

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

總結(jié)

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

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

而鍵盤(pán)輸入后會(huì)將其喚醒,實(shí)際上就是運(yùn)行到了這段代碼。

wake_up(&tty-》secondary.proc_list);

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

// 阻塞

current-》state = TASK_INTERRUPTIBLE;

// 喚醒

(**p).state = TASK_RUNNABLE;

所以開(kāi)篇兩個(gè)問(wèn)題,你可以回答了么?

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

public void function() {

while(true){}

}

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

public void function() {

Thread.sleep(2000);

}

答案都是否定的,因?yàn)檫@兩個(gè)都沒(méi)有讓出 CPU 資源。(筆誤,sleep是讓出CPU資源的)

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

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

怎么樣,阻塞這個(gè)妖魔,除了么?

編輯:jq

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

    關(guān)注

    0

    文章

    24

    瀏覽量

    8294

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

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

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    關(guān)于 PCB 拼板完整教程

    、名詞解釋 在下面說(shuō)明具體怎么操作前,先把幾個(gè)關(guān)鍵名詞解釋下Mark 點(diǎn):如圖 2.1 所示, 圖 2.1 用來(lái)幫助貼片機(jī)的光學(xué)定位有貼片器件的 PCB 板對(duì)角至少有兩個(gè)不對(duì)稱基準(zhǔn)點(diǎn),整塊 PCB
    發(fā)表于 04-19 15:36

    在testbench中如何使用阻塞賦值和非阻塞賦值

    本文詳細(xì)闡述了在一個(gè)testbench中,應(yīng)該如何使用阻塞賦值與非阻塞賦值。首先說(shuō)結(jié)論,建議在testbench中,對(duì)時(shí)鐘信號(hào)(包括分頻時(shí)鐘)使用阻塞賦值,對(duì)其他同步信號(hào)使用非阻塞賦值
    的頭像 發(fā)表于 04-15 09:34 ?667次閱讀
    在testbench中如何使用<b class='flag-5'>阻塞</b>賦值和非<b class='flag-5'>阻塞</b>賦值

    你一定要理解的晶振常見(jiàn)專業(yè)名詞解釋

    晶振(Crystal****Oscillator) 晶振是一種能夠產(chǎn)生穩(wěn)定頻率信號(hào)的電子元件。它基于石英晶體的壓電效應(yīng)工作,當(dāng)在石英晶體兩端施加電壓時(shí),晶體會(huì)產(chǎn)生機(jī)械振動(dòng);反之,當(dāng)晶體受到機(jī)械應(yīng)力時(shí),其兩端又會(huì)產(chǎn)生電壓。利用這種特性,通過(guò)適當(dāng)?shù)碾娐吩O(shè)計(jì),就能使晶體持續(xù)穩(wěn)定地振動(dòng),從而輸出精確的頻率信號(hào)。晶振在各類電子設(shè)備中起著至關(guān)重要的作用,如計(jì)算機(jī)、通信設(shè)備、電子手表等,為這些設(shè)備提供穩(wěn)定的時(shí)鐘信號(hào),確保設(shè)備各部分協(xié)調(diào)工作。
    的頭像 發(fā)表于 04-10 17:17 ?424次閱讀
    你一定要理解的晶振常見(jiàn)專業(yè)<b class='flag-5'>名詞解釋</b>

    算力領(lǐng)域常用名詞解釋

    本文系統(tǒng)地整理和解釋了算力領(lǐng)域中常用的數(shù)十個(gè)關(guān)鍵名詞,并按照以下維度進(jìn)行了分類:基礎(chǔ)概念、系統(tǒng)架構(gòu)、硬件架構(gòu)、基礎(chǔ)運(yùn)算類型、計(jì)算模式、相關(guān)軟件架構(gòu)與部署模式、浮點(diǎn)精度格式、算力類型、算力關(guān)聯(lián)與服務(wù)
    的頭像 發(fā)表于 04-07 11:21 ?512次閱讀
    算力領(lǐng)域常用<b class='flag-5'>名詞解釋</b>

    晶振圈專業(yè)名詞解釋,你都知道嗎(下)

    晶振在最低階振動(dòng)模式下產(chǎn)生的頻率,也就是它的“主振動(dòng)頻率”。基頻是晶振最基礎(chǔ)、最主要的振動(dòng)頻率,其他振動(dòng)模式(如泛音)都是基于基頻的倍數(shù)或衍生?;l決定了晶振的核心工作頻率。
    的頭像 發(fā)表于 03-19 17:03 ?785次閱讀
    晶振圈專業(yè)<b class='flag-5'>名詞解釋</b>,你都知道嗎(下)

    晶振圈專業(yè)名詞解釋,你都知道嗎(上)

    晶振是晶體振蕩器的簡(jiǎn)稱,是一種利用石英晶體的壓電效應(yīng)產(chǎn)生穩(wěn)定振蕩頻率的電子元件
    的頭像 發(fā)表于 03-18 16:22 ?798次閱讀
    晶振圈專業(yè)<b class='flag-5'>名詞解釋</b>,你都知道嗎(上)

    大模型領(lǐng)域常用名詞解釋(近100個(gè))

    本文總結(jié)了大模型領(lǐng)域常用的近100個(gè)名詞解釋,并按照模型架構(gòu)與基礎(chǔ)概念,訓(xùn)練方法與技術(shù),模型優(yōu)化與壓縮,推理與應(yīng)用,計(jì)算與性能優(yōu)化,數(shù)據(jù)與標(biāo)簽,模型評(píng)估與調(diào)試,特征與數(shù)據(jù)處理,倫理與公平性、其他
    的頭像 發(fā)表于 02-19 11:49 ?797次閱讀
    大模型領(lǐng)域常用<b class='flag-5'>名詞解釋</b>(近100個(gè))

    “串口阻塞”你真的會(huì)用嗎?

    導(dǎo)讀在串口通信開(kāi)發(fā)中,數(shù)據(jù)錯(cuò)亂是常見(jiàn)問(wèn)題。本文將快速介紹串口標(biāo)志位的作用及配置方法,幫助解決數(shù)據(jù)傳輸錯(cuò)誤。這是一個(gè)真實(shí)案例,用戶反饋“串口向另外的設(shè)備發(fā)送數(shù)據(jù),發(fā)現(xiàn)運(yùn)行一段時(shí)間后,發(fā)送的消息會(huì)阻塞
    的頭像 發(fā)表于 02-13 11:42 ?766次閱讀
    “串口<b class='flag-5'>阻塞</b>”你真的會(huì)用嗎?

    顯示器驅(qū)動(dòng)版專業(yè)名詞介紹

    ? ?? 顯示器的世界充滿了各種奇妙的技術(shù)和專業(yè)知識(shí),而顯示器驅(qū)動(dòng)板更是其中的關(guān)鍵所在。此前給大家分享了相關(guān)的部分專業(yè)名詞,受到了不少朋友的關(guān)注呢。現(xiàn)在,“顯示器驅(qū)動(dòng)版專業(yè)名詞(二)”重磅來(lái)襲
    的頭像 發(fā)表于 12-02 09:10 ?974次閱讀

    TPA3004功放,開(kāi)機(jī)保護(hù),聲音阻塞,為什么?

    TPA3004功放,開(kāi)機(jī)保護(hù),聲音阻塞。過(guò)一段時(shí)間,芯片發(fā)熱后正常。詢求問(wèn)題解決方法
    發(fā)表于 11-08 08:30

    socket編程中的阻塞與非阻塞

    在網(wǎng)絡(luò)編程中, socket 是一個(gè)非常重要的概念,它提供了一個(gè)抽象層,使得開(kāi)發(fā)者可以不必關(guān)心底層的網(wǎng)絡(luò)通信細(xì)節(jié)。 socket 編程中的阻塞與非阻塞模式是兩種不同的操作方式,它們對(duì)程序的響應(yīng)性
    的頭像 發(fā)表于 11-01 16:13 ?755次閱讀

    虛擬主機(jī)名詞解釋

    虛擬主機(jī)(Virtual Host/Virtual Server)是一種在單一主機(jī)或主機(jī)群上,實(shí)現(xiàn)多網(wǎng)域服務(wù)的方法,可以運(yùn)行多個(gè)網(wǎng)站或服務(wù)的技術(shù)。以下是關(guān)于虛擬主機(jī)的詳細(xì)解釋,rak小編為您整理發(fā)布虛擬主機(jī)名詞解釋
    的頭像 發(fā)表于 10-25 09:26 ?736次閱讀

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

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

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

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

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

    共讀好書(shū) 晶圓制造工藝流程 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)電漿增強(qiáng) CVD (Plasma Enhanced CVD) (5)MOCVD (Metal Organic (6)外延生長(zhǎng)法 (LPE) 4、 涂敷光刻膠 (1)光刻膠的涂敷 (2)預(yù)烘 (3)曝光 (4)顯影 (5)后烘 (6)腐蝕 (7)光刻膠的去除 5、 此處用干法氧化法將氮化硅去除 6 、離子布植將硼離子 (B+3) 透過(guò) SiO2 膜注入襯底,形成 P 型阱 7、
    的頭像 發(fā)表于 07-30 08:43 ?3088次閱讀
    晶圓制造工藝流程及一些常用<b class='flag-5'>名詞解釋</b>