linux下如何用c语言调用shell命令

linux下如何用c语言调用shell命令,第1张

参数type可使用“r”代表读取,“w”代表写入。依照此type值,popen()会建立管道连到子进程的标准输出设备或标准输入设备,然后返回一个文件指针。随后进程便可利用此文件指针来读取子进程的输出设备或是写入到子进程的标准输入设备中。此外,所有使用文件指针(FILE) *** 作的函数也都可以使用,除了fclose()以外。 返回值:若成功则返回文件指针,否则返回NULL,错误原因存于errno中。 注意:在编写具SUID/SGID权限的程序时请尽量避免使用popen(),popen()会继承环境变量,通过环境变量可能会造成系统安全的问题。 例:C程序popentestc内容如下: #include main() { FILE fp; charbuffer[80]; fp=popen(“~/myprogram/testsh”,”r”); fgets(buffer,sizeof(buffer),fp); printf(“%s”,buffer); pclose(fp); } 执行结果如下: xiakeyou@ubuntu:~/myprogram$ vim popentestc xiakeyou@ubuntu:~/myprogram$ gcc popentestc -o popentest xiakeyou@ubuntu:~/myprogram$ /popentest /home/d/e/xiakeyou xiakeyou@ubuntu:~/myprogram$ 只是偶能力可能有点有限,没有太看懂。直接用system()倒是脚本可是执行,只是返回值却是一塌糊涂,试了多次也没有找到什么规律。不免又看了一下上面的那篇博文,得到一些启发,可以这样来实现: 先将脚本的返回值利用 echo > XXXXX 输出到一个本地文件中 当需要这个返回值是,可是通过C语言的文件 *** 作函数来直接从文件中读取 后来一想,这应该就是上文中POPEN的实现方法! C程序调用shell脚本共有三种法子 :system()、popen()、exec系列函数 system() 不用你自己去产生进程,它已经封装了,直接加入自己的命令exec 需要你自己 fork 进程,然后exec 自己的命令 popen() 也可以实现执行你的命令,比system 开销小 1)system(shell命令或shell脚本路径); 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 来确认履行 成功 。 system命令以其简略 高效的作用得到很很广泛 的利用 ,下面是一个例子 例:在~/test/目录下有shell脚本testsh,内容为 #!bin/bash #testsh echo hello 在同层目录下新建一个c文件system_testc,内容为: #include int main() { system("~/test/testsh"); } 履行 效果 如下: [root@localhost test]$gcc system_testc -o system_test [root@localhost test]$/system_test hello [root@localhost test]$ 2)popen(char command,char type) popen()会调用fork()产生 子历程,然后从子历程中调用/bin/sh -c来履行 参数command的指令。参数type可应用 “r”代表读取,“w”代表写入。遵循此type值,popen()会建立 管道连到子历程的标准 输出设备 或标准 输入设备 ,然后返回一个文件指针。随后历程便可利用 此文件指针来读取子历程的输出设备 或是写入到子历程的标准 输入设备 中。此外,所有应用 文 件指针(FILE) *** 作的函数也都可以应用 ,除了fclose()以外。 返回值:若成功 则返回文件指针,否则返回NULL,差错 原因存于errno中。

1、Linux的shell调用C语言是一定的,就像你在脚本中使用 echo "helloworld" | passwd --stdin user001一样啊。echo本身就是一个C语言的可行的二进制的可执行文件。

2、通过mac地址和硬盘序列号确定一台机器是可行的,通过将二者组合(比如mac地址+硬盘序列号)后,通过散列算法是可以得到一个唯一的16位的序列,但是这种算法是存在风险的,比如我更换了硬盘,那么对已你的系统来说还是以前的机器呢?还有就是mac地址是唯一的,但是硬盘的序列号不一定唯一,貌似现在的硬盘没有一个统一的编号管理机构,都是硬盘厂商自己对硬盘编号。

使用fork加pipe就加以了,

首先创建一个管道

int fd[2];

pipe(fd); //创建一个管道,fd[0] 是读取端,fd[1]是写入端

然后fork

int pid=0;

pid=fork();

if( pid < 0 )

{

/fork失败,退出处理/

}

else if( pid == 0 ) /子进程/

{

dup2( fd[1], 1 ); /把标准输出重定向到管理的写入端, 这个当你往标准输出写东西的时候,就相当于写到管道里了,这一步不是必须的,也可以输出的时候直接指定往fd[1]里写,我这是为了方便,这样写后面直接使用printf函数就能往管道里写东西/

sleep(1); //暂停一秒再输密码,防止你的system还没运行到输密码的地方就输出了,那system调用的命令就收不到了

printf("你的密码\n");

fflush(stdout); //刷新缓冲区,这样printf才能马上生效

//要是有其他的东西要输出也可以继续printf

return(0);//子进程要做的事做完了直接退出就行了

}

else //父进程

{

//这里面就是你原来的处理,但是要改一下,把system调用的命令的标准输入重定向到fd[0],这样子进程往fd[1]里写入密码,由于system是从标准输入读取你的输入的,重定向到fd[0]就能读取子进程写到fd[1]里的密码了

dup2( fd[0], 0 ); //把标准输入重定到fd[0]

system("");

//也可以不使用dup2来重定向,可以直接在system命令里重定向

// 如 sprintf(cmd,"你要调的命令 <&%d", fd[0]);

// system(cmd);

}

以上就是关于linux下如何用c语言调用shell命令全部的内容,包括:linux下如何用c语言调用shell命令、我有两个问题: 1.linux,shell脚本可以调用c程序吗 2.通过网卡mac地址与硬盘序列号怎么确定一个唯一序列、C程序用system调用执行shell命令,shell终端提示要输入密码,怎么做等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/zz/10122821.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-05-05
下一篇2023-05-05

发表评论

登录后才能评论

评论列表(0条)

    保存