昨晚研究了一个通宵,关于fork()和execlp()函数,基本上用fork函数实现管道,用execlp来实现重定向。不过它们两个实现功能的方法都非常奇怪。
fork函数是让程序创建一个跟自己一模一样的副本,就跟当下流行的很多网络游戏中副本的概念差不多,昨晚在练习的时候忽然感到,这玩意儿又有点像WEB 编程中的表单自提交。在同一个程序里面写两套方案,运行时让其中的一套(安排在fork>=1的分支结构中)调用来自自身代码文件中的另一套方案 (安排在fork==0分支结构)乍一看这跟管道根本就挨不着边,我一开始也是觉得这样,就像一个进程又去调用了一个进程一样,不过另外调用的进程又是本 身,大脑里一团浆糊一样。那么请看代码吧:[code]/**//*
============================================================================
Name : fork_example.c
Author : newflypig
Version :
Copyright : Your copyright notice
Description : Hello World in C, Ansi-style
============================================================================
*/
#include
#include
#include
#include
#include
#include
int main(void) ...{
int i=1;
printf("in the begining,the value=%d ",i);
switch(fork())...{
case -1:
fprintf(stderr,"%s ","fork error");
break;
case 0:
printf("child process start,at this time value=%d ",i);
i++;
printf("child process end,at this time value=%d ",i);
break;
default:
printf("parent process: value=%d ",i);
}
return 0;
}[/code]运行结果是这样的:
in the begining,the value=1
child process start,at this time value=1
child process end,at this time value=2
parent process: value=1
可 以看到父进程首先设置i=1然后调用子进程,子进程一开始就有了父进程的i值,然后子进程在自己的基础上将i++了,子进程结束时输出了i=2。当程序返 回父进程时,子进程的改变并没有影响父进程中i的值,i依然为1。这个fork()的功能仅仅如此,有谁会想到让这个函数在Linux最具特色的管道机制 中大显伸手呢。
花开两朵,各表一枝。
下面看看重定向的概念:
先做一个演示程序:[code]/**//*
============================================================================
Name : execlp_example.c
Author : newflypig
Version :
Copyright : Your copyright notice
Description : Hello World in C, Ansi-style
============================================================================
*/
#include
#include
#include
#include
#include
#include
int main(void) ...{
int filedes;
if((filedes=open("dd.txt",O_WRONLY|O_CREAT,S_IRUSR|S_IWUSR))<0)
fprintf(stderr,"%s ","open file error");
close(1);
dup(filedes);
close(filedes);
execlp("ls","-l",(char * )0);
return 0;
}[/code]这是一段输出重定向的程序,为了解释方便,首先了解一下Linux中标准输入输出文件,Linux中基本上所有的元素都可以理解为文件,包括文件夹,设备等。程序中,首先使用close(1)关闭标准输出端子,{/*当然如果你想输入重定向的话可以关闭标准输入端子,它的序列是0;序列为3的端子是异常抛出端子,一般可以让此端子与输出设备保持一致。*/}, 然后使用dup函数将main函数的第一步所创建的名为dd.txt的文件设置为此程序的输出设备。接着要关闭该文件的连接,释放资源锁以便让程序来对此 文件进行读写。最后一步调用execlp函数进行命令的执行,这里执行了一个ls的命令,让当前目录下的所有文件名输出,execlp函数是可变参数函 数,第一个参数需要设置系统环境变量中所能获取的命令文件,或者自己设置绝对路径的命令文件,最后一个参数必须设置为空,以便标记这个函数的参数已经设置 完成,中间的所有参数设置为第一个命令的参数,如此段程序其实是执行ls -l这个命令。这样就将输出信息不直接打印到终端而输出到dd.txt文本文件中,完成输出重定向。
回到管道的问题上来。
如果将 刚刚讲得重定向技术配合fork()进程产生函数,便可以实现管道的作用了。这里说一下思路,就不给具体代码了,也就是设置两个全局的文件变量,将子进程 的输出重定向到其中一个文件变量,因为是全局变量,这个文件变量就被子进程彻底修改了,然后将父进程的输入重定向到刚刚子进程的输出,以此来完成 Linux中的管道机制。
不知各位看观对本人关于Linux中的管道以及重定向问题的看法是否认同,昨晚就看这两个函数的API和源代码了。
综上所述,一个毋庸置疑的结论:Linux编程比Windows编程艰巨而有趣多了。以前从来不高兴碰C语言,认为C++和java此类OO语言才是最人性化,最舒服的编程语言,这次初探C,感受到是另一种源于代码和算法的舒服感。
zhenjiaseu 于 2008-08-11 22:23:38发表:
有点意思。。。
harly 于 2008-08-11 13:02:02发表:
“Linux中C语言的编程有两个Windows环境下根本无须考虑的问题,关于管道和重定向的概念。”,这句话是什么意思?你在写的是汉语吗??!
改成如下句是不是顺口了??
“Linux中的C语言编程有两个在Windows环境下无须考虑的概念,此即管道和重定向”
这么改,语法上是顺当了,也符合这狗屎作者的臭屁原作意思
可这意思是错的!windows下管道和重定向,还是很重要的,不能“无须考虑”!!
最看不惯这种在linux下写着从main开始的控制台程序还洋洋得意的小人!
harly 于 2008-08-11 12:44:07发表:
看你的第二个例子,这句
dup(filedes);
有什么意义吗?filedes是一个入参,dup的返回值你不要,你想干什么?真他妈的SB一个,写程序都玩无厘头,caonima的!