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

    浏览量

    8038

原文标题:究竟什么是阻塞?

文章出处:【微信号:gh_3980db2283cd,微信公众号:开关电源芯片】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    模块电源相关名词解释 稳压精度负载调整率动态负载等

    BOSHIDA 模块电源相关名词解释 稳压精度负载调整率动态负载等 16、稳压精度: 指电源在标称输入电压和标称输出负载的条件下,实测输出电压与输出电压标称值之差同输出电压标称值的百分比。 17
    的头像 发表于 04-09 09:10 115次阅读

    模块电源性能参数名词解释

    模块电源性能参数名词解释 BOSHIDA规格书 在电源模块技术手册中,有很多特性参数,下面给新手解释下各名词。 1、输入电压范围: 指电源在规定的工作条件下所指定输入端子间的电压标称值。 2、输入
    的头像 发表于 04-07 09:12 103次阅读
    模块电源性能参数<b class='flag-5'>名词解释</b>

    verilog同步和异步的区别 verilog阻塞赋值和非阻塞赋值的区别

    Verilog是一种硬件描述语言,用于设计和模拟数字电路。在Verilog中,同步和异步是用来描述数据传输和信号处理的两种不同方式,而阻塞赋值和非阻塞赋值是两种不同的赋值方式。本文将详细解释
    的头像 发表于 02-22 15:33 377次阅读

    常用AI名词解释

    AGI:Artificial General Intelligence (通用人工智能):是指具备与人类同等或超越人类的智能,能够表现出正常人类所具有的所有智能行为。又被称为强人工智能。
    的头像 发表于 12-21 15:40 467次阅读

    什么是CMOS图像传感器?CMOS图像传感器的基本名词解释

    Sensor,作为摄像头模组最重要的一部分,其总价值占比超过50%,在摄像头模组及相关行业,提到“Sensor”这个词,一般代指图像传感器。目前,除了一些特殊领域,CMOS图像传感器占据绝大部分市场。本期,我们将简单介绍CMOS图像传感器的一些基础概念和名词概念。
    的头像 发表于 11-30 16:19 2528次阅读
    什么是CMOS图像传感器?CMOS图像传感器的基本<b class='flag-5'>名词解释</b>

    什么是阻塞?怎么设计才能满足阻塞指标?

    阻塞就是外部有阻塞干扰信号的时候,设备还可以正常运行。一般分为带内阻塞和带外阻塞,由于直放站都是做宽带设备,一般只提带外阻塞
    的头像 发表于 10-10 11:22 667次阅读

    网络IO模型:阻塞与非阻塞

    阻塞 IO 模型 在Linux ,默认情况下所有的 socket 都是阻塞的,一个典型的读操作流程如图所示。 阻塞和非阻塞的概念描述的是用户线程调用内核 IO 操作的方式:
    的头像 发表于 10-08 17:16 486次阅读
    网络IO模型:<b class='flag-5'>阻塞</b>与非<b class='flag-5'>阻塞</b>

    线材基本电气特性名词解释分享

    Sparktest火花测试火花测试用于发现绝缘导体的绝缘皮不良.火花测试机通常用于芯线押出或芯线对绞工段.有时也用于总绞工段.一般带屏蔽的电缆(编织线,铝箔向外)押出外被时也用火花机测其不良点.基本方法为在与被测物相接触的电极与接地导体之间施加一电压.若绝缘介质不良(如太薄或一部分缺失),施加的电压会在接地导体上产生电弧.从而激发与此相连的指示器(如蜂鸣器.
    的头像 发表于 09-14 08:28 449次阅读
    线材基本电气特性<b class='flag-5'>名词解释</b>分享

    阻塞赋值与非阻塞赋值

    ”=“阻塞赋值与”
    的头像 发表于 09-12 09:06 651次阅读
    <b class='flag-5'>阻塞</b>赋值与非<b class='flag-5'>阻塞</b>赋值

    线材基本电气特性名词解释分享

    火花测试用于发现绝缘导体的绝缘皮不良.火花测试机通常用于芯线押出或芯线对绞工段.有时也用于总绞工段.一般带屏蔽的电缆(编织线,铝箔向外)押出外被时也用火花机测其不良点.基本方法为在与被测物相接触的电极与接地 导体之间施加一电压.若绝缘介质不良(如太薄或一部分缺失),施加的电压会在接地导体上产生电弧.从而激发与此相连的指示器(如蜂鸣器.灯.计数器等).;火花测试中存在危险高压,故相关设备必须完全接地.一般测试机可采用AC或DC电压,老使用AC可使用不同的频率.为了安全,测试电流通常限制为无致命危险的水平.
    的头像 发表于 09-08 10:24 294次阅读
    线材基本电气特性<b class='flag-5'>名词解释</b>分享

    一文了解阻塞赋值与非阻塞赋值

    今天给大家普及一下阻塞赋值和非阻塞赋值的相关知识
    的头像 发表于 07-07 14:15 1463次阅读
    一文了解<b class='flag-5'>阻塞</b>赋值与非<b class='flag-5'>阻塞</b>赋值

    阻塞与非阻塞通信的区别 阻塞和非阻塞应用场景

    阻塞通信(Blocking Communication):当进行阻塞通信时,调用者在发起一个I/O操作后会被阻塞,直到该操作完成返回才能继续执行后续代码。
    的头像 发表于 06-15 17:32 4125次阅读

    自动驾驶名词解释

    毫米波 :波长 1-10mm、频率 30-300GHz 的无线电频谱。 多普勒效应 :当声音、光和无线电波等振动源与观测者以相对速度运动时观测者所收到的振动频率与振动源所发出的频率不同当观测者靠近雷达天线时反射信号频率将高于发射信号频率。 ECU :电子控制单元(Electronicl Control Unit)控制汽车工作的微机控制器。 MCU : 微控制单元(Microcontrol
    发表于 06-06 11:18 0次下载
    自动驾驶<b class='flag-5'>名词解释</b>

    Verilog中阻塞和非阻塞赋值金规

    对于VerilogHDL语言中,经常在always模块中,面临两种赋值方式:阻塞赋值和非阻塞赋值。对于初学者,往往非常迷惑这两种赋值方式的用法,本章节主要介绍这两种文章的用法。其实,有时候概念稍微不清楚,Bug就会找到我们,下面一文扫清
    的头像 发表于 06-01 09:21 573次阅读

    超级源随SSF稳定性补偿分析(一)

    先来名词解释,SSF:Super Source Follower,也就是大家常说的超级源随。
    的头像 发表于 05-23 17:14 4564次阅读
    超级源随SSF稳定性补偿分析(一)