热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

进程控制2—进程编程

紧接着上一篇,我们来写写关于进程的编程。获取ID通过mangetpid这个命令我们可以在Linux终端上获取到这个函数详细的讲解#include<systypes.h>

紧接着上一篇,我们来写写关于进程的编程。

获取ID

通过man getpid这个命令我们可以在Linux终端上获取到这个函数详细的讲解

#include 
#include //头文件

//获取本进程ID
pid_t getpid(void)

//获取父进程ID
pid_t getppid(void)

下面通过一段代码来看看查找进程ID

#include 
#include
#include

int main()
{
printf ("当前进程id:%d\n",getpid());
printf ("当前父进程id:%d\n",getppid());
printf ("当前用户进程id:%d\n",getuid());

while (1);
return 0;
}

输出结果
输出结果在不同终端上会有所不同。

进程创建fork()

#include 

pid_t fork(void)//创建进程

功能:创建子进程
fork()有一个很神奇的地方,就是在于它被调用一次,却返回两次,它可能有三个不同的返回值:
1.在父进程中,fork返回新创建的子进程的PID;
2.在子进程中,fork返回0;
3若出现错误,fork返回一个负值;

☆当fork()创建成功时,这时候会存在两个进程,每个进程都会fork()处开始执行。
☆两个进程执行相同的代码段(text),但是却各自的堆栈段(stack)、数据段(data)以及堆(heap)。
☆其中子进程的stack,data,heap都是从父进程拷贝过来的,两个进程只是共享代码段。

fork()之后,不能确定哪个进程先执行,会有什么隐患吗?
产生原因:
fork()之后,不能确定是父进程还是子进程获得CPU。
危害:
这种bugs很难被发现。
措施:
如果需要确保特定的执行顺序,需要采用某种同步(synchronization)技术(semaphores,file locks…)。

fork()代码:

#include 
#include
#include

int main()
{
pid_t pid = fork();

if (pid == -1) //创建失败
{
perror ("fork");
return -1;
}

if (pid > 0)
{
printf ("我是父进程,pid = %d\n",getpid());
}

if (pid == 0)
{
printf ("我是子进程,pid = %d\n",getpid());
}

return 0;
}
//那么问题来了
int main()
{
fork();
fork();
fork();
//创建了几个进程
return 0;
}

int main()
{
fork();
fork() && fork() || fork();
fork();
//这又创建了几个进程
return 0;
}

这里提一下fork()的兄弟vfork(),两者功能几乎一样,唯一不同的是vfork()需要用exit()来退出。

exec函数家族

execl()

#include 

int execl(const char * path, const char* arg1,...)

参数:
path : 被执行程序名(含完整路径)。
arg1 - argn: 被执行程序所需的命令行参数,含程序名。以空指针(NULL)结束。

execl代码:

#include 
#include


int main1()
{
int ret = execl("/bin/ls", "ls", NULL);
if (ret == -1)
{
perror ("execl");
return 0;
}
return 0;
}

execlp()
execlp()与execl()唯一不同的是直接写执行文件在当前目录下没有找到会到环境变量中查找
Linux是在bin目录下

int main()
{
int ret = execlp("/mnt/hgfs/share/0809/file", "./file", NULL);
//int ret = execlp("./file", "./file", NULL);一样的路径
//直接写file会到bin目录下查找文件
if (ret == -1)
{
perror ("execlp");
return 0;
}
return 0;
}

execv()

#include 

int execv(const char * path, const char *argv[])

参数:
path : 被执行程序名(含完整路径)。
argv[]: 被执行程序所需的命令行参数数组。

代码:

int main()
{
char *a[100] = {"./file", NULL};
int ret = execv("/mnt/hgfs/share/0809/file", a);
//int ret = execv("./file", a);一样的路径
//后面的参数必须以数组的形式传进来
if (ret == -1)
{
perror ("execlp");
return 0;
}
return 0;
}

system()

#include 

int system(const char* string)

功能:
调用fork产生子进程,由子进程来调用 /bin/sh -c string来执行参数string所代表的命令

首先表示个人很喜欢这个函数,功能强大,使用起来不墨迹,平时喜欢用的system(“clear”)就是清屏,也可以在括号内填上执行文件的路径,我们甚至可以利用它来制造一个伪终端,但有个瑕疵,就是部分功能实现不了。
代码:

int main()
{
char str[100];
while (1)
{
printf ("[root@localhost 0809]# ");
fgets (str, 100, stdin);

system (str);
}
return 0;
}

执行结果这里写图片描述
可以看出来进入伪终端后实行退出当前目录并没有成功。

进程的终止

这里用到exit,_exit来终止进程。两者有个明显的区别:
_exit:直接停止进程,清除其使用的内存,并清除缓冲区的内容。(像是一个冷酷的特工,只求结果不求过程)
exit:在停止进程之前,要检查文件的打开情况,并把文件缓冲区的内容写回到文件才停止进程(好人,帮人帮到底)

下一篇将会详细说明子父进程之间的恩怨情仇。


推荐阅读
  • C语言的经典程序有哪些
    本篇内容介绍了“C语言的经典程序有哪些”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何 ... [详细]
  • C语言注释工具及快捷键,删除C语言注释工具的实现思路
    本文介绍了C语言中注释的两种方式以及注释的作用,提供了删除C语言注释的工具实现思路,并分享了C语言中注释的快捷键操作方法。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文介绍了H5游戏性能优化和调试技巧,包括从问题表象出发进行优化、排除外部问题导致的卡顿、帧率设定、减少drawcall的方法、UI优化和图集渲染等八个理念。对于游戏程序员来说,解决游戏性能问题是一个关键的任务,本文提供了一些有用的参考价值。摘要长度为183字。 ... [详细]
  • ejava,刘聪dejava
    本文目录一览:1、什么是Java?2、java ... [详细]
  • Metasploit攻击渗透实践
    本文介绍了Metasploit攻击渗透实践的内容和要求,包括主动攻击、针对浏览器和客户端的攻击,以及成功应用辅助模块的实践过程。其中涉及使用Hydra在不知道密码的情况下攻击metsploit2靶机获取密码,以及攻击浏览器中的tomcat服务的具体步骤。同时还讲解了爆破密码的方法和设置攻击目标主机的相关参数。 ... [详细]
  • 图解redis的持久化存储机制RDB和AOF的原理和优缺点
    本文通过图解的方式介绍了redis的持久化存储机制RDB和AOF的原理和优缺点。RDB是将redis内存中的数据保存为快照文件,恢复速度较快但不支持拉链式快照。AOF是将操作日志保存到磁盘,实时存储数据但恢复速度较慢。文章详细分析了两种机制的优缺点,帮助读者更好地理解redis的持久化存储策略。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 本文介绍了Linux Shell中括号和整数扩展的使用方法,包括命令组、命令替换、初始化数组以及算术表达式和逻辑判断的相关内容。括号中的命令将会在新开的子shell中顺序执行,括号中的变量不能被脚本余下的部分使用。命令替换可以用于将命令的标准输出作为另一个命令的输入。括号中的运算符和表达式符合C语言运算规则,可以用在整数扩展中进行算术计算和逻辑判断。 ... [详细]
  • 恶意软件分析的最佳编程语言及其应用
    本文介绍了学习恶意软件分析和逆向工程领域时最适合的编程语言,并重点讨论了Python的优点。Python是一种解释型、多用途的语言,具有可读性高、可快速开发、易于学习的特点。作者分享了在本地恶意软件分析中使用Python的经验,包括快速复制恶意软件组件以更好地理解其工作。此外,作者还提到了Python的跨平台优势,使得在不同操作系统上运行代码变得更加方便。 ... [详细]
  • 全面介绍Windows内存管理机制及C++内存分配实例(四):内存映射文件
    本文旨在全面介绍Windows内存管理机制及C++内存分配实例中的内存映射文件。通过对内存映射文件的使用场合和与虚拟内存的区别进行解析,帮助读者更好地理解操作系统的内存管理机制。同时,本文还提供了相关章节的链接,方便读者深入学习Windows内存管理及C++内存分配实例的其他内容。 ... [详细]
  • 《2017年3月全国计算机等级考试二级C语言上机题库完全版》由会员分享,可在线阅读,更多相关《2017年3月全国计算机等级考试二级C语言上机题库完全版( ... [详细]
  • C语言自带的快排和二分查找
    Author🚹:CofCaiEmail✉️:cai.dongjunnexuslink.cnQQ😙:1664866311personalPage&#x ... [详细]
  • c语言基础编写,c语言 基础
    本文目录一览:1、C语言如何编写?2、如何编写 ... [详细]
  • 利用空间换时间减少时间复杂度以及以C语言字符串处理为例减少空间复杂度
    在处理字符串的过程当中,通常情况下都会逐个遍历整个字符串数组,在多个字符串的处理中,处理不同,时间复杂度不同,这里通过利用空间换时间等不同方法,以字符串处理为例来讨论几种情况:1: ... [详细]
author-avatar
江南水恋
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有