2。进程的标志
上面我们知道了进程都有一个ID,那么我们怎么得到进程的ID呢?系统调用getpid可 以得到进程的ID,而getppid可以得到父进程(创建调用该函数进程的进程)的ID.
#i nclude pid_t getpid(void);
pid_t getppid(void);
进程是为程序服务的,而程序是为了用户服务的.系统为了找到进程的用户名,还为进程和 用户建立联系.这个用户称为进程的所有者.相应的每一个用户也有一个用户ID.通过系统 调用getuid可以得到进程的所有者的ID.由于进程要用到一些资源,而Linux对系统资源是 进行保护的,为了获取一定资源进程还有一个有效用户ID.这个ID和系统的资源使用有关 ,涉及到进程的权限. 通过系统调用geteuid我们可以得到进程的有效用户ID. 和用户ID 相对应进程还有一个组ID和有效组ID系统调用getgid和getegid可以分别得到组ID和有效 组ID
#i nclude #i nclude uid_t getuid(void);
uid_t geteuid(void); gid_t getgid(void); git_t getegid(void);
有时候我们还会对用户的其他信息感兴趣(登录名等等),这个时候我们可以调用getpwuid 来得到.
struct passwd {
char *pw_name; /* 登录名称 */ char *pw_passwd; /* 登录口令 */ uid_t pw_uid; /* 用户ID */ gid_t pw_gid; /* 用户组ID */
char *pw_gecos; /* 用户的真名 */ char *pw_dir; /* 用户的目录 */
char *pw_shell; /* 用户的SHELL */ };
#i nclude #i nclude
struct passwd *getpwuid(uid_t uid);
下面我们学习一个实例来实践一下上面我们所学习的几个函数:
#i nclude #i nclude #i nclude #i nclude
int main(int argc,char **argv) {
pid_t my_pid,parent_pid; uid_t my_uid,my_euid; gid_t my_gid,my_egid; struct passwd *my_info; my_pid=getpid(); parent_pid=getppid(); my_uid=getuid(); my_euid=geteuid(); my_gid=getgid();
my_egid=getegid();
my_info=getpwuid(my_uid);
printf(\ printf(\ printf(\
printf(\ printf(\ ID:%ld\\n\
printf(\ if(my_info)
{
printf(\ printf(\ printf(\ printf(\ printf(\ printf(\ printf(\ }
}
3。进程的创建
创建一个进程的系统调用很简单.我们只要调用fork函数就可以了.
#i nclude
pid_t fork();
当一个进程调用了fork以后,系统会创建一个子进程.这个子进程和父进程不同的地方只
有他的进程ID和父进程ID,其他的都是一样.就象符进程克隆(clone)自己一样.当然创建 两个一模一样的进程是没有意义的.为了区分父进程和子进程,我们必须跟踪fork的返回 值. 当fork掉用失败的时候(内存不足或者是用户的最大进程数已到)fork返回-1,否则f ork的返回值有重要的作用.对于父进程fork返回子进程的ID,而对于fork子进程返回0.我
们就是根据这个返回值来区分父子进程的. 父进程为什么要创建子进程呢?前面我们已经 说过了Linux是一个多用户操作系统,在同一时间会有许多的用户在争夺系统的资源.有时 进程为了早一点完成任务就创建子进程来争夺资源. 一旦子进程被创建,父子进程一起从 fork处继续执行,相互竞争系统的资源.有时候我们希望子进程继续执行,而父进程阻塞直 到子进程完成任务.这个时候我们可以调用wait或者waitpid系统调用.
#i nclude
#i nclude
pid_t wait(int *stat_loc);
pid_t waitpid(pid_t pid,int *stat_loc,int options);
wait系统调用会使父进程阻塞直到一个子进程结束或者是父进程接受到了一个信号.如果 没有父进程没有子进程或者他的子进程已经结束了wait回立即返回.成功时(因一个子进 程结束)wait将返回子进程的ID,否则返回-1,并设置全局变量errno.stat_loc是子进程的 退出状态.子进程调用exit,_exit 或者是return来设置这个值. 为了得到这个值Linux定 义了几个宏来测试这个返回值.
WIFEXITED:判断子进程退出值是非0
WEXITSTATUS:判断子进程的退出值(当子进程退出时非0). WIFSIGNALED:子进程由于有没有获得的信号而退出.
WTERMSIG:子进程没有获得的信号号(在WIFSIGNALED为真时才有意义).
waitpid等待指定的子进程直到子进程返回.如果pid为正值则等待指定的进程(pid).如果 为0则等待任何一个组ID和调用者的组ID相同的进程.为-1时等同于wait调用.小于-1时等
待任何一个组ID等于pid绝对值的进程. stat_loc和wait的意义一样. options可以决定 父进程的状态.可以取两个值
WNOHANG:父进程立即返回当没有子进程存在时.
WUNTACHED:当子进程结束时waitpid返回,但是子进程的退出状态不可得到.
父进程创建子进程后,子进程一般要执行不同的程序.为了调用系统程序,我们可以使用系 统调用exec族调用.exec族调用有着5个函数.
#i nclude
int execl(const char *path,const char *arg,...); int execlp(const char *file,const char *arg,...); int execle(const char *path,const char *arg,...); int execv(const char *path,char *const argv[]); int execvp(const char *file,char *const argv[]):
exec族调用可以执行给定程序.关于exec族调用的详细解说可以参考系统手册(man exec l). 下面我们来学习一个实例.注意编译的时候要加 -lm以便连接数学函数库.
#i nclude #i nclude #i nclude #i nclude #i nclude #i nclude
void main(void) {
pid_t child; int status;
printf(\ if((child=fork())==-1)
{
printf(\ exit(1); }
else if(child==0)
{
int i;
printf(\ for(i=0;i #i nclude #i nclude #i nclude #i nclude #i nclude #i nclude
/* Linux 的默任个人的邮箱地址是 /var/spool/mail/用户的登录名 */ #define MAIL \ /* 睡眠10秒钟 */ #define SLEEP_TIME 10
main(void) {
pid_t child;
if((child=fork())==-1)
{
printf(\ exit(1);
}
else if(child>0) while(1);
if(kill(getppid(),SIGTERM)==-1) {
printf(\ exit(1); } {
int mailfd; while(1) {
if((mailfd=open(MAIL,O_RDONLY))!=-1) {
fprintf(stderr,\ close(mailfd); }
sleep(SLEEP_TIME); } } }
你可以在默认的路径下创建你的邮箱文件,然后测试一下这个程序.当然这个程序还有很 多地方要改善的.我们后面会对这个小程序改善的,再看我的改善之前你可以尝试自己改 善一下.比如让用户指定邮相的路径和睡眠时间等等.相信自己可以做到的.动手吧,勇敢 的探险者.
好了进程一节的内容我们就先学到这里了.进程是一个非常重要的概念,许多的程序都会用子进程.创建一个子进程是每一个程序员的基本要求!
Linux下文件的操作 (第三章)
作者:Hoyt Email:hoytluo@21cn.com 前言:
我们在这一节将要讨论linux下文件操作的各个函数. 文件的创建和读写 文件的各个属性 目录文件的操作 管道文件
----------------------------------------------------------------------------
1。文件的创建和读写
我假设你已经知道了标准级的文件操作的各个函数(fopen,fread,fwrite等等).当然