【Linux】进程通信 — 信号(下篇)

文章目录

  • 📖 前言
  • 1. 阻塞信号
    • 1.1 信号其他相关常见概念:
    • 1.2 sigset_t:
      • 1.2 - 1 信号集操作函数
    • 1.3 sigprocmask:
    • 1.4 sigpending:
  • 2. 进程处理信号
    • 2.1 内核页表和用户页表:
    • 2.2 内核态和用户态:
    • 2.3 信号检测过程:
      • 2.3 - 1 便捷记忆图
    • 2.4 sigaction:
  • 3. volatile关键字
  • 4. 子进程给父进程发信号

📖 前言

上一篇我们讲述了信号的基本概念和相应系统接口的使用,本章我们想更深入的学习信号发送的一系列过程,目标已经确定,接下来就要搬好小板凳,准备开讲了…🙆🙆🙆🙆


1. 阻塞信号

1.1 信号其他相关常见概念:

  • 实际执行信号的处理动作称为信号递达(Delivery)。
  • 信号从产生到递达之间的状态,称为信号未决(Pending)。
  • 进程可以选择阻塞 (Block)某个信号。
  • 被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才执行递达的动作。
  • 注意,阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是在递达之后可选的一种处理动作。

在这里插入图片描述
pending表可以理解为是一张位图,32个比特位。

  • 比特位为0/1,表示哪一个信号是否收到。

handler表可以理解为是一个函数指针数组。

  • 对应的该信号也有对应的方法。

block叫做阻塞信号集。阻塞信号集(Block Signal Set)和信号屏蔽字(Signal Mask)是指相同的概念。

  • 拦不住发信号,但是可以拦得住递达这个信号。
  • 有些信号我们不想处理,但是防不住别人发,这个block也是一个位图,位图结构和pending位图是一模一样的。
  • 第几个比特位就代表着是几号信号,不一样的是,比特位的内容,在pending表里代表的是,是否收到内容,在block表中代表是否阻塞该信号。

block位图对应的比特位,为1的时候会拦截对应的信号去执行对应的方法。即使pending收到了该信号,只要是block位图对应的比特位为1,那么这个信号就无法去递达。

阻塞和忽略有什么区别呢?

忽略信号是处理信号的一种,只不过处理的方式是忽略它。(就是什么都不做,将pending位图由1置0就完了)

补充:

  • 在Linux中, 普通信号(非实时信号)多次发送并不会被记录多次。当同一个信号被多次发送给进程时,操作系统只会在进程的信号处理程序中记录次,而不会累积多个相同的信号。
  • 当进程接收到一个信号时,操作系统会将该信号标记为已挂起,直到进程处理完当前正在处理的信号或者通过信号处理程序返回后,才会再次传递给进程。
  • 这意味着,如果进程在处理信号期间接收到了多个相同的信号, 那么只有一 个信号会被记录和传递给进程的信号处理程序。
    例如:多次发送二号信号,只有一个会被递达,多余发出的信号被丢弃掉了。

1.2 sigset_t:

sigset_t是操作系统专门针对信号所构建的用户级的数据类型。

sigset_ t类型称之为信号集。

  • 可以表示每个信号的有或者无这样的概念。
  • 在阻塞信号集中表示有没有被阻塞这样的概念。
  • 在未决信号集中表示有或者没有被pending起来,或者未决起来。
  • sigset_t不能手动修改进制位图,要用对应的接口。

1.2 - 1 信号集操作函数

#include <signal.h>

int sigemptyset(sigset_t *set);对信号集做清空,可以理解为全清零。

int sigfillset(sigset_t *set);对信号集全置1。

int sigaddset (sigset_t *set, int signo);在特定的信号集当中,将特定的信号加进来。

int sigdelset(sigset_t *set, int signo);在特定的信号集当中,将特定的信号去掉。

int sigismember(const sigset_t *set, int signo);判断特定的一个信号,是否在该集合当中。

这一批接口,就是针对于位图结构天然设计好的各种各样的增删差改的操作。

1.3 sigprocmask:

sigprocmask: signal - 信号,process - 进程,mask - 掩码的意思。

  • 可以更改或者获取特定调用进程的信号屏蔽字。
#include <signal.h>
int sigprocmask(int how, const sigset_t *set, sigset_t *oset);
// 返回值:若成功则为0,若出错则为-1

可以理解成old或者是output,将老的信号屏蔽字返回出来,目的是为了将来的恢复需求。

这个函数第一个参数就决定了要做什么操作:

在这里插入图片描述

1.4 sigpending:

在这里插入图片描述

  • 获得当前进程的pending信号集。
  • 读取当前进程的末决信号集通过set参数传出,调用成功则返回0,出错则返回-1。

综上几个接口,我们用代码演示一下:

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <signal.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>using namespace std;void handler(int signo)
{// sleep(1);cout << "我是一个进程,刚刚获取了一个信号: " << signo << endl;// exit(1);
}static void showPending(sigset_t* pendings)
{for (int sig = 1; sig <= 31; sig++){// 检测这31个信号是否在这个集合里if (sigismember(pendings, sig)){cout << '1';}else{cout << '0';}}cout << endl;
}int cnt = 0;int main()
{// 3. pending收到信号很快就递达了,所以先block,这样就能看到pending表里的信号了// 屏蔽二号信号sigset_t bsig, obsig;sigemptyset(&bsig);sigemptyset(&obsig);// 3.1 添加2号信号到信号屏蔽字中sigaddset(&bsig, 2);// 3.2 设置用户级的信号屏蔽字到内核中,让当前进程屏蔽掉2号信号sigprocmask(SIG_SETMASK, &bsig, &obsig);// 2. signal将二号信号进行自定义捕捉signal(2, handler);// 1. 不断获取当前进程的pending信号集// 表示当前进程的所有pending信号sigset_t pendings;while (true){// 1.1 清空信号集sigemptyset(&pendings);// 1.2 获取当前进程(谁调用,获取谁)的pending信号集if (sigpending(&pendings) == 0){// 获取成功// 1.3 打印一下当前进程的pending信号集showPending(&pendings);}sleep(1);// 先跑十秒钟,再解除屏蔽cnt++;if (cnt == 10){cout << "解除对所有信号的block...." << endl;sigset_t sigs;sigemptyset(&sigs);sigaddset(&sigs, 2);// 只对2号解除屏蔽// sigprocmask(SIG_UNBLOCK, &sigs, nullptr);// 解除全部信号屏蔽sigprocmask(SIG_SETMASK, &obsig, nullptr);}}return 0;
}

演示结果:

在这里插入图片描述


2. 进程处理信号

2.1 内核页表和用户页表:

进程收到了信号该如何处理呢?

进程处理信号,不是立即处理的:

  • 而是在合适的时候处理的。

信号可能不是立即处理的,可能当前进程做着更重要的事情。 是在合适的时候处理的。

具体在什么时候处理呢?

  • 当当前进程从内核态,切换回用户态的时候,进行信号的检测与处理!

每一个进程都有一个内核空间,用于内核级页表的映射:
在这里插入图片描述
内核页表:

  • 负责3G到4G之间数据的映射。
  • 所有进程共享的,只有一份内核级页表。
  • 前提是你得有权利访问!

用户级页表:

  • 每一个进程,都有一份,而且大家的用户级页表都是不一样的!

2.2 内核态和用户态:

内核态vs用户态:

  • 内核态可以访问所有的代码和数据 —— 具备更高权限
  • 用户态只能访问自己的

OS在不在内存中被加载呢??在!

  • 无论进程怎么切换,我们都可以找到内核的代码和数据,前提是你只要能够有权力访问!

当前进程如何具备权力,访问这个内核页表,乃至访问内核数据呢?

  • 要进行身份切换。
  • 进程如果是用户态的 —— 只能访问用户级页表。
  • 进程如果是内核态的 —— 访问内核级和用户级的页表。
  • 进程也有用户态和内核态的差别。

我怎么知道我是用户态的还是内核态的呢?

  • CPU内部有对应的状态寄存器CR3,有比特位标识当前进程的状态:0是内核态3是用户态

普通用户的身份是无法访问到操作系统中的任何数据的。

补充:

  • 当我们想调用某些系统调用的时候,这些系统调用的代码,实际上在执行时,除了要跳转到目标函数之外,还要陷入内核就是通过计算机帮我们直接去执行某些寄存器操作,将CR3寄存器权限标志位由3 (用户态)改为0(内核态),操作系统当在进行身份认证的时候,发现是0就有权访问,否则就不能访问。
  • 当把操作系统的代码执行完,准备返回的时候,返回时CPU内的级别再由0被改成了3再返回代码处继续执行。

达成的共识:

  • 地址空间分为用户空间和内核空间,每个用户都有自己的私有页表,但共享所有的是内核页表。CPU内有寄存器用来识别标识用户身份的。

最终的认识:

  • 无论进程再怎么切换,3~4G的内核空间是完全一样的,所以任何进程经过身份切换都可以,变成内核态去执行操作系统的代码。
  • 内核态可以访问地址空间内的所有代码和数据,任意进程。
  • 用户态只能访问自己的0~3G的数据,更高的访问不了。

补充:

我们的程序,会无数次直接或者间接的访问系统级软硬件资源(管理者是OS),本质上,你并没有自己去操作这些软硬件资源,而是必须通过OS -> 无数次的陷入内核(1.切换身份 2.切换页表) -> 调用内核的代码 -> 完成访问的动作 -> 结果返回给用户(1.切换身份 2.切换页表) -> 得到结果。

例如:

while(1); -> 必须有自己的时间片 -> 时间片到了的时候 -> 内核态,更换内核级页表 -> 保护上下文,执行调度算法 -> 选择了新的进程 -> 恢复新进程的上下文 -> 用户态,更换成用户级页表 -> CPU执行的就是新进程的代码!

什么是陷入内核:

  • 在Linux中,"陷入内核"是指用户程序或进程进入内核空间执行的一种状态。当用户程序需要执行特权操作或需要访问受限资源时,例如打开文件、创建进程等,就会触发一个系统调用来请求内核的帮助。
  • 当一个用户程序调用系统调用时,CPU会从用户态切换到内核态,进入内核空间执行相应的内核代码。在内核态下,用户程序可以访问受限资源并执行特权操作。这种切换是通过将用户程序的上下文保存起来,并加载内核的上下文来实现的。
  • 一旦用户程序陷入内核,它会执行内核提供的相关功能,并等待内核完成请求的操作后返回结果。完成后,CPU会从内核态切换回用户态,并将结果返回给用户程序继续执行。
  • 通过将用户程序和内核区分开来,Linux实现了安全性和稳定性的目标。用户程序无法直接访问和修改内核的数据结构,这样可以避免用户程序对系统造成破坏。同时,内核提供了一套系统调用接口,使得用户程序能够通过请求内核来获取系统资源和执行特权操作。

2.3 信号检测过程:

进程的生命周期中,会有很多次机会去陷入内核(中断,陷阱,系统调用,异常…),一定会存在很多次的机会进行内核态返回用户态
在这里插入图片描述

检查信号一些列的过程:
在这里插入图片描述

open调用一定会陷入内核~

调用接口执行open的代码。

  • 从磁盘当中读取文件的属性,在内核当中创建该文件的sturct file结构。
  • 该文件匹配的inode,以及该文件所对应的路径信息全部都设置好。
  • 将文件的地址填到文件描述符表的下标里,然后将下标返回进而就可以继续执行了。

实际上操作系统,在准备返回之前(open继续向后执行之前),其实不是简单的返回了,而是返回之前先查,查进程的信号列表。

  • handler中如果执行的是自定义方法时:

为什么自定义方法只能用用户身份来执行?

  • 是为了自定义方法是用户写的(用户提供的),是为了防止用户写了一段恶意代码,而内核身份权力又很大。

为什么不在自定义方法调用结束时直接返回?

  • 如果直接返回了,open的返回值没有返回,也返回不了,所以不能直接由处理信号的逻辑直接跳转过去,是不允许的,也做不到。
  • 系统调用要正常返回,是要把状态各方面要做切换,数据要返回,用户层到用户层无法做到这个工作,包括再用户层无法知道当时在哪被切到内核态的,不知道上下文数据,也没办法恢复,而且严重不推荐。

处理完走到内核当中,在内核里面特定的系统调用,特定的系统返回,把代码寄存器状态等方面恢复出来,让它继续跑到当前进程的代码里继续执行。

  • handler中如果执行的是非自定义方法时:

pending和block都为0:

  • 那么是没有信号要处理,直接返回到调用出正常运行了。

pending和block都是1:

  • 操作系统做不了任何事情,无法被处理,无法被递达,操作系统照样返回。

pending为1,block是0:(默认)

  • 去找handler表,如果是默认,一般指向操作系统中的默认处理方法,一般都是终止这个进程。
  • 把这个代码不要在CPU上跑了,然后把代码全都释放掉。
  • 设置好之后保留PCB,设置僵尸状态,将PCB内的信号编号填充成收到的信号编号。
  • 此时退出码已设置,进程状态设置成Z状态,此时这个进程就退出了,也不需要返回了。

pending为1,block为0:(忽略)

  • 去找handler表,发现handler表是SIG_IGN,就是忽略。
  • 直接将该信号由1置为0,然后就处理完了这个信号,然后直接返回就好了。
  • 返回对应处完成对应处理。

2.3 - 1 便捷记忆图

在这里插入图片描述
中间交点一定要在横线以下。

2.4 sigaction:

这个函数除了能处理普通信号,实时信号也能处理。

在这里插入图片描述
sigaction函数可以读取和修改与指定信号相关联的处理动作。调用成功则返回0,出错则返回-1。

我们不考虑实时信号,所以有些字段我们不考虑:

在这里插入图片描述
基本使用:

#include <iostream>
#include <signal.h>
#include <unistd.h>using namespace std;void handler(int signo)
{cout << "获取到一个信号,信号编号是: " << signo << endl;sigset_t pending;// 永远都会正在处理2号信号while (true){cout << "." << endl;sigpending(&pending);for (int i = 1; i <= 31; i++){if (sigismember(&pending, i))cout << '1';elsecout << '0';}cout << endl;sleep(1);}
}int main()
{struct sigaction act, oact;act.sa_handler = handler;// act.sa_handler = SIG_IGN;// act.sa_handler = SIG_DFL;act.sa_flags = 0;sigemptyset(&act.sa_mask);// 三号信号拦截sigaddset(&act.sa_mask, 3);// 对二号信号的捕捉sigaction(2, &act, &oact);// sigaction的更大意义在于,当我们在做信号处理时// 操作系统不允许嵌套式的递归式的处理多个信号。while (true){cout << "main running" << endl;sleep(1);}return 0;
}

在这里插入图片描述
在这里插入图片描述

  • 当某个信号的处理函数被调用时,内核自动将当前信号加入进程的信号屏蔽字,当信号处理函数返回时自动恢复原来的信号屏蔽字。
  • 这样就保证了在处理某个信号时,如果这种信号再次产生,那么它会被阻塞到当前处理结束为止。
  • 如果在调用信号处理函数时,除了当前信号被自动屏蔽之外,还希望自动屏蔽另外一些信号,则用sa_mask字段说明这些需要额外屏蔽的信号,当信号处理函数返回时自动恢复原来的信号屏蔽字。

原因:

  • 通过将当前信号加入信号屏蔽字,内核确保在信号处理函数执行期间,同一信号不会再次中断进程。
  • 这种机制是必要的,因为信号处理函数是在异步上下文中执行的,即当信号发生时,处理函数会立即执行,而不管进程当前正在进行什么操作。
  • 如果不使用信号屏蔽字来屏蔽同一信号的再次中断,就可能导致信号处理函数被递归调用,而且多个信号处理函数同时执行,可能会引起不可预测的行为或系统崩溃。

一直处理某个信号,查看pending表:

在这里插入图片描述


3. volatile关键字

看下面一段代码:

#include <stdio.h>
#include <signal.h>// 保持内存的可见性,每次做检测必须从内存里拿
volatile int flags = 0;void handler(int signo)
{printf("更改flags: 0->1\n");flags = 1;
}int main()
{signal(2, handler);while (!flags);printf("进程是正常退出的!\n");return 0;
}

如果上述代码不带上volatile,则不同编译器会有不同的结果:

在这里插入图片描述
用新一点的编译器:

gcc test.c -o test -O2
  • 有的编译器发现在main执行流里发现没有对falgs做任何修改。
  • 高级别的编译器,会将这个flags值优化到寄存器里,从此往后再做while循环检测时候,只做一件事,从这个寄存器里做数据读取,所以这个寄存器里的值永远不会被修改了。
  • 编译器只能检测语法,不能检测逻辑。

volatile关键字,告诉编译器,不准对flags做任何优化,每次CPU计算的时候,拿内存中的数据,都必须在内存中拿!!
volatileconst可以同时修饰一个变量。


4. 子进程给父进程发信号

  • 子进程退出的时候,不是同学们想的那样,默默的退出(X状态)
  • 子进程退出的时候,自动给父进程发送SIGCHLD信号!!
#include <iostream>
#include <signal.h>
#include <unistd.h>using namespace std;void handler(int signo)
{cout << "子进程退出啦,我确实收到了信号: " << signo << " 我是: " << getpid() << endl;
}int main()
{signal(SIGCHLD, handler);pid_t id = fork();if (id == 0){while (true){cout << "我是子进程: " << getpid() << endl;sleep(1);}exit(0);}// parentwhile (true){cout << "我是父进程: " << getpid() << endl;sleep(1);}
}

在这里插入图片描述
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/110834.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

「Vue|网页开发|前端开发」02 从单页面到多页面网站:使用路由实现网站多个页面的展示和跳转

本文主要介绍如何使用路由控制来实现将一个单页面网站扩展成多页面网站&#xff0c;包括页面扩展的逻辑&#xff0c;vue的官方路由vue-router的基本用法以及扩展用法 文章目录 本系列前文传送门一、场景说明二、基本的页面扩展页面扩展是在扩什么创建新页面的代码&#xff0c;…

面试题-React(七):React组件通信

在React开发中&#xff0c;组件通信是一个核心概念&#xff0c;它使得不同组件能够协同工作&#xff0c;实现更复杂的交互和数据传递。常见的组件通信方式&#xff1a;父传子和子传父 一、父传子通信方式 父组件向子组件传递数据是React中最常见的一种通信方式。这种方式适用…

SpringBoot异步方法支持注解@Async应用

SpringBoot异步方法支持注解Async应用 1.为什么需要异步方法&#xff1f; 合理使用异步方法可以有效的提高执行效率 同步执行(同在一个线程中): 异步执行(开启额外线程来执行): 2.SpringBoot中的异步方法支持 在SpringBoot中并不需要我们自己去创建维护线程或者线程池来…

C#_进程单例模式.秒懂Mutex

什么是Mutex? 可以定义调用线程是否具有互斥性&#xff0c;程序创建者拥有控制权&#xff0c;相反只能引用程序。 参数1&#xff1a;如果是程序创建者&#xff0c;就获得控制权。 参数2&#xff1a;名称&#xff0c;可使用GUID生成。 参数3&#xff1a;out 返回值&#xf…

Linux下套接字TCP实现网络通信

Linux下套接字TCP实现网络通信 文章目录 Linux下套接字TCP实现网络通信1.引言2.具体实现2.1接口介绍1.socket()2.bind()3.listen()4.accept()5.connect() 2.2 服务器端server.hpp2.3服务端server.cc2.4客户端client.cc 1.引言 ​ 套接字(Socket)是计算机网络中实现网络通信的一…

1688API技术解析,实现获得1688商品详情

要实现获得1688商品详情&#xff0c;你需要使用1688 API。1688 API是阿里巴巴旗下的开放平台&#xff0c;它提供了一套丰富的接口&#xff0c;可以让开发者通过编程的方式获取到1688网站上的商品信息。 首先&#xff0c;你需要在阿里开放平台注册一个账号&#xff0c;并创建一…

【NLP的python库(01/4) 】: NLTK

一、说明 NLTK是一个复杂的库。自 2009 年以来不断发展&#xff0c;它支持所有经典的 NLP 任务&#xff0c;从标记化、词干提取、词性标记&#xff0c;包括语义索引和依赖关系解析。它还具有一组丰富的附加功能&#xff0c;例如内置语料库&#xff0c;NLP任务的不同模型以及与S…

深眸科技创新赋能视觉应用产品,以AI+机器视觉解决行业应用难题

随着工业4.0时代的加速到来&#xff0c;我国工业领域对于机器视觉技术引导的工业自动化和智能化需求持续上涨&#xff0c;国内机器视觉行业进入快速发展黄金期&#xff0c;但需求广泛出现同时也对机器视觉产品的检测能力提出了更高的要求。 传统机器视觉由人工分析图像特征&am…

JAVA JNA 调用C接口的三种方式

文章目录 1. 准备一个共享库文件2. JNA姿势1—继承Library接口3. JNA姿势2—直接NativeLibrary.getInstance3. JNA姿势3—Native方法 1. 准备一个共享库文件 test.c #include <stdio.h> int test(char *input){printf("input:%s\n",input);return 0; }libtes…

从Matrix-ResourceCanary看内存泄漏监控

作者&#xff1a;小海编码日记 不同于LeakCanary&#xff0c;在Matrix中&#xff0c;主要是通过Resource Canary来监控内存泄漏问题的&#xff0c;且监听的泄漏对象只支持Activity&#xff0c;官方说明如下&#xff1a; 结合分析LeakCanary的经验可知&#xff0c;要实现Activit…

fastadmin iis伪静态应用入口文件index.php

<?xml version"1.0" encoding"UTF-8"?> <configuration><system.webServer><rewrite><rules><rule name"OrgPage" stopProcessing"true"><match url"^(.*)$" /><conditions…

【python知识】用 Tkinter实现“剪刀-石头-布”和“弹球游戏 ”

一、提要 Tkinter是一个Python内置模块&#xff0c;它提供了一个简单易用的界面来创建GUI。 在实现一些动态的画面、如游戏还是需要一些创新性思维的。在本文中&#xff0c;我们将使用 Tkinter 探索 Python GUI 编程。我们将介绍 Tkinter 的基础知识&#xff0c;并演示如何使用…

Element Plus 日期选择器的使用和属性

element plus 日期选择器如果如果没有进行处理 他会返回原有的属性值data格式 如果想要获取选中的日期时间就需要通过以下的代码来实现选中的值 format"YYYY/MM/DD" value-format"YYYY-MM-DD" <el-date-pickerv-model"formInline.date" type&…

「MySQL-05」MySQL Workbench的下载和使用

目录 一、MySQL workbench的下载和安装 1. MySQL workbench介绍 2. 到MySQL官网下载mysql workbench 3. 安装workbench 二、创建能远程登录的用户并授权 1. 创建用户oj_client 2. 创建oj数据库 3. 给用户授权 4. 在Linux上登录用户oj_client检查其是否能操作oj数据库 三、使用…

gerrit 如何提交进行review

前言 本文主要介绍如何使用gerrit进行review。 下述所有流程都是参考&#xff1a; https://gerrit-review.googlesource.com/Documentation/intro-gerrit-walkthrough.html 先给一个commit后但是还没有push上去的一个办法&#xff1a; git reset --hard HEAD^可以多次reset.…

DC/DC开关电源学习笔记(一)开关电源技术概述

&#xff08;一&#xff09;开关电源技术概述 1.什么是开关电源&#xff1f;2.开关电源技术概述2.1 小型化、薄型化、轻量化、高频化2.2 高可靠性2.3 低噪声2.4 采用计算机辅助设计和控制 1.什么是开关电源&#xff1f; 开关模式电源&#xff08;Switch Mode Power Supply&…

opencv的使用(Ubuntu linux环境,AS jni,AS java)

最近要完成一个功能&#xff0c;就是把四个视频合成左右上下分布的一个视频。尝试很多方法&#xff0c;最终使用opencv来实现该功能。&#xff08;通过opencv实现的视频好像没有声音。&#xff09;研究的步骤&#xff0c;首先在Ubuntu环境测试&#xff0c;该功能是否实现。然后…

【Luniux】解决Ubuntu外接显示器不显示的问题

Luniux】解决Ubuntu外接显示器不显示的问题 文章目录 Luniux】解决Ubuntu外接显示器不显示的问题1. 检查nvidia显卡驱动是否正常2. 更新驱动3. 检查显示器是否能检测到Reference 1. 检查nvidia显卡驱动是否正常 使用命令行 nvidia-smi来检查显卡驱动是否正常&#xff0c;如果…

3、监测数据采集物联网应用开发步骤(3)

监测数据采集物联网应用开发步骤(2) 系统整体结构搭建 新建项目 输入项目名称&#xff1a;MonitorData 所谓兵马未动粮草先行&#xff0c;按下图创建好对应的模块备用&#xff1a; com.plugins 业务插件模块 com.zxy.adminlog 日志或文本文…

《C和指针》笔记12: 存储类型(自动变量、静态变量和寄存器变量)

文章目录 1. 自动变量&#xff08;auto&#xff09;1.1 自动变量的初始化 2. 静态变量&#xff08;static&#xff09;2.1 静态变量的初始化 3. 寄存器变量&#xff08;register&#xff09; 1. 自动变量&#xff08;auto&#xff09; 在代码块内部声明的变量的缺省存储类型是…