前面我们讲了C语言的基础知识,也了解了一些数据结构,并且讲了有关C++的一些知识,也学习了一些Linux的基本操作,也了解并学习了有关Linux开发工具vim 、gcc/g++ 使用、yum工具以及git 命令行提交代码也相信大家都掌握的不错,上一篇文章我们了解了关于进程的地址空间,今天博主带大家了解一下 —— 进程的控制——创建、终止、等待、程序替换, 下面话不多说坐稳扶好咱们要开车了!!!
fork
函数是操作系统中的一个系统调用,用于创建一个新的进程,该进程是调用fork函数的进程的一个副本。新创建的进程称为子进程,原始进程称为父进程。
fork
函数的函数原型:
#include
pid_t fork(void);
父进程中的返回值:
fork
函数返回一个大于0的值,表示当前执行的是父进程。这个返回值是子进程的PID(进程ID),可以用来操作子进程。fork
函数返回-1,表示创建子进程失败,通常是因为系统资源不足或权限不够等原因,此时应该处理错误情况。子进程中的返回值:
fork
函数返回0,表示当前执行的是子进程。可以根据需要在子进程中执行相应的任务逻辑。根据fork函数的返回值,可以在程序中使用条件语句来区分父进程和子进程的不同逻辑,从而实现不同的处理方式。例如,可以在父进程中等待子进程的完成,或者在子进程中执行某种特定的任务。
#include
#include
#include
int main() {
pid_t pid = fork();
if (pid > 0) {
// 父进程逻辑
printf("This is the parent process. Child's PID: %d\n", pid);
} else if (pid == 0) {
// 子进程逻辑
printf("This is the child process. Parent's PID: %d\n", getppid());
} else {
// fork失败
fprintf(stderr, "Failed to create child process.\n");
return 1;
}
// 父子进程共享的代码
printf("This message is printed by both parent and child processes.\n");
return 0;
}
需要注意的是,fork
函数的使用可能会导致代码的分支,需要小心处理父子进程之间共享的资源以及避免产生竞争条件,以确保程序的正确性和可靠性。
fork
函数的写时拷贝(Copy-on-Write,COW)是一种优化策略,用于在创建子进程时避免立即复制父进程的整个地址空间,这种机制可以提高性能和减少内存消耗。
在传统的fork操作中,父进程会创建一个子进程,并且子进程会复制父进程的所有资源,包括内存空间、文件描述符等。这样的完全复制操作非常消耗时间和内存。而使用写时拷贝机制,只有在需要修改共享的内存页时才会进行复制操作,从而节省了系统资源。
具体来说,当调用fork函数创建子进程时,操作系统会执行以下步骤:
通过写时拷贝技术,父进程和子进程共享大部分内存页,只在需要修改共享内存时才进行复制操作。这样可以节省时间和内存,并提高系统性能。例如,在fork之后,如果子进程立即执行exec函数加载了一个新的程序,那么就不需要进行任何复制操作,这是因为子进程并不需要修改父进程的内存数据。
需要注意的是,写时拷贝只是在逻辑上实现了共享,而不是物理上的共享。父进程和子进程仍然拥有各自独立的虚拟地址空间,它们之间的共享是通过允许读取相同的物理内存来实现的,只有在修改时才会发生内存复制。
总结起来,fork函数的写时拷贝机制使得父进程和子进程在初始阶段共享相同的内存空间,只有在需要修改共享内存时才进行复制操作,从而提高了性能和降低了资源消耗。
代码运行完毕,结果正确:当我们运行一个程序时,通常会期望它在合理的时间内完成指定的任务,并返回正确的结果。如果程序能够正常执行并得到正确的输出,那么我们就称之为结果正确。
代码运行完毕,结果不正确:然而,在某些情况下,程序可能会返回错误的输出,这时我们就说结果不正确。这可能是由于程序中的逻辑错误、算法错误、数据不一致或格式错误等原因导致的。
代码异常终止:另外一种情况是当程序因为错误或异常而非正常退出时,我们称之为代码异常终止。在这种情况下,程序可能会抛出异常、崩溃或者停止响应。
exit()
函数和Java的System.exit()
方法。这些函数可以接受一个整数作为参数,表示进程的退出状态码,通常使用0表示正常终止。int main() {
// 执行完所有任务后,通过返回0表示正常终止进程
return 0;
}
ctrl + c
,信号终止exit
函数是一个库函数,它用于正常终止程序并返回到操作系统。通过调用 exit
函数,进程会经历一系列清理操作,例如关闭打开的文件、释放动态分配的内存等,然后将控制权交还给操作系统。这个过程被称为进程的正常退出。
⭕exit
函数的原型:
#include
void exit(int status);
其中,status
参数表示进程的退出状态值。它可以是整数类型,用来传递进程的运行结果或者其他信息。通常情况下,使用 0 表示程序的正常退出,非零值表示程序异常退出或者出错的特定状态。
⭕当调用 exit
函数时,以下操作将在进程退出之前执行:
atexit
函数注册的函数)。调用 exit
函数后,程序将不会返回到 exit
调用之后的代码位置,而是直接返回到操作系统。因此,在调用 exit
函数后的代码将不会被执行。例如:
#include
#include
int main() {
printf("Before exit\n");
exit(0); // 正常终止进程,退出状态码为0
printf("After exit\n"); // 这行代码不会执行
return 0;
}
注意:使用 exit
函数时应确保在终止前完成必要的清理工作,以免资源泄漏或未定义行为。此外,可以通过在命令行或者父进程中获取退出状态码来获取程序的运行结果或其他信息。
_exit
函数是一个系统调用,它可以用来立即终止进程的执行。与 exit
函数不同的是, _exit
函数不会调用任何注册的终止处理程序,也不会冲洗输出缓冲区,而是通过直接向内核发送退出状态码来终止进程。因此,使用 _exit
函数可以避免在进程终止时执行一些不必要的或者危险的操作,如清理缓存、关闭文件等。
⭕_exit
函数的原型:
#include
void _exit(int status);
其中,status
参数表示进程的退出状态值。它与在 exit
函数中的参数含义相同。
与 exit
函数一样,由 _exit
函数终止的进程将永远不会返回到其调用进程的代码路径处。进程资源(包括打开的文件、未释放的内存等)都将被释放,并通知父进程该进程已经结束。
注意:由于 _exit
函数是一个系统调用,因此它并没有对库函数进行善后工作,如果你在 _exit
函数之前使用了库(如 iostream),则可能会导致内存泄露或者其他未定义的行为。为了避免这种情况,应该在使用 _exit
函数前显式地调用 std::flush
函数将缓冲区清空,并且进行必要的资源释放工作。
库函数与系统调用:exit
是一个库函数,而 _exit
是一个系统调用。 exit
函数通过调用 _exit
系统调用来终止进程。
清理操作的执行:exit
函数在终止进程之前执行一系列清理操作,例如关闭文件、刷新缓冲区等。它还会调用注册的终止处理程序(通过 atexit
函数注册的函数)进行额外的清理工作。相比之下,_exit
函数不会执行这些清理操作,直接终止进程。
缓冲区刷新:exit
函数会自动刷新输出缓冲区,确保所有的输出都被写入文件。而 _exit
函数不会主动刷新缓冲区,可能导致部分输出被丢失。
返回控制权:exit
函数返回到调用 exit
的代码位置,因此可以在程序中根据退出状态码进行特定操作。相反,_exit
直接将控制权返回给操作系统,不会返回到调用 _exit
的代码位置。
资源释放:由于 exit
函数会执行清理操作,可以确保打开的文件被正常关闭、动态分配的内存被释放等。而 _exit
函数不会执行这些操作,可能导致资源泄漏。
⭕exit
函数是一个高级的、安全的方式来终止程序。它会执行一系列清理工作并确保输出被正确处理。相比之下,_exit
函数适用于需要立即终止进程且不进行任何额外处理的情况,如遇到致命错误或者需要在子进程中终止时使用。
return
是一种更常见的退出进程方法。执行return n;
等同于执行exit(n)
,因为调用main
的运行时函数会将main
的返回值当做 exit
的参数。与 exit
和 _exit
函数不同,return
语句只能在函数内部使用,并且它是一种正常的控制流程操作。
下面是 return
语句的一些关键点:
函数中止: return
语句的执行将导致函数立即结束并返回到函数的调用者。在函数执行 return
后,没有其他代码会被执行。
返回值: return
语句可以选择性地返回一个值给调用者。例如,在有返回类型的函数中,如 int
类型,可以使用 return
返回一个整数值。
返回类型检查: 在函数定义时,需要声明函数的返回类型。如果函数声明为 void
类型,则表示没有任何返回值,此时可以省略 return
语句或使用 return;
来提前结束函数。
多个返回点: 函数中可以存在多个 return
语句,表示在不同的条件下提前返回。在这种情况下,只有一个 return
语句会被执行,其他的 return
语句都将被忽略。
例如:
#include
int sum(int a, int b) {
int result = a + b;
return result; // 返回计算结果给调用者
}
void printMessage() {
printf("Hello, World!\n");
return; // 结束函数,没有返回值
}
int main() {
printf("Before return\n");
return 0; // 终止 main 函数并退出程序
printf("After return\n"); // 这行代码不会执行
}
注意:return
语句只能在函数内部使用,不能用于终止整个程序的执行。要终止整个程序,可以使用 exit
或 _exit
函数。而 return
语句只影响当前函数的执行流程。
进程等待是指一个进程暂停执行,直到某个条件满足或者某个事件发生后再继续执行的操作。在操作系统中,进程等待是一种同步机制,用于实现多个进程之间的协调和互动。
下面是一些常见的情况,需要使用进程等待:
资源共享:多个进程可能需要同时访问某些共享资源,如数据库、文件、网络连接等。为了避免并发访问导致资源冲突和数据不一致,需要通过进程等待来协调各个进程对资源的使用。
任务依赖:有些任务的执行可能依赖于其他任务的完成。如果一个进程要执行的任务需要使用其他进程尚未完成的结果,就需要使用进程等待来确保所需的数据已经准备好了。
同步操作:在多线程或者多进程编程中,有时候需要保证某些操作的顺序性和一致性。例如,在生产者-消费者模型中,消费者进程必须等待生产者进程将数据放入共享缓冲区后才能进行消费。
并发控制:有时候需要限制同时执行某个代码块的进程数量。例如,资源受限的情况下,只允许有限数量的进程同时执行某个操作,其他进程需要等待。
进程等待的主要目的是确保多个进程之间的协调和同步,以避免资源竞争、数据不一致和并发冲突等问题。通过使用进程等待,可以实现进程之间的合作和资源共享,从而提高系统的可靠性和效率。
wait
方法:wait
方法是在父进程中调用的,用于等待任一子进程退出或终止。它的语法为:
pid_t wait(int *status);
status
指向一个整型变量,用于获取子进程的退出状态信息。wait
方法会将调用进程挂起,直到任一子进程退出或终止。如果有多个子进程同时退出,它会返回任一子进程的进程ID。status
参数获取子进程的退出状态。实战演示:
#include
#include
#include
#include
#include
int main() {
pid_t pid;
int status;
pid = fork();
if (pid < 0) {
fprintf(stderr, "Fork failed.\n");
exit(1);
} else if (pid == 0) {
// 子进程执行的代码
printf("This is the child process.\n");
exit(123); // 子进程以退出状态123结束
} else {
// 父进程执行的代码
printf("This is the parent process.\n");
wait(&status);
if (WIFEXITED(status)) {
printf("Child process exited with status: %d\n", WEXITSTATUS(status));
}
}
return 0;
}
在上面的示例中,父进程通过fork()
创建了一个子进程。子进程输出一条消息后调用exit(123)
退出,并传递了退出状态值为123给父进程。父进程使用wait()
等待子进程的退出,并通过WIFEXITED
和WEXITSTATUS
宏判断子进程是否正常退出,并获取其退出状态值。
waitpid()
函数是一个用于等待指定子进程状态改变的系统调用。它具有比wait()
更为灵活的功能,可以指定等待的子进程、等待选项以及获取子进程的退出状态信息。
它的语法为:
pid_t waitpid(pid_t pid, int *status, int options);
pid
:表示要等待的子进程的PID。
status
:一个整型指针,用于获取子进程的退出状态信息。(如果不关心子进程的退出状态,可以将该参数设置为NULL
)。
options
:参数用于指定等待选项,可以使用多个选项,通过按位或(|)进行组合。常见的选项有:
返回值:
PID
(即pid参数指定的值)。-1
,并设置errno
来指示具体的错误原因。注意事项:
waitpid()
函数时,父进程会挂起(阻塞),直到指定的子进程状态发生改变。waitpid()
函数提供了比wait()
更灵活的方式来等待子进程状态改变,并且可以通过指定pid
、options
参数来满足不同的等待需求。waitpid()
立即返回子进程的PID
;如果指定的子进程尚未退出,则根据指定的选项来确定是否挂起等待。waitpid()
函数时,父进程可以同时等待多个子进程的状态改变。waitpid()
函数可以用于实现非阻塞的等待子进程状态改变的操作,即通过设置WNOHANG
选项,在没有可等待的子进程时立即返回。示例代码:
#include
#include
#include
#include
#include
int main() {
pid_t pid;
int status;
pid = fork();
if (pid < 0) {
fprintf(stderr, "Fork failed.\n");
exit(1);
} else if (pid == 0) {
// 子进程执行的代码
printf("This is the child process.\n");
exit(123); // 子进程以退出状态123结束
} else {
// 父进程执行的代码
printf("This is the parent process.\n");
while (waitpid(pid, &status, WNOHANG) == 0) {
sleep(1);
printf("Waiting for child process...\n");
}
if (WIFEXITED(status)) {
printf("Child process exited with status: %d\n", WEXITSTATUS(status));
}
}
return 0;
}
在上面的示例中,父进程通过fork()
创建了一个子进程。子进程输出一条消息后调用exit(123)
退出,并传递了退出状态值为123给父进程。父进程使用waitpid()
和WNOHANG
选项进行非阻塞地等待子进程的退出,并通过WIFEXITED
和WEXITSTATUS
宏判断子进程是否正常退出,并获取其退出状态值。
这两个方法都可以实现父进程对子进程的等待操作,并获取子进程的状态信息。waitpid
方法相比wait
方法更加灵活,可以选择具体的子进程进行等待,在一些场景下更加有用。
wait
和waitpid
,都有一个status
参数,该参数是一个输出型参数,由操作系统填充。NULL
,表示不关心子进程的退出状态信息。status
不能简单的当作整形来看待,可以当作位图来看待,具体细节如下图(只研究status
低16比特位)WIFEXITED
和WEXITSTATUS
是用于解析子进程退出状态信息的宏定义。下面是对这两个宏的详细介绍:
WIFEXITED
宏:int WIFEXITED(int status);
WIFEXITED
宏用于判断子进程是否正常退出(非被信号终止),即子进程是否调用了exit
或返回了main
函数。
参数status
:子进程的退出状态信息,通常是通过wait()
或waitpid()
函数获取的。
返回值:若子进程正常退出,则返回非零值(true
),否则返回0(false
)。
int WEXITSTATUS(int status);
WEXITSTATUS
宏用于获取子进程的退出状态值(0-255
)。只有在WIFEXITED
宏为true
时才有效。
参数status
:子进程的退出状态信息,通常是通过wait()
或waitpid()
函数获取的。
返回值:子进程的退出状态值。
⭕进程程序替换(Process Program Replacement)是指在一个正在运行的进程中将当前执行的程序替换为新的程序的操作。也可以称之为进程映像替换(Process Image Replacement)。通过进程程序替换,可以使一个进程在执行过程中切换到另一个不同的程序,并继续执行。使用进程程序替换时,当前进程的内存空间会被新的程序覆盖,包括代码、数据和堆栈等信息。新程序的代码取代原先进程的代码,新程序的数据取代原先进程的数据,新程序的堆栈取代原先进程的堆栈,从而实现了进程的切换。
进程程序替换的原理是通过操作系统提供的相关函数,将当前进程的执行映像(包括代码、数据和堆栈等信息)替换为新的程序的执行映像。具体步骤如下:
需要注意的是,进程程序替换只会替换当前进程的执行映像,不会影响其他进程。替换后的程序会继承原进程的一些状态信息,如文件描述符、信号处理设置等,以确保新程序能够正常运行。
六种以exec开头的函数,统称exec函数:
int execl(const char *path, const char *arg0, ... /* (char *) NULL */);
path
是指定可执行文件的路径,参数 arg0
和之后的参数是传递给新程序的命令行参数。int execlp(const char *file, const char *arg0, ... /* (char *) NULL */);
$PATH
指定的目录中查找。file
是指定可执行文件的名称,参数 arg0
和之后的参数是传递给新程序的命令行参数。int execle(const char *path, const char *arg0, ... /* (char *) NULL, char *const envp[] */);
envp
参数设置新程序的环境变量。path
是指定可执行文件的路径,参数 arg0
和之后的参数是传递给新程序的命令行参数,最后的 envp
参数是环境变量数组。int execv(const char *path, char *const argv[]);
path
是指定可执行文件的路径,参数 argv
是传递给新程序的参数数组,以NULL结尾。int execvp(const char *file, char *const argv[]);
$PATH
指定的目录中查找。file
是指定可执行文件的名称,参数 argv
是传递给新程序的参数数组,以NULL结尾。int execve(const char *path, char *const argv[], char *const envp[]);
path
是指定可执行文件的路径,参数 argv
是传递给新程序的参数数组,以NULL结尾,envp
是环境变量数组。这些函数都可以在C/C++等编程语言中使用,并通过设置参数来实现进程程序替换的功能。具体使用哪个函数取决于是否需要指定路径和环境变量以及参数的形式(以列表或数组表示)。
函数名 | 参数格式 | 是否带路径 | 是否使用当前环境变量 |
---|---|---|---|
execl | 列表 | 不是 | 是 |
execlp | 列表 | 是 | 是 |
execle | 列表 | 不是 | 不是,需自己组装环境变量 |
execv | 数组 | 不是 | 是 |
execvp | 数组 | 是 | 是 |
execve | 数组 | 不是 | 不是,需自己组装环境变量 |
进程创建
fork
函数,可以复制当前进程创建一个新的子进程。fork
函数的返回值不同,子进程中返回0,父进程中返回子进程的PID。fork
函数利用写时拷贝技术实现了高效的进程创建。进程终止
exit
函数和_exit
函数、return
语句以及异常终止。exit
函数是标准库函数,进行一些清理工作后终止进程。_exit
函数是系统调用函数,直接终止进程。return
语句是从main
函数返回,等价于调用exit
函数。进程等待
wait
和waitpid
函数来实现进程等待。wait
函数会阻塞父进程,直到子进程退出,返回子进程的PID。waitpid
函数可以指定等待的子进程PID,并具有更多的选项。进程程序替换
exec
函数族可以实现进程程序的替换。exec
函数会加载新的程序代码和数据,并开始执行新程序。感谢您对博主文章的关注与支持!如果您喜欢这篇文章,可以点赞、评论和分享给您的同学,这将对我提供巨大的鼓励和支持。另外,我计划在未来的更新中持续探讨与本文相关的内容。我会为您带来更多关于Linux以及C++编程技术问题的深入解析、应用案例和趣味玩法等。如果感兴趣的话可以关注博主的更新,不要错过任何精彩内容!
再次感谢您的支持和关注。我们期待与您建立更紧密的互动,共同探索Linux、C++、算法和编程的奥秘。祝您生活愉快,排便顺畅!