Linux进程——system函数、popen函数

system函数(执行shell 命令)

头文件

cs 复制代码
#include <stdlib.h>

函数定义

cs 复制代码
int system(const char * string);

函数说明

system()会调用fork()产生子进程,由子进程来调用/bin/sh-c string来执行参数string字符串所代表的命令,此命令执行完后随即返回原调用的进程。在调用system()期间SIGCHLD 信号会被暂时搁置,SIGINT和SIGQUIT 信号则会被忽略。

返回值

如果system()调用/bin/sh时失败则返回127,其他失败原因返回-1。若参数string为空指针(NULL),则返回非零值。如果 system()调用成功则最后会返回执行shell命令后的返回值,但是此返回值也有可能为system()调用/bin/sh失败所返回的127,因此最好能再检查errno 来确认执行成功。

附加说明

在编写具有SUID/SGID权限的程序时请勿使用system(),system()会继承环境变量,通过环境变量可能会造成系统安全的问题。

代码示例

cs 复制代码
#include<stdio.h>
#include <unistd.h>
int main()
{
        int a= 0;
        int fork_r=0;
        while(1)
	    {                
		        printf("please input a num\n");
                scanf("%d",&a);
                if(a==1)
		        {
                	fork_r=fork();
                    if(fork_r==0)
			        {
//             	 		execl("./changeData","changeData",NULL,NULL);
				        system("./changeData");//直接将可执行文件字符串输入便会执行指令
                    }
                }
		        else
		        {
                        printf("no change success\n");
                }
        }
        return 0;
}

直接调用changeData可执行文件并执行,成功将数据修改。

popen函数(执行shell 命令,同时可以获取运行的输出结果)

头文件

cs 复制代码
#include <stdio.h>

函数定义

cs 复制代码
FILE popen( const char* command, const char* mode )

函数说明

command: 是一个指向以 NULL 结束的 shell 命令字符串的指针。这行命令将被传到 bin/sh 并使用 -c 标志,shell 将执行这个命令。
type: 只能是读或者写中的一种。
如果 type 是 "r" 则文件指针连接到 command 的标准输出;
如果 type 是 "w" 则文件指针连接到 command 的标准输入。

依照此type值,popen()会建立管道连到子进程的标准输出设备或标准输入设备,然后返回一个文件指针,随后进程便可利用此文件指针来读取子进程的输出设备或是写入到子进程的标准输入设备中。

返回值

成功,则返回一个读或者打开文件的指针
失败,返回NULL,具体错误要根据errno判断

附加说明

popen函数建立的管道连到子进程的输出或输入设备即读写操作需要调用到前面学过的fread(读)和fwrite(写)

cs 复制代码
size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
size_t fwrite(const void *ptr, size_t size, size_t nmemb,FILE *stream);

popen() 函数用于创建一个管道:其内部实现为调用 fork 产生一个子进程,执行一个 shell 以运行命令来开启一个进程,而这个进程必须由 pclose() 函数关闭。stream是popen返回的文件指针。

cs 复制代码
int pclose(FILE *stream);

代码示例

cs 复制代码
#include <stdio.h>

int main()
{
	char result[512]={0};
	FILE *fp = NULL;
	int n_read = 0;

	fp = popen("ps","r");//r为输出(读取)
	n_read = fread(result,1,128,fp);//每次读一个字节,总共读512次(小于512则会读真实值,大于512会报错),将fp的内容读取到result
	printf("read %d byte to result,result = %s\n",n_read,result);
	pclose(fp);	

	return 0;
}

成功获取运行输出结果并将其打印出来。

相关推荐
TDD_06288 小时前
【运维】Centos硬盘满导致开机时处于加载状态无法开机解决办法
linux·运维·经验分享·centos
x66ccff8 小时前
vLLM 启动 GGUF 模型踩坑记:从报错到 100% GPU 占用的原因解析
linux
William.csj8 小时前
Linux——开发板显示器显示不出来,vscode远程登录不进去,内存满了的解决办法
linux·vscode
KeithTsui9 小时前
GCC RISCV 后端 -- 控制流(Control Flow)的一些理解
linux·c语言·开发语言·c++·算法
森叶9 小时前
linux如何与windows进行共享文件夹开发,不用来回用git进行拉来拉去,这个对于swoole开发者来说特别重要
linux·git·swoole
liulilittle9 小时前
Linux 高级路由策略控制配置:两个不同路由子网间通信
linux·网络·智能路由器
学习至死qaq9 小时前
windows字体在linux访问异常
linux·运维·服务器
在野靡生.10 小时前
Ansible(4)—— Playbook
linux·运维·ansible
Linux技术芯10 小时前
Linux内核内存管理 ARM32内核内存布局的详细解析和案例分析
linux
烨鹰10 小时前
戴尔电脑安装Ubuntu双系统
linux·运维·ubuntu