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

Linux:进程控制(一)

目录

一、写时拷贝

1.创建子进程

2.写时拷贝

 二、进程终止

1.函数返回值

2.错误码

3.异常退出

4.exit

5._exit


一、写时拷贝

        父子进程,代码共享,不作写入操作时,数据也是共享的,当任意一方试图写入,便通过写时拷贝的方式各自有一份副本。

1.创建子进程

        这是父进程的大致蓝图。

        创建子进程,分配内存块内核数据结构给子进程。

         将父进程的部分数据结构内容拷贝到子进程,即代码内容共享数据内容暂时共享

2.写时拷贝

  • 写实拷贝的大致蓝图

  • 为什么要作写时拷贝?

        子进程不一定会用到父进程的全部数据,故作写时拷贝用来完成特定的数据需求。

  • 页表

        页表并不是只有虚拟地址和物理地址两列,还有其他属性列,如“权限”、“是否存在于内存”。权限列有只读、可读可写等属性。

  • 写时拷贝的具体过程

        子进程复制父进程的部分内容后,操作系统在各自的页表中将数据段的属性设置为只读

        当发生写时拷贝时,由于页表项的属性是只读,故造成缺页中断,中断处理过程时,操作系统将页表项属性再设置为可读可写

 二、进程终止

1.函数返回值

        main函数返回值是当前进程的退出码,而普通函数的返回值仅仅是函数调用的结果。

        main函数返回值为0,即退出码为0,表示当前程序运行成功,非零则表示失败,而失败的原因由错误码表示。

2.错误码

        函数strerror用来打印错误码详情,Linux下错误码如下所示。

int main()
{for(int i = 0;i < 200 ;++i){printf("%d : %s\n",i, strerror(i));}return 0;
}
0 : Success
1 : Operation not permitted
2 : No such file or directory
3 : No such process
4 : Interrupted system call
5 : Input/output error
6 : No such device or address
7 : Argument list too long
8 : Exec format error
9 : Bad file descriptor
10 : No child processes
11 : Resource temporarily unavailable
12 : Cannot allocate memory
13 : Permission denied
······

        错误码转换为错误描述,一般有两种情形,使用系统提供的或者自定义

  • 自定义错误码
enum 
{success=0,open_err,malloc_err
};
const char* errornumToDisc(int code)
{switch(code){case success:return "Success!";case open_err:return "Open fail";case malloc_err:return "malloc error";default:return "unkonw errornum";}
}
int main()
{int code = open_err;printf("%d->%s\n",code,errornumToDisc(code));return code;
}
  •  系统提供的错误码errno

        当前程序退出后,默认errno保存着错误码。

        如果程序正常退出,errno的值为0

int main()
{printf("%d->%s\n",errno,strerror(errno));
}

        可以通过echo $?查看最近一次运行的退出码。 

[euto@VM-4-13-centos 241002]$ ./myprocess 
0->Success
[euto@VM-4-13-centos 241002]$ echo $?
0

        如果遇到错误退出,则可以通过errno打印错误信息。

int main()
{FILE* pf = fopen("a.txt","r");printf("%d->%s\n",errno,strerror(errno));return 0;
}
[euto@VM-4-13-centos 241002]$ ./myprocess 
2->No such file or directory

3.异常退出

        程序正常退出有三种方式。

  1. main函数的返回值
  2. 库函数exit
  3. 系统调用接口_exit

        进程退出的场景只有三种:

        1.正常退出:程序运行结束,运行成功,main函数返回0。

        2.正常退出:程序运行结束,运行失败,main函数返回非零值。

        3.异常退出:代码并没有被全部执行,一般异常退出是由于进程收到了异常信号,信号编码对应着发生异常的原因。

        执行kill -l这些都是可以让进程异常退出的信号。

[euto@VM-4-13-centos 241002]$ kill -l1) SIGHUP	 2) SIGINT	 3) SIGQUIT	 4) SIGILL	 5) SIGTRAP6) SIGABRT	 7) SIGBUS	 8) SIGFPE	 9) SIGKILL	10) SIGUSR1
11) SIGSEGV	12) SIGUSR2	13) SIGPIPE	14) SIGALRM	15) SIGTERM
16) SIGSTKFLT	17) SIGCHLD	18) SIGCONT	19) SIGSTOP	20) SIGTSTP
21) SIGTTIN	22) SIGTTOU	23) SIGURG	24) SIGXCPU	25) SIGXFSZ
26) SIGVTALRM	27) SIGPROF	28) SIGWINCH	29) SIGIO	30) SIGPWR
31) SIGSYS	34) SIGRTMIN	35) SIGRTMIN+1	36) SIGRTMIN+2	37) SIGRTMIN+3
38) SIGRTMIN+4	39) SIGRTMIN+5	40) SIGRTMIN+6	41) SIGRTMIN+7	42) SIGRTMIN+8
43) SIGRTMIN+9	44) SIGRTMIN+10	45) SIGRTMIN+11	46) SIGRTMIN+12	47) SIGRTMIN+13
48) SIGRTMIN+14	49) SIGRTMIN+15	50) SIGRTMAX-14	51) SIGRTMAX-13	52) SIGRTMAX-12
53) SIGRTMAX-11	54) SIGRTMAX-10	55) SIGRTMAX-9	56) SIGRTMAX-8	57) SIGRTMAX-7
58) SIGRTMAX-6	59) SIGRTMAX-5	60) SIGRTMAX-4	61) SIGRTMAX-3	62) SIGRTMAX-2
63) SIGRTMAX-1	64) SIGRTMAX	

4.exit

EXIT(3)                                                             Linux Programmer's Manual                                                             EXIT(3)NAMEexit - cause normal process terminationSYNOPSIS#include <stdlib.h>void exit(int status);

        exit函数是由C语言库提供,用来直接退出当前进程,参数的值即为退出码,如果运行成功,则执行exit(0)退出。

5._exit

_EXIT(2)                                                            Linux Programmer's Manual                                                            _EXIT(2)NAME_exit, _Exit - terminate the calling processSYNOPSIS#include <unistd.h>void _exit(int status);#include <stdlib.h>void _Exit(int status);

        _exit是Linux系统调用接口,由Linux操作系统提供,总体上的用法和exit没有区别。差异在于exit函数在退出进程的时候会刷新缓冲区,而_exit函数则不会刷新缓冲区,不会刷新缓冲区也说明这个缓冲区不在操作系统内部。
        分别执行下面两段代码,观察有何不同。

int main()
{printf("hello");exit(0);
}
int main()
{printf("hello");_exit(0);
}

        总结,推荐使用exit,原因不是因为它会刷新缓冲区,而是因为它是由C语言库提供,内部其实封装了_exit,进程的结束肯定是由操作系统调用系统调用接口来结束,exit只是封装Linux系统调用接口_eixt罢了,同时也封装了Windows平台的系统调用接口,这样一来,使用exit的程序具备可移植性,在Linux平台调用Linux平台的系统调用接口_exit,在Windows平台调用Windows平台的系统调用接口。

http://www.lryc.cn/news/452317.html

相关文章:

  • 初识算法 · 双指针(3)
  • 【AI知识点】近似最近邻搜索(ANN, Approximate Nearest Neighbor Search)
  • 编程工具简介
  • 汽车信息安全 -- 存到HSM中的密钥还需包裹吗?
  • 【PostgreSQL】入门篇——SELECT、INSERT、UPDATE 和 DELETE 语句,SQL 中最常用的四种操作用法
  • 【Ubuntu】安装常用软件包-mysql
  • 幂等性及技术解决方案
  • 正向代理 反向代理
  • 【分布式微服务云原生】如何在ActiveMQ中优雅处理提前支付的延时订单
  • Easy Excel从入门到精通!!!
  • 简易CPU设计入门:取指令(三),ip_buf与rd_en的非阻塞赋值
  • 【算法】---归并排序(递归非递归实现)
  • UniVue大版本更新:UniVue2.0.0-preview
  • RabbbitMQ篇(环境搭建 - 下载 安装)(持续更新迭代)
  • C++基础补充(02)C++其他控制语句break continue goto等
  • 决策树中联合概率分布公式解释说明
  • 计算机毕业设计 农场投入品运营管理系统的设计与实现 Java实战项目 附源码+文档+视频讲解
  • php email功能实现:详细步骤与配置技巧?
  • MapBox Android版开发 6 关于Logo
  • 2024年房市
  • index索引
  • 理解互联网链路:从本地ISP到Tier 1 ISP运营商
  • 基于元神操作系统实现NTFS文件操作(三)
  • 深度学习与数学归纳法
  • 《Linux从小白到高手》理论篇(六):Linux软件安装一篇通
  • 【Spring】运行Spring Boot项目,请求响应流程分析以及404和500报错
  • ②EtherCAT转Modbus485RTU网关多路同步高速采集无需编程串口服务器
  • matlab-对比两张图片的HSV分量的差值并形成直方图
  • 微服务SpringGateway解析部署使用全流程
  • Solidity 存储和内存管理:深入理解与高效优化