在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

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

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

掃碼添加小助手

加入工程師交流群

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

問兩個問題

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

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

這個函數是阻塞的么?

public void function() {

while(true){}

}

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

public void function() {

Thread.sleep(2000);

}

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

從一段 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);

}

}

運行這段代碼發現,程序將會“阻塞”在 scanner.nextLine() 這一行代碼,直到用戶輸入并且按下了回車鍵,程序才會繼續往下走,打印我們輸入的內容,并且結束。

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

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

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

由于我們知道這個代碼一定最終會觸發一次 linux 的 IO 操作相關的系統調用,所以我們用 strace 命令直接將其找到。

strace -ff -e trace=desc java Zuse

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

read(0,

當我們輸入一個字符串 “hello” 并按下回車后,這個系統調用函數被補全。

read(0, “hello

”, 8192)

OK大功告成,觸發 linux 的系統調用就是 read()

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

來到 linux 內核

linux 的系統調用會注冊到系統調用表(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 函數,這個函數在 linux 內核源碼的 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 (。。。);

。。。

}

我們讀取的是標準輸入,屬于字符型文件,走第一個分支。

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

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

跟到最后,發現一句關鍵代碼,讓我提起了精神。

if (EMPTY (tty-》secondary)) {

sleep_if_empty (&tty-》secondary);

}

再往里跟

static void sleep_if_empty (struct tty_queue *queue) {

// 關中斷

cli ();

// 只要隊列為空

while (EMPTY (*queue))

// 可中斷睡眠

interruptible_sleep_on (&queue-》proc_list);

// 開中斷

sti ();

}

繼續往里跟

// 將當前任務置為可中斷的等待狀態void interruptible_sleep_on (struct task_struct **p) {

。。。

current-》state = TASK_INTERRUPTIBLE;

schedule ();

。。。

}

OK,整個流程簡單描述就是,只要用戶不輸入,字符隊列就為空,此時將調用一個 interruptible_sleep_on 函數,將線程狀態變為可中斷的等待狀態,同時調用 schedule() 函數,強制進行一次進程調度。

從進程調度看阻塞的本質

關于進程是怎么調度的,可以看《上帝視角看進程調度》。

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

很簡答,這個函數就做了三件事:

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

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

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

而我們剛剛將當前任務設置為 TASK_INTERRUPTIBLE,就是告訴進程調度算法,下次不要調度我,相當于放棄了 CPU 的執行權,相當于將當前進程掛起。

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

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);

}

}

這就是阻塞的本質。

再看喚醒的本質就簡單了

有阻塞就有喚醒,當我們按下鍵盤時,會觸發鍵盤中斷,會進入鍵盤中斷處理函數,keyboard_interrupt。

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

再次經過九曲十八彎的跟蹤后,發現這樣一句代碼。

wake_up(&tty-》secondary.proc_list);

跟進去。

void wake_up(struct task_struct **p)

{

if (p && *p) {

(**p).state = TASK_RUNNABLE;

*p = NULL;

}

}

一目了然,將進程的狀態改為 RUNNABLE,一會進程調度時,就可以參與了。

這就是阻塞后,喚醒的本質。

總結

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

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

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

wake_up(&tty-》secondary.proc_list);

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

// 阻塞

current-》state = TASK_INTERRUPTIBLE;

// 喚醒

(**p).state = TASK_RUNNABLE;

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

這個函數是阻塞的么?

public void function() {

while(true){}

}

這個函數是阻塞的么?

public void function() {

Thread.sleep(2000);

}

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

而阻塞的本質,是將進程掛起,不再參與進程調度。

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

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

編輯:jq

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 阻塞
    +關注

    關注

    0

    文章

    24

    瀏覽量

    8276

原文標題:究竟什么是阻塞?

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

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

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    關于 PCB 拼板完整教程

    名詞解釋 在下面說明具體怎么操作前,先把幾個關鍵名詞解釋下Mark 點:如圖 2.1 所示, 圖 2.1 用來幫助貼片機的光學定位有貼片器件的 PCB 板對角至少有兩個不對稱基準點,整塊 PCB
    發表于 04-19 15:36

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

    本文詳細闡述了在一個testbench中,應該如何使用阻塞賦值與非阻塞賦值。首先說結論,建議在testbench中,對時鐘信號(包括分頻時鐘)使用阻塞賦值,對其他同步信號使用非阻塞賦值
    的頭像 發表于 04-15 09:34 ?629次閱讀
    在testbench中如何使用<b class='flag-5'>阻塞</b>賦值和非<b class='flag-5'>阻塞</b>賦值

    你一定要理解的晶振常見專業名詞解釋

    晶振(Crystal****Oscillator) 晶振是一種能夠產生穩定頻率信號的電子元件。它基于石英晶體的壓電效應工作,當在石英晶體兩端施加電壓時,晶體會產生機械振動;反之,當晶體受到機械應力時,其兩端又會產生電壓。利用這種特性,通過適當的電路設計,就能使晶體持續穩定地振動,從而輸出精確的頻率信號。晶振在各類電子設備中起著至關重要的作用,如計算機、通信設備、電子手表等,為這些設備提供穩定的時鐘信號,確保設備各部分協調工作。
    的頭像 發表于 04-10 17:17 ?385次閱讀
    你一定要理解的晶振常見專業<b class='flag-5'>名詞解釋</b>

    算力領域常用名詞解釋

    本文系統地整理和解釋了算力領域中常用的數十個關鍵名詞,并按照以下維度進行了分類:基礎概念、系統架構、硬件架構、基礎運算類型、計算模式、相關軟件架構與部署模式、浮點精度格式、算力類型、算力關聯與服務
    的頭像 發表于 04-07 11:21 ?452次閱讀
    算力領域常用<b class='flag-5'>名詞解釋</b>

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

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

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

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

    大模型領域常用名詞解釋(近100個)

    本文總結了大模型領域常用的近100個名詞解釋,并按照模型架構與基礎概念,訓練方法與技術,模型優化與壓縮,推理與應用,計算與性能優化,數據與標簽,模型評估與調試,特征與數據處理,倫理與公平性、其他
    的頭像 發表于 02-19 11:49 ?762次閱讀
    大模型領域常用<b class='flag-5'>名詞解釋</b>(近100個)

    “串口阻塞”你真的會用嗎?

    導讀在串口通信開發中,數據錯亂是常見問題。本文將快速介紹串口標志位的作用及配置方法,幫助解決數據傳輸錯誤。這是一個真實案例,用戶反饋“串口向另外的設備發送數據,發現運行一段時間后,發送的消息會阻塞
    的頭像 發表于 02-13 11:42 ?744次閱讀
    “串口<b class='flag-5'>阻塞</b>”你真的會用嗎?

    顯示器驅動版專業名詞介紹

    ? ?? 顯示器的世界充滿了各種奇妙的技術和專業知識,而顯示器驅動板更是其中的關鍵所在。此前給大家分享了相關的部分專業名詞,受到了不少朋友的關注呢。現在,“顯示器驅動版專業名詞(二)”重磅來襲
    的頭像 發表于 12-02 09:10 ?944次閱讀

    socket編程中的阻塞與非阻塞

    在網絡編程中, socket 是一個非常重要的概念,它提供了一個抽象層,使得開發者可以不必關心底層的網絡通信細節。 socket 編程中的阻塞與非阻塞模式是兩種不同的操作方式,它們對程序的響應性
    的頭像 發表于 11-01 16:13 ?724次閱讀

    虛擬主機名詞解釋

    虛擬主機(Virtual Host/Virtual Server)是一種在單一主機或主機群上,實現多網域服務的方法,可以運行多個網站或服務的技術。以下是關于虛擬主機的詳細解釋,rak小編為您整理發布虛擬主機名詞解釋
    的頭像 發表于 10-25 09:26 ?701次閱讀

    socket阻塞和非阻塞的區別是什么

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

    半導體知識百科:不可不知的50大專業名詞

    在半導體行業中,掌握專業名詞對于從業者來說至關重要。這些名詞不僅是行業交流的基礎,更是理解和掌握相關技術、工藝及產品的關鍵。以下是半導體人必須知道的50個專業名詞解釋
    的頭像 發表于 08-15 11:02 ?4939次閱讀
    半導體知識百科:不可不知的50大專業<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)預烘 (3)曝光 (4)顯影 (5)后烘 (6)腐蝕 (7)光刻膠的去除 5、 此處用干法氧化法將氮化硅去除 6 、離子布植將硼離子 (B+3) 透過 SiO2 膜注入襯底,形成 P 型阱 7、
    的頭像 發表于 07-30 08:43 ?3009次閱讀
    晶圓制造工藝流程及一些常用<b class='flag-5'>名詞解釋</b>

    如何在espconn_gethostbyname上創建阻塞線程?

    在 espconn_gethostbyname 上創建阻塞線程
    發表于 07-10 07:45
    主站蜘蛛池模板: 三级理论在线播放大全 | 人人看人人鲁狠狠高清 | 狠狠色综合久久婷婷 | 日日夜夜操天天干 | 超级乱淫伦网站 | 欧美色欧美亚洲高清在线视频 | 久久视频免费看 | 手机在线观看你懂的 | 五月婷婷六月天 | 操狠狠 | 免费高清视频免费观看 | free性欧美video69 | 四虎永久地址4hu紧急入口 | 亚洲激情综合 | 久久这里只精品热在线8 | 99国产精品农村一级毛片 | 色女人久久 | 狠狠色噜噜狠狠狠狠98 | 免费观看成年欧美1314www色 | 国产午夜精品不卡片 | 精品女同 | 国产三级精品三级在线观看 | 欧美性猛交xxxx黑人喷水 | 午夜寂寞在线一级观看免费 | 欧亚精品卡一卡二卡三 | 天天噜噜色 | 国产在线成人一区二区 | 欧美日本一区 | 男人日女人视频在线观看 | 五月婷婷社区 | 四虎永久影院永久影库 | 色婷婷综合在线 | 高清一区二区 | 黄色激情小说视频 | 中文久草| 丁香狠狠| 视频一区 在线 | 国产精品久久久久久久午夜片 | 美女一级牲交毛片视频 | 亚洲大香伊人蕉在人依线 | 久久久久久国产精品mv |