第八章代码运行测试
exec1
打印当前目录下所有文件的相关信息,命令等价于ls -l。
它的操作过程是调用execvp函数:
头文件:
#include<unistd.h>
定义函数:
int execvp(const char file ,char const argv []);
函数说明:
execvp()会从PATH 环境变量所指的目录中查找符合参数file 的文件名,找到后便执行该文件,然后将第二个参数argv传给该欲执行的文件。
返回值:
如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno 中。
exec2
运行结果同exec1相同。
代码也是一模一样的,只不过一个是“ls”,一个是数组中第一个元素,两者显然是等价的。
exec3
运行结果一样。
调用了execlp函数:
头文件:
#include<unistd.h>
定义函数:
int execlp(const char * file,const char * arg,....);
函数说明:
execlp()会从PATH 环境变量所指的目录中查找符合参数file的文件名,找到后便执行该文件,然后将第二个以后的参数当做该文件的argv[0]、argv[1]……,最后一个参数必须用空指针(NULL)作结束。如果用常数0来表示一个空指针,则必须将它强制转换为一个字符指针,否则将它解释为整形参数,如果一个整形数的长度与char * 的长度不同,那么exec函数的实际参数就将出错。如果函数调用成功,进程自己的执行代码就会变成加载程序的代码,execlp()后边的代码也就不会执行了.
返回值:
如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno 中。
forkdemo1
先打印子进程pid,休息一秒后根据fork()一次调用两次返回的性质,父进程返回子进程pid,子进程返回0;
forkdemo2
因为调用了两次fork()故产生了2*2共四个进程。
forkdemo3
调用fork()使得父进程打印子进程pid,子进程打印自身pid。
forkdemo4
父进程打印子进程pid,子进程打印两个pid。
forkgdb
两个进程依次打印一堆数字,意义不明。
psh1
输入所要执行的命令,然后用execvp函数执行它。
psh2
通过循环判断持续运行上一个代码。
testbuf1
输出“hello”,然后清空标准输出,然后进入死循环while(1)。
testbuf2
输出一个hello然后换行,进入死循环。
fflush(stdout)作用是清空缓存区。主要是防止多次输出出现上一个缓存还未输出就立马进入下一个输出导致输出错误。
testbuf3
fprintf作用是格式化输出到流,流可以是文件或标准输入、输出和错误。
fprintf(stderr,"%s",S);
fprintf(stdout,"%s",S);
printf("%s",S);
三条语句效果基本一致,不同处在重定向时,只有第一条会在屏幕输出,其余都会输出到重定向的文件。
此外从输出情况可看出标准错误会最优先显示于屏幕上。
testpid
打印当前进程和其父进程的pid。
testpp
该代码无法运行,错误显示为:段错误 (核心已转储)
首先要明确的是:
**pp为原整形数
*pp为存放这个数的地址
pp为存放这个数的地址的地址
pp[0]等价于*pp,malloc函数返回所开辟空间的首地址,即将首地址赋给存放这个数的地址,这样操作本身是没有问题的,但因为此时pp尚未开辟空间,*pp也没有空间,不可以跳过pp对*pp开辟空间,所以在之前加入:pp=malloc(4)即可。
testsystem
调用了system函数:
头文件:
#include<stdlib.h>
定义函数:
int system(char* command);
函数说明:
发出一个dos命令,同execvp函数类似。
不过注意的是这里数组的两个值对应的是两条命令。
waitdemo1
调用wait函数:
头文件:
#include<sys/types.h>
#include<sys/wait.h>
定义函数:
pid_t wait(int* status);
函数说明:
wait()会暂时停止目前进程的执行,直到有信号来到或子进程结束。如果在调用wait()时子进程已结束,则wait会立即返回子进程结束状态值。子进程的结束状态值会由status返回,而子进程的进程识别码也会一并返回。如果不在意结束状态值,则参数status可设置为NULL。子进程的结束状态值参考waitpid()。
返回值:
成功则返回子进程pid,失败返回-1,原因存于errno中。
本程序中父进程会等到子进程运行完时再继续运行。
waitdemo2
同waitdemo1相似,区别在于会分三部分打印子进程结束状态值。
argv文件夹
类似于psh1,不过只能输入一个指令。
env文件夹
environ
getenu()获取环境变量,setenu()修改环境变量。
environvar
打印环境变量列表。
fifo文件夹
consumer
memset函数:作用是在一段内存块中填充某个给定的值,它是对较大的结构体或数组进行清零操作的一种最快方法。
头文件:
#include<memory.h>或#include<string.h>
定义函数:
void memset(void s, int ch, size_t n);
函数说明:
将s中前n个字节替换为ch并返回s。
本程序用来打印所读文件的字节个数,因为无该文件故返回失败结果。
producer
access函数
头文件:
#include<unistd.h>
定义函数:
int access(const char *pathname ,int mode);
函数说明:
access()会检查是否可读/写某一已存在文件。参数mode有几种情况组合,R_OK,W_OK,X_OK,F_OK。R_OK,W_OK与X_OK用来检查文件是否具有读取、写入和执行的权限。F_OK用来判断该文件是否存在。由于access()只作权限的核查,而不理会文件形态或文件内容,因此,如果一目录表示为“可写入”,表示可以在该目录下建立新文件等操作,而非意味此目录可以被当作文件处理。
返回值:
若所有要查的权限均通过返回0,否则返回-1。
mkfifo函数
头文件:
#include<sys/types.h>
#include<sys/stat.h>
定义函数:
int mkfifo(const char *pathname,mode_t mode);
函数说明:
mkfifo()会依参数pathname建立特殊的FIFO文件,该文件必须不存在,而参数mode作为该文件的权限。
返回值:
如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno 中。
该程序的不同之处在于创建了一个fifo管道文件,打开原文件的操作变为打开fifo管道文件。