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

    浏览量

    8454

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

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

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    智驾圈该醒醒了!文远知行韩旭:别再迷信技术名词,能解决问题才是最重要的!

    ,成为推动行业从“概念喧嚣”走向“务实发展”的三大核心主线。而包括韩旭在内的行业人士普遍预判,自动驾驶系统能力全面超越人类驾驶的时间窗口,将集中在2032—2033年。 技术发展的核心是解决实际问题,而非陷入名词与路线的无意义博弈
    的头像 发表于 02-04 17:20 588次阅读
    智驾圈该醒醒了!文远知行韩旭:别再迷信技术<b class='flag-5'>名词</b>,能解决问题才是最重要的!

    宏展科技北京应力筛选专门名词整理

    宏展科技北京应力筛选专门名词整理:part(元器件)产品中可以拆装的最小可分辨专桉,如分立半导体器件、电阻、积体电路、焊点和连接器等。assembly(组件)设计成可装入某一单元并与类似或其他的元件
    的头像 发表于 02-02 11:58 276次阅读
    宏展科技北京应力筛选专门<b class='flag-5'>名词</b>整理

    电信接收机 IF/RF 前端 SAW 滤波方案:70–300MHz 选择性提升 + 阻塞防护 + 匹配验证全流程(FSF-5050 系列)

    电信接收机 IF/RF 前端易受强带外干扰引发压缩、互调与阻塞,DSP 难补救。本方案以 SAW 滤波器为“门禁”,覆盖 70–300MHz IF 规划,给出放置位置、IL/阻带/群时延取舍、匹配与 PCB 要点,并提供 VNA+阻塞测试验证流程,适配 FSF 5×5mm
    的头像 发表于 01-29 14:00 1243次阅读
    电信接收机 IF/RF 前端 SAW 滤波方案:70–300MHz 选择性提升 + <b class='flag-5'>阻塞</b>防护 + 匹配验证全流程(FSF-5050 系列)

    SMT贴片加工必备术语手册:49个常用名词及其详细定义

    SMT(Surface Mount Technology)贴片加工是现代电子制造中的关键技术,涉及众多专业名词。以下是49个SMT贴片加工中常用名词及其详细定义: 1. SMT(Surface
    发表于 01-27 11:14

    Modbus产品脚本编辑器“映射关系”解释?映射关系的地址对应关系?

    Modbus产品脚本编辑器“映射关系”解释?映射关系的地址对应关系?
    发表于 01-20 15:28

    解析SN65LVDS125A和SN65LVDT125A:4x4非阻塞交叉点开关的卓越性能

    解析SN65LVDS125A和SN65LVDT125A:4x4非阻塞交叉点开关的卓越性能 在电子设计领域,高速信号处理和灵活的信号路由是许多应用的关键需求。德州仪器(TI)的SN65LVDS125A
    的头像 发表于 12-29 17:40 741次阅读

    探索SN65LVDS125A和SN65LVDT125A:4x4非阻塞交叉点开关的卓越性能

    探索SN65LVDS125A和SN65LVDT125A:4x4非阻塞交叉点开关的卓越性能 在当今高速发展的电子领域,数据传输和信号处理的速度和效率至关重要。而SN65LVDS125A
    的头像 发表于 12-29 17:40 887次阅读

    具有阻塞总线恢复功能的 TCA4307 热插拔 I²C 总线和 SMBus 缓冲器介绍

    具有阻塞总线恢复功能的 TCA4307 热插拔 I²C 总线和 SMBus 缓冲器介绍 在电子设计领域,热插拔 I²C 总线缓冲器是实现系统灵活扩展和维护的重要组件。今天,我们就来深入了解一下
    的头像 发表于 12-17 15:40 525次阅读

    GPIO的常见名词

    2、输出输入GPIOGPIO是指通用输入输出(General Purpose Input/Output),是一种通用的数字输入输出接口,可以通过程序控制来读取或控制外部设备。 在单片机或嵌入式系统中,GPIO通常由一个或多个引脚组成,可以通过编程来配置引脚的工作模式(输入或输出)、电平状态(高电平或低电平)等参数,即, 数字芯片的IO口,一般分为输入和输出 数字芯片内部单元大都是CMOS,一般mos的栅极为输入,漏极为输出 2.1 基本类型输出: 开漏输出(OP) 推挽输出(PP) 复用开漏 复用推挽 输入: 浮空输入 下拉输入 上拉输入 3. 输出3.1 推挽输出推挽输出是指输出端口通过一个晶体管的开关来控制电平状态,推挽输出可以实现高电平和低电平两种状态 输出高电平时,NPN型晶体管导通,PNP型晶体管截止,呈现向外推的形式,为推 输出低电平时,PNP型晶体管导通,NPN型晶体管截止,呈现向回挽的形式,为挽 优点: 输出高低电平、电源电压基本没有压差 高低电平驱动能力较强,一般数字芯片推挽,输出IO口驱动电流最大可到20ma 电平切换速度快 缺点: 不支持线与(指俩个输出不可以接到一起) 3.2 开漏输出输出端口通过一个晶体管的开关来控制电平状态,但与推挽输出不同,开漏输出只能实现低电平状态,而不能直接输出高电平。 开漏输出的原理是,在输出端口接入一个NPN型晶体管,它的集电极连接到输出端口,发射极接地。当需要输出低电平时,晶体管导通,输出端口接地,形成低电平;当需要输出高电平时,晶体管截止,输出端口处于悬空状态,即高阻态。为了实现完整的输出功能,开漏输出通常需要结合外部上拉电阻。当输出端口处于高阻态时,外部上拉电阻将输出端口拉高至所需的高电平。因此,开漏输出可以通过控制晶体管的导通和截止来实现低电平输出,而高电平输出则由外部上拉电阻提供。开漏输出常用于多路设备共享总线的情况下,如I2C、SPI等通信协议中。通过多个开漏输出端口的组合,可以实现多路设备对总线的控制,同时避免输出冲突和电平干扰。 优点: 可实现电平转换,输出电平取决于上拉电阻电源 可以实现IO的线与 缺点: 高电平的驱动能力差,取决于外部上拉电阻 电平切换速率取决于外部上拉电阻 3.3 高阻态关于开漏输出的高阻态,他通常用在总线上,当主设备与其中一个从设备相连的时候,阻断其他设备与主设备相连,即呈现出高阻态的形式。 4. 输入4.1 上拉输入是指芯片输入引脚通过电阻接到电源电压 4.2 下拉输入指芯片输入引脚通过电阻借到参考0电平 4.3 浮空输入浮空输入是指将输入引脚未连接到任何外部信号源或电路,使其处于未定义的状态。在这种情况下,输入引脚既不连接到高电平(VDD)也不连接到低电平(GND),而是处于开路状态。 浮空输入可能会导致输入信号漂移或产生不确定的结果。这是因为未连接的输入引脚可以受到周围环境中的电磁干扰,从而引起电压波动,进而影响输入状态。在某些情况下,浮空输入可能会被误认为是高电平或低电平,这取决于具体的电路设计和输入引脚的特性。 5. 野火的STM32 GPIO电路设计 我们主要看一下他这个推挽、开漏输出的设计,N-MOS晶体管导通,P-MOS晶体管截止,呈现向外推的形式,为推,P-MOS管截止,N-MOS导通,为挽。 当为开漏状态时,只需要N-MOS管,当需要输出低电平时,晶体管导通,输出端口接地,形成低电平;当需要输出高电平时,晶体管截止,输出端口处于悬空状态,即高阻态。
    发表于 12-02 07:58

    飞凌嵌入式ElfBoard-文件I/O的深入学习之阻塞I/O与非阻塞I/O

    1.4.1.1 概念 阻塞I/O顾名思义就是对文件的I/O操作是阻塞式的,即假如对某些类型文件(管道文件、网络设备文件和字符设备文件)进行读操作时,如果数据未准备好、文件当前无数据可读,那么读操作
    发表于 12-01 13:07

    ESD测试的详细解释

    解释: 01 定义与目的 ● 定义:ESD测试是模拟操作人员或物体在接触设备时产生的放电,以及人或物体对邻近物体之放电,以检测被测设备抵抗静电放电之干扰能力。 ● 目的:通过ESD测试,可以有效地
    发表于 11-26 07:37

    40个嵌入式常见名词一次讲清!

    一个人的困惑。嵌入式开发涉及的术语确实又杂又多,而且很多词不仅缩写相似,使用语境也相互交叉。为了帮你打通这些基础概念,我们整理了一份嵌入式常见名词速查表,从最基础的G
    的头像 发表于 11-14 10:28 1349次阅读
    40个嵌入式常见<b class='flag-5'>名词</b>一次讲清!

    关于保偏光纤的详细解释

    保偏光纤(Polarization-Maintaining Fiber,简称PMF)是一种特殊设计的光纤,其核心功能是在传输过程中保持光的偏振态不变。以下是关于保偏光纤的详细解释: 1. 偏振态
    的头像 发表于 09-25 10:13 1287次阅读
    关于保偏光纤的详细<b class='flag-5'>解释</b>

    CH32V307 串口发送阻塞的原因?怎么解决?

    */ modbus_set_rx_mode(master); modbus发送,不知为何阻塞在rt_device_write,初始化 find open 正常
    发表于 09-18 06:06

    CH32V307串口发送阻塞的原因?如何解决?

    */ modbus_set_rx_mode(master); modbus发送,不知为何阻塞在rt_device_write,初始化 find open 正常
    发表于 06-11 08:05