当前位置: 首页 > news >正文

【Linux】进程控制详解

目录

前言

进程创建

认识fork

写时拷贝

 再谈fork

 进程终止

进程退出码 

用代码来终止进程

常见的进程终止的方式

exit 

_exit

进程等待

进程等待的必要性

进程等待的方式

wait 

waitpid

详解status参数

详解option参数


前言

本文适合有一点基础的人看的,否则的话有点难以理解,如果有问题,可以在评论区将你的问题打出来,我会一一解答的。

关于本文可以先去看看上篇的进程地址空间可以更好的理解这里的内容

进程创建

认识fork

fork函数是从已存在进程中创建一个新进程,新进程为子进程,原进程为父进程

 

返回值

  • 创建成功,在父进程中返回子进程的PID,并在子进程中返回0
  • 创建失败,父进程返回-1,子进程不返回,并设置错误码

 当进程中调用了fork,那么内核就会做以下事情

  • 分配新的内存块和内核数据结构给子进程
  • 将父进程部分数据结构内容拷贝至子进程
  • 添加子进程到系统进程列表中
  • fork返回,开始调度器调度
#include <stdio.h>
#include <unistd.h>int main()
{printf("这是父进程, pid:%d\n", getpid());pid_t id = fork();if(id < 0){printf("创建子进程失败!!!");return 1;}else if(id == 0){//子进程while(1){printf("我是子进程, pid:%d, ppid:%d\n", getpid(), getppid());sleep(1);}}else {//父进程while(1){printf("我是父进程, pid:%d, ppid:%d\n", getpid(), getppid());sleep(1);}}return 0;
}

 

fork之前只有父进程在执行,fork之后,父子进程分别执行。注:fork之后,父子进程谁先执行是由调度器决定的。

写时拷贝

从上面的内容我们知道,子进程也有自己的数据和代码,但是一般而言,我们的代码和数据没有在磁盘加载的过程,也就是说,子进程没有自己的代码和数据,所以子进程和父进程的共享代码和数据。对于代码而言,都是不可写的只能读取,所以父子共享没有问题;但对于数据而言可能会被修改,所以父子进程的数据必须分离。

那么数据分离是创建进程的时候就直接拷贝分离,还是修改数据时才拷贝分离呢?

答案肯定是需要修改数据时才拷贝分离

因为当你创建子进程时,可能其中的数据根本不会用到,即使用到了,也只是读取,那么这样的话在创建进程时就将数据直接拷贝分离是非常浪费空间的,所以操作系统选择了,当你需要对数据进行修改时才拷贝一份,这样父子进程的数据进行了分离且互不影响。而这个技术就叫做写时拷贝


看一下下面代码的结果

从结果可以看出,编译器编译程序的时候,尚且知道节省空间何况是操作系统呢


那么操作系统为何要选择写时拷贝技术对父子进程进行分离呢?

  • 当用到数据时,再将数据进行分离是高效使用内存的一种表现
  • 操作系统在代码执行前无法预知哪些空间会被访问,也就无法对数据进行分离

 再谈fork

在重新回到认识fork的图中,fork之后,父子进程代码共享,是after共享,还是所有的共享?

答案是:所有的啦

那么子进程为什么不会从before之后在执行代码,而是直接从after开始执行代码呢?也就是说子进程是怎么知道代码执行到哪里了呢?

答案是:通过进程的上下文数据

我们的代码形成汇编之后,会有很多行代码,并且每行代码加载到内存之后都有对应的地址。但是进程由于一些原因随时可能会被中断(还没执行完),那么当进程下次回来时,还必须从当前中断的位置继续向后执行,这时就必须要求CPU随时记录下当前进程执行的位置,所以CPU内会有对应的寄存器(EIP,也叫PC指针)用来记录当前进程执行的位置,寄存器在CPU内只有一份,但是寄存器内的数据可以有多份。而这个寄存器中的数据就叫做进程的上下文数据,是数据就会发生写时拷贝。虽然父子进程各自会调度,各自会修改EIP,但是子进程已经认为自己EIP起始值就是fork之后的代码。

fork的常规用法

  • 父子进程同时执行不同的代码段。例如父进程等待客户端请求,子进程处理请求。
  • 一个进程要执行一个不同的程序。例如子进程从fork返回后调用exec函数。

fork调用失败的原因

  1. 系统中有太多的进程
  2. 用户的进程数超过了限制 

 进程终止

进程退出码 

进程退出的三种情况

  1. 代码运行完,结果正确
  2. 代码运行完,结果错误
  3. 代码没有运行完,程序崩溃了

当我们写代码时总是在main函数的最后return 0,那么main函数返回的意义是什么呢,return 0的含义又是什么,为什么返回的总是0呢?

下面就让我来一一解答这些问题吧

其实main函数返回的并不总是0,也可以是其它数字,而main函数返回的这个值其实是叫做进程的退出码。例如下面代码

int main()
{printf("hello world");return 10;
}

echo $?:用于获取最近一个进程执行完毕的退出码

我们返回的这个10的作用其实是用来标识代码执行完毕,结果是正确的,所以我们return 0的含义是:0表示代码执行完毕,结果正确;非0表示的是运行的结果不正确。非0值有无数个,不同的非0值可以标识不同的错误原因。所以我们也就可以根据这个返回值来判定代码运行完后的正确性。

因此我们也就明白了main函数返回的意义就是返回给上一级进程用来评判该进程的执行结果,如果结果不正确方便我们定位错误的原因细节。

在Linux下就定义了134个退出码,分别标识了不同的错误原因。

可以用strerror函数将这些退出码对应的错误原因打印出来

#include <stdio.h>
#include <string.h>int main()
{for(int number = 0; number <= 134; number++){printf("%d:%s\n", number, strerror(number));}return 0;
}

当然我们也可以使用这些退出码和含义,但是如果我们想自己定义也可以自己设计一套退出方案来。

上面的分析都是针对进程退出的前两种情况,那么第三种情况呢?

 程序崩溃而导致的退出其退出码没有任何意义,因为崩溃退出没有执行对应的rreturn语句。

#include <stdio.h>int main()
{int* ptr= NULL;*ptr = 10;//野指针printf("hello world\n");return 0;
}

用代码来终止进程

常见的进程终止的方式

  1. 在main函数内使用return语句
  2. 调用exit函数,可以在代码的任何地方调用都表示终止进程。
  3. 调用系统接口_exit

exit 

#include <stdio.h>
#include <stdlib.h>void Func()
{//打印1-10for(int i = 1; i <= 10; i++){printf("%d ", i);}printf("\n");exit(22);
}int main()
{printf("hello world\n");printf("hello world\n");Func();printf("hello world\n");exit(11);printf("hello world\n");return 0;
}

_exit

 

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>int main()
{printf("hello world\n");sleep(3);exit(22);
}

上面这个代码是先打印hello world还是先sleep呢?

很显然是先打印hello world再sleep,那么我们把\n去掉呢?

很显然是先sleep再打印hello world。\n其实就是将我们打印的内容从缓冲区刷新到我们的屏幕上。

那么我们换用系统接口_exit试一试

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>int main()
{printf("hello world");sleep(3);_exit(22);
}

可以看出它执行了sleep但是为什么最后没有将hello world给我们打印出来呢?

因为exit()函数是C标准库给我们提供的库函数,它最后也会调用_exit,但在调用_exit之前还做了其它工作

  1. 执行用户通过atexit或者on_exit定义的清理函数
  2. 关闭所有打开的流,所有的缓冲区数据均被写入
  3. 调用_exit

而其中的缓冲区其实是C标准库在给我们维护的而不是操作系统,它在操作系统的上层,所以_exit最后不会将hello world刷新出来

进程等待

进程等待的必要性

当我们用fork创建子进程时,子进程退出,父进程如果不管子进程,就可能造成僵尸进程的问题,进而造成内存泄漏。另外,进程一旦变成僵尸状态,就连kill -9号命令也拿它没办法,因为谁也没办法杀死一个已经死去的进程。最后由于子进程变成僵尸,那么父进程指派给子进程完成的任务我们也无法知道是否已经完成,所以我们必须让父进程通过进程等待的方式回收子进程的资源,获取子进程的退出信息。

进程等待的方式

wait 

 没调wait之前

#include <stdio.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <stdlib.h>
#include <unistd.h>int main()
{pid_t id = fork();if(id < 0){perror("fork");exit(0);//创建子进程失败}else if(id == 0){//子进程int cnt = 5;while(cnt){printf("cnt:%d,我是子进程,pid:%d,ppid:%d\n", cnt, getpid(),getppid());sleep(1);cnt--;}exit(1);//子进程退出}else {//父进程while(1){printf("我是父进程,pid:%d,ppid:%d\n", getpid(), getppid());sleep(1);}}return 0;
}

 

5秒之后子进程变僵尸,那么我们调用wait看看现象

返回值:成功则返回被等待进程的pid,失败则返回-1

参数:输出型参数,获取子进程退出状态,不关心其状态则可以设置为NULL

#include <stdio.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <stdlib.h>
#include <unistd.h>int main()
{pid_t id = fork();if(id < 0){perror("fork");exit(0);//创建子进程失败}else if(id == 0){//子进程int cnt = 5;while(cnt){printf("cnt:%d,我是子进程,pid:%d,ppid:%d\n", cnt, getpid(),getppid());sleep(1);cnt--;}exit(1);//子进程退出}else {//父进程printf("我是父进程,pid:%d,ppid:%d\n", getpid(), getppid());sleep(7);pid_t ret = wait(NULL);if(ret > 0){printf("等待子进程成功,ret:%d\n",ret);}while(1){printf("我是父进程,pid:%d,ppid:%d\n", getpid(), getppid());sleep(1);}}return 0;
}

 

从结果很容易看出,5秒后子进程进入僵尸状态,7秒后父进程将子进程回收掉了。

waitpid

 

返回值:正常返回子进程的进程ID,如果设置了选项WNOHANG并且子进程均已退出,则返回0;如果调用失败则返回-1,并且error会被设置相应的错误码

参数

  • pid==-1,等待任意一个子进程;pid > 0,等待其进程ID与pid相等的子进程。
  • 输出型参数,WIFEXITED(status):若为正常终止子进程返回的状态则为真。(可以用来查看进程是否是正常退出),WEXITSTATUS(status): 若WIFEXITED非零,提取子进程的退出码。(查看进程的退出码)。若不关心status可以设置为NULL和wait一样。
  • WNOHANG: 若pid指定的子进程没有结束则函数返回0,不予以等待;若正常结束则返回该子进程的ID,这种方式也叫做非阻塞等待。默认为0表示阻塞等待。

 将上面代码中的wait换成waipid即可,现象都是一样的

详解status参数

wait和waitpid都有一个status参数,该参数是一个输出型参数,由操作系统填充,如果设置为NULL则表示不关心子进程的退出状态信息,否则操作系统会根据该参数,将子进程的退出信息反馈给父进程。

那么我们用代码实际验证一下status吧

#include <stdio.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <stdlib.h>
#include <unistd.h>int main()
{pid_t id = fork();if(id < 0){perror("fork");exit(0);//创建子进程失败}else if(id == 0){//子进程int cnt = 5;while(cnt){printf("cnt:%d,我是子进程,pid:%d,ppid:%d\n", cnt, getpid(),getppid());sleep(1);cnt--;}exit(100);//子进程退出}else {//父进程printf("我是父进程,pid:%d,ppid:%d\n", getpid(), getppid());int status = 0;pid_t ret = waitpid(id, &status, 0);//阻塞式等待if(ret > 0){printf("等待子进程成功,ret:%d,子进程的退出码:%d\n",ret, status);}}return 0;
}

上面的代码中子进程的退出码明明是100,为什么打印出来的却是比100大的多的25600呢?

原因是:status并不是按照整数来整体使用的,而是按照比特位的方式将32个比特位进行划分,这里我们只分析低16位

所以我们要获取子进程的退出码就要右移8位并与上0xff

当我们的进程(程序)异常退出或者崩溃,本质上是操作系统杀掉了我们的进程,那么操作系统是通过什么方式来杀掉我们的进程的呢?

答:通过向进程发送信号的方式来杀掉进程。 

一共有以下信号 

 

将上面代码进行如下改造

#include <stdio.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <stdlib.h>
#include <unistd.h>int main()
{pid_t id = fork();if(id < 0){perror("fork");exit(0);//创建子进程失败}else if(id == 0){//子进程int cnt = 5;while(cnt){printf("cnt:%d,我是子进程,pid:%d,ppid:%d\n", cnt, getpid(),getppid());sleep(1);cnt--;}exit(100);//子进程退出}else {//父进程printf("我是父进程,pid:%d,ppid:%d\n", getpid(), getppid());int status = 0;pid_t ret = waitpid(id, &status, 0);//阻塞式等待if(ret > 0){printf("等待子进程成功,ret:%d,子进程收到的信号编号:%d,子进程的退出码:%d\n",ret, status & 0x7f, (status >> 8)&0xff);}}return 0;
}

0表示正常跑完,100表示结果正确

在代码中加个除0操作

8表示收到了8号信号,代码是异常退出,那么我们的退出码也就无意义了。这里也验证了上面退出码的内容

在代码中加个野指针

11表示收到了11号信号,也就是段错误。使用kill -9杀掉进程也是类似的,这里就不在演示了。

所以程序(进程)异常不光是内部代码有问题,也可能是外力因素将进程直接干掉了,那么子进程是否跑完,我们也不确定。

知道了上面这些那么让我们来思考一下下面的问题

1、父进程通过wait/waitpid可以拿到子进程的退出结果,那么为什么要用wait/waitpid函数,直接全局变量不行吗?

code为全局变量

很显然是不行的,因为进程具有独立性,全局变量也是数据,那么是数据就要发生写时拷贝,父进程也就无法拿到。

2、既然信号具有独立性,子进程退出的信息也是子进程的数据,那么为什么父进程调用了wait/waitpid就能拿到子进程退出的信息呢?

子进程退出会变成僵尸进程,但是即使变成了僵尸进程也会保留该进程的PCB(task_struct)信息(就跟人死亡后身上会保留死亡的原因),task_struct里面就保留了任何进程退出时的退出信息,所以wait和waitpid本质上就是读取了子进程的task_struct里面相关信息。

3、wait/waitpid有这个权利从PCB中拿到相关信息吗?

肯定有这个权利啊,它们可是系统调用的接口,娘胎里自带的就有这个权利。

我们在获取子进程的退出码时也可以不用位运算,可以使用宏也就是下面的这种写法。

#include <stdio.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <stdlib.h>
#include <unistd.h>int main()
{pid_t id = fork();if(id < 0){perror("fork");exit(0);//创建子进程失败}else if(id == 0){//子进程int cnt = 5;while(cnt){printf("cnt:%d,我是子进程,pid:%d,ppid:%d\n", cnt, getpid(),getppid());sleep(1);cnt--;int* p =NULL;*p = 100;}exit(100);//子进程退出}else {//父进程printf("我是父进程,pid:%d,ppid:%d\n", getpid(), getppid());int status = 0;pid_t ret = waitpid(id, &status, 0);//阻塞式等待if(ret > 0){if(WIFEXITED(status)){printf("子进程执行完毕,子进程的退出码:%d\n", WEXITSTATUS(status));}else{printf("子进程异常退出:%d,退出信号为:%d\n", WIFEXITED(status), WTERMSIG(status));}}return 0;
}

详解option参数

 option为0,默认是阻塞式等待;WNOHANG选项,代表父进程非阻塞式等待

WNOHANG其实是一个宏定义

非阻塞式等待的意思是:父进程通过调用waitpid来进行等待,如果子进程没有退出,则立马返回。

举个例子:你家冰箱坏了,你要打电话叫师傅上你家来维修,当你打通了电话,可是师傅说没有空这时你就立马挂断了电话去处理自己的事情了;过了一段时间,你再次去拨打师傅的电话,可是师傅还在忙,你又立马挂断了电话去处理别的事情了;又过了一段时间,你又拨通了师傅的电话,师傅这时有空了,说立马上门维修。在这个例子中你每次拨打电话的过程其实就是非阻塞调用,非阻塞调用采用的是轮询检测的方案。而如果你在第一次拨打师傅电话时,你一直不挂断电话,等到师傅有空了你才挂断电话,而这就是阻塞等待。

阻塞的本质其实是进程阻塞在系统调用的内部 

废话不多说上代码演示

#include <iostream>
#include <vector>
#include <stdio.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <stdlib.h>
#include <unistd.h>typedef void (*handler_t)(); //函数指针类型std::vector<handler_t> handlers; //函数指针数组void fun_one()
{int a = 10;int b = 20;printf("这是一个加法: %d + %d = %d\n", a, b, a + b);
}
void fun_two()
{int a = 10;int b = 20;printf("这是一个减法: %d - %d = %d\n", b, a, b - a);
}// 设置对应的方法回调
void Load()
{handlers.push_back(fun_one);handlers.push_back(fun_two);
}int main()
{pid_t id = fork();if(id < 0){perror("fork");exit(0);//创建子进程失败}else if(id == 0){//子进程int cnt = 5;while(cnt){printf("cnt:%d,我是子进程,pid:%d,ppid:%d\n", cnt--, getpid(),getppid());sleep(1);}exit(100);//子进程退出}else {//父进程int quit = 0;while(!quit){int status = 0;pid_t res = waitpid(-1, 0, WNOHANG);if(res > 0){//等待成功并且子进程退出printf("等待子进程退出成功,退出码:%d\n", WEXITSTATUS(status));quit = 1;}else if(res == 0){//等待成功并且子进程还未退出printf("子进程还在运行,暂时还没有退出,父进程在等等,先处理其它事情吧!!\n");if(handlers.empty()) Load();for(auto iter : handlers){//执行其它任务iter();}}else{//等待失败printf("wait失败\n");quit = 1;}sleep(1);}}return 0;
}

 


今天的分享就到这里,如果内容有错的话,还望指出谢谢!!!

相关文章:

【Linux】进程控制详解

目录 前言 进程创建 认识fork 写时拷贝 再谈fork 进程终止 进程退出码 用代码来终止进程 常见的进程终止的方式 exit _exit 进程等待 进程等待的必要性 进程等待的方式 wait waitpid 详解status参数 详解option参数 前言 本文适合有一点基础的人看的&#…...

Mysql 高性能的sql优化方案和建议

优化MySQL的性能是一项复杂而关键的任务&#xff0c;它可以通过多种方式来实现。下面是一些SQL优化的方案和建议&#xff1a; 索引优化&#xff1a; 确保经常查询的列都有索引。但不要过度索引&#xff0c;因为它可能会增加写入操作的开销。使用组合索引来覆盖多个查询条件。…...

鸿蒙实战开发:【实现应用悬浮窗】

如果你要做的是系统级别的悬浮窗&#xff0c;就需要判断是否具备悬浮窗权限。然而这又不是一个标准的动态权限&#xff0c;你需要兼容各种奇葩机型的悬浮窗权限判断。 fun checkPermission(context: Context): Boolean if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M)…...

应用开发:python解析斗鱼弹幕

解决问题 互动弹幕&#xff0c;关注提问 &#xff0c;ai回答 技术 python playwright 调用接口 https://github.com/broven/DouYudanmu/blob/master/douyu.py 演示 放弃 这个根本不是研究方向 定位错误 你浪费下午时间&#xff0c;定位错误 这个跟本不是你的方向。 4个小时看斗…...

【面试经典150 | 动态规划】交错字符串

文章目录 写在前面Tag题目来源解题思路方法一&#xff1a;动态规划 写在最后 写在前面 本专栏专注于分析与讲解【面试经典150】算法&#xff0c;两到三天更新一篇文章&#xff0c;欢迎催更…… 专栏内容以分析题目为主&#xff0c;并附带一些对于本题涉及到的数据结构等内容进行…...

设计模式(17):中介者模式

核心&#xff1a; 如果一个系统中对象之间的联系呈现网状结构&#xff0c;对象之间存在大量多对多关系&#xff0c;导致关系及其复杂&#xff0c;这些对象称为“同事对象”。我们可以引入一个中介者对象&#xff0c;使各个同事对象只跟中介者对象打交道&#xff0c;将复杂的网…...

echart 折线图或散点图当横坐标为小数位时,若想显示整数该如何处理?

如图当前是这样的&#xff1a; 横坐标刻度目前是小数位&#xff0c;如果直接将小数位取整则会失去精度&#xff0c;所以我们要做的是刻度即是整数&#xff0c;又能显示小数位对应的数值&#xff1b; 思路就是直接手动设置刻度&#xff1a;设置xAxis的min,max,splitNumber,同时不…...

一套C#自主版权+应用案例的手麻系统源码

手术麻醉信息管理系统源码&#xff0c;自主版权应用案例的手麻系统源码 手术麻醉信息管理系统包含了患者从预约申请手术到术前、术中、术后的流程控制。手术麻醉信息管理系统主要是由监护设备数据采集子系统和麻醉临床系统两个子部分组成。包括从手术申请到手术分配&#xff0c…...

31.2k star, 免费开源的白板绘图工具 tldraw

31.2k star, 免费开源的白板绘图工具 tldraw 分类 开源分享 项目名: tldraw -- 无限画布白板 Github 开源地址&#xff1a; https://github.com/tldraw/tldraw 在线测试地址&#xff1a; tldraw 文档地址&#xff1a; tldraw SDK tldraw 是一款开源免费的无限画布白板&…...

Redis开源协议调整,我们怎么办?

2024年3月20日, Redis官方宣布&#xff0c;从 Redis 7.4版本开始&#xff0c;Redis将获得源可用许可证 ( RSALv2 ) 和服务器端公共许可证 ( SSPLv1 ) 的双重许可&#xff0c;时间点恰逢刚刚完成最新一轮融资&#xff0c;宣布的时机耐人寻味。 Redis协议调整&#xff0c;对云计算…...

干了三年外包。。。忘了什么是CICD。。。

干了三年外包。。。忘了什么是CICD。。。 CI/CD(持续集成与持续交付) 是一种软件开发实践&#xff0c;它可以帮助我们更快地交付高质量的软件产品。CI/CD的核心思想是将软件开发过程中的各个阶段自动化&#xff0c;从而减少人工干预&#xff0c;提高开发效率和产品质量。本文将…...

【LeetCode】454. 四数相加 II

目录 题目 思路 代码 题目 题目链接&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 给你四个整数数组 nums1、nums2、nums3 和 nums4 &#xff0c;数组长度都是 n &#xff0c;请你计算有多少个元组 (i, j, k, l) 能满足&#xff1a; 0 < i, j, k, l < nnums1…...

搜索(DFS BFS)

DFS 常规DFS: 二叉树前序,中序&#xff0c;后序遍历-CSDN博客 void postorderTraversal(root)初始化一个空列表 arrfind访问总树(root,arr)return arrvoid find(temp, arr)if temp 为空return // 调用顺序由前中后序决定find递归访问左子树find递归访问右子树arr加入当前节点…...

koc和kol是什么意思?

一、koc和kol是什么意思&#xff1f; koc和kol是专业术语。KOC是关键意见消费者的意思&#xff0c;是Key Opinion Consumer的缩写&#xff1b;KOL是关键意见领袖的意思&#xff0c;是Key Opinion Leader的缩写。 1、关键意见领袖kol “关键意见领袖”通俗地讲是达人。这些人…...

基于vscode Arduino插件开发Arduino项目

基于vscode Arduino插件开发arduino项目 插件配置问题记录1. 指定编译输出文件夹2. 编译下载时不输出详细信息3. 输出端口信息乱码4. 通过串口输出中文&#xff0c;vscode对应的串口助手上会显示乱码&#xff08;未解决&#xff09; 插件配置 环境&#xff1a;Arduino插件版本…...

AI 驱动强大是视频转换处理软件

由 AI 驱动的视频工具包。 增强、转换、录制和编辑视频AI 驱动的顶级视频工具包。 不论是老旧、低质、噪声或模糊的影片/图像&#xff0c;都能升级至 4K&#xff0c;稳定抖动的影片&#xff0c;提升帧率至 120/240fps&#xff0c;并能以全面 GPU 加速进行转换、压缩、录制和编辑…...

Python+requests+Pytest+logging+allure+pymysql框架详解

一、框架目录结构 1)tools目录用来放公共方法存储,如发送接口以及读取测试数据的方法,响应断言 数据库断言 前置sql等方法;2)datas目录用例存储接口用例的测试数据,我是用excel来存储的数据,文件数据 图片数据等;3)testcases目录用来存放测试用例,一个python文件对应…...

菜鸟笔记-Numpy函数-full/random.randint/random.choice

full函数 numpy.full 是 NumPy 库中的一个函数&#xff0c;它用于创建一个具有指定形状、数据类型和填充值的数组。此函数非常有用&#xff0c;因为它允许你快速生成一个具有相同值的数组&#xff0c;而无需手动设置每个元素。 1函数介绍 numpy.full(shape, fill_value, dty…...

蓝桥杯每日一题:牛的学术圈I(二分,双指针)

由于对计算机科学的热爱&#xff0c;以及有朝一日成为 「Bessie 博士」的诱惑&#xff0c;奶牛 Bessie 开始攻读计算机科学博士学位。 经过一段时间的学术研究&#xff0c;她已经发表了 N篇论文&#xff0c;并且她的第 i 篇论文得到了来自其他研究文献的 ci次引用。 Bessie 听…...

fping命令

fping是一个用于网络扫描的工具&#xff0c;它可以在 Linux 系统上使用。fping可以发送 ICMP ECHO_REQUEST&#xff08;即 ping&#xff09;数据包到指定的网络地址范围&#xff0c;并等待响应。通过这种方式&#xff0c;fping可以用来检测哪些 IP 地址是活跃的。 可以测试多个…...

奇富科技推出新一代全自研智能语音模型,打破沟通壁垒

“您好&#xff01;请问是李先生噻&#xff1f;” 李先生刚接起电话&#xff0c;就被这熟悉的乡音逗乐了。这不是他所预料的常规客服&#xff0c;而是奇富科技新一代全自研智能语音模型——QI语精灵。这款模型不仅能用方言与人自然交流&#xff0c;还能在智能营销、贷后提醒、风…...

穿越代码之海:探寻结构体深层逻辑,展望未来应用新天地

欢迎来到白刘的领域 Miracle_86.-CSDN博客 系列专栏 C语言知识 先赞后看&#xff0c;已成习惯 创作不易&#xff0c;多多支持&#xff01; 结构体作为一种数据结构&#xff0c;其定义和特点决定了它在各种应用中的广泛适用性。随着科技的进步和新兴行业的不断涌现&#xf…...

layui框架实战案例(26):layui-carousel轮播组件添加多个Echarts图标的效果

在Layui中&#xff0c;使用layui-carousel轮播组件嵌套Echarts图表来实现多个图表的展示。 css层叠样式表 调整轮播图背景色为白色&#xff1b;调整当个Echarts图表显示loading…状态&#xff1b;同一个DIV轮播项目添加多个Echarts的 .layui-carousel {background-color: #f…...

Unity开发一个FPS游戏之三

在前面的两篇博客中&#xff0c;我已实现了一个FPS游戏的大部分功能&#xff0c;包括了第一人称的主角运动控制&#xff0c;武器射击以及敌人的智能行为。这里我将继续完善这个游戏&#xff0c;包括以下几个方面&#xff1a; 增加一个真实的游戏场景&#xff0c;模拟一个废弃的…...

NIUSHOP完美运营版商城 虚拟商品全功能商城 全能商城小程序 智慧商城系统 全品类百货商城

完美运营版商城/拼团/团购/秒杀/积分/砍价/实物商品/虚拟商品等全功能商城 干干净净 没有一丝多余收据 还没过手其他站 还没乱七八走的广告和后门 后台可以自由拖曳修改前端UI页面 还支持虚拟商品自动发货等功能 挺不错的一套源码 前端UNIAPP 后端PHP 一键部署版本 源码免费…...

vue2开发好还是vue3开发好vue3.0开发路线

Vue 2和Vue 3都是流行的前端框架&#xff0c;它们各自有一些特点和优势。选择Vue 2还是Vue 3进行开发&#xff0c;主要取决于你的项目需求、团队的技术栈、以及对新特性的需求等因素。以下是一些关于Vue 2和Vue 3的比较&#xff0c;帮助你做出决策&#xff1a; Vue 2&#xff1…...

爬虫 新闻网站 并存储到CSV文件 以红网为例 V2.0 (控制台版)升级自定义查询关键词、时间段,详细注释

爬虫&#xff1a;红网网站&#xff0c; 获取指定关键词与指定时间范围内的新闻&#xff0c;并存储到CSV文件 V2.0&#xff08;控制台版&#xff09; 爬取目的&#xff1a;为了获取某一地区更全面的在红网已发布的宣传新闻稿&#xff0c;同时也让自己的工作更便捷 对比V1.0升级的…...

JavaSE-11笔记【多线程2(+2024新)】

文章目录 6.线程安全6.1 线程安全问题6.2 线程同步机制6.3 关于线程同步的面试题6.3.1 版本16.3.2 版本26.3.3 版本36.3.4 版本4 7.死锁7.1 多线程卖票问题 8.线程通信8.1 wait()和sleep的区别&#xff1f;8.2 两个线程交替输出8.3 三个线程交替输出8.4 线程通信-生产者和消费者…...

WebKit是什么?

WebKit是一个开源的浏览器引擎&#xff0c;它用于呈现网页内容在许多现代浏览器中&#xff0c;包括Safari浏览器、iOS内置浏览器、以及一些其他浏览器如Google Chrome的早期版本。以下是一些关于WebKit的重要信息&#xff1a; 起源和发展&#xff1a;WebKit最初是由苹果公司为其…...

谷歌(Google)历年编程真题——接雨水

谷歌历年面试真题——数组和字符串系列真题练习。 接雨水 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图&#xff0c;计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 示例 1&#xff1a; 输入&#xff1a;height [0,1,0,2,1,0,1,3,2,1,2,1] 输出&#xff1a;…...

盲盒小程序加盟/河北seo公司

SpringBoot 简介 微框架&#xff0c;与 Spring4 一起诞生&#xff0c;基于约定、生来为了简化 spring 的配置 优点 可以快速的上手&#xff0c;整合了一些子项目(开源框架或者第三方开源库)可以依赖很少的配置快速的搭建项目基于 spring 使开发者快速入门&#xff0c;门槛很低。…...

网站如何做seo排名/谈谈自己对市场营销的理解

详细连接https://blog.csdn.net/qq_40213457/article/details/81021562https://blog.csdn.net/coolwaterld/article/details/72467942下载不下来&#xff0c;直接利用全部模型有170M&#xff0c;但在线下载速度非常慢&#xff0c;我下了一天一夜才下完。不过现在小伙伴已经将其…...

使用php的大型网站/怎样优化网站排名

同源策略&#xff08;Same origin policy&#xff09;是一种约定&#xff0c;它是浏览器最核心也最基本的安全功能. 就浏览器而言的, http://127.0.0.1:8000 协议 域名 端口 跨域 问题//简单请求跨域 S1项目 127.0.0.1:8000 from django.shortcuts import render from…...

wordpress变成中文/网站设计公司建设网站

据 Gartner 2022 年最新趋势分析&#xff0c;数据分析将成为创新起源与企业核心能力&#xff0c;数据越来越重要了。在更早前 IDC 和数据存储公司希捷的报告表示&#xff0c;我国产生的数据量从 2019 年的约 9.4ZB 将猛增至 2025 年的 48.6ZB。现在&#xff0c;数据工程师需要面…...

好设计英文网站/推广是什么意思

原创不易&#xff0c;转载前请注明博主的链接地址&#xff1a;Blessy_Zhu https://blog.csdn.net/weixin_42555080 本次代码的环境&#xff1a; 运行平台&#xff1a; Windows Python版本&#xff1a; Python3.x IDE&#xff1a; PyCharm 一、 前言 在上一篇博文机器学习笔记…...

做日签的网站/泉州百度广告

问题 1 如何批量删除macro 2 导入的macro乱码怎么办 解答 问题1: 通过编辑 C:\Users\你的电脑用户名\AppData\Roaming\MobaXterm\MobaXterm.ini 搜索 [Macros] 和 [MacrosHotkeys] 他们两个标题下的内容即对应你的macro. 注意: 编辑前必须先关闭mobaxterm软件. 问题2: macr…...