Linux操作系统下c语言编程入门 件的具体实现。一个程序可能有许多进程,而每一个进程又可以有许多子进程依次循环 下去而产生子孙进程.当程序被系统调用到内存以后,系统会给程序分配一定的资源(内 存设备等等)然后进行一系列的复杂操作,使程序变成进程以供系统调用.在系统里面只 有进程没有程序为了区分各个不同的进程,系统给每一个进程分配了一个ID(就象我们的 身份证)以便识别.为了充分的利用资源,系统还对进程区分了不同的状态将进程分为新 建运行,阻塞就绪和完成五个状态.新建表示进程正在被创建,运行是进程正在运行,阻 塞是进程正在等待某一个事件发生就绪是表示系统正在等待CPU来执行命令,而完成表示 进程已经结束了系统正在回收资源.关于进程五个状态的详细解说我们可以看《操作系 统》上面有详细的解说 2。进程的标志 上面我们知道了进程都有一个ID,那么我们怎么得到进程的ID呢?系统调用 getpid可 以得到进程的ID,而 getppid可以得到父进程(创建调用该函数进程的进程)的ID. #include <unistd> pid_t getpid(void); pid_t getppid(void) 进程是为程序服务的而程序是为了用户服务的系统为了找到进程的用户名还为进程和 用户建立联系这个用户称为进程的所有者.相应的每一个用户也有一个用户ID通过系统 调用 getuid可以得到进程的所有者的ID由于进程要用到一些资源,而Lnux对系统资源是 进行保护的,为了获取一定资源进程还有一个有效用户ID这个ID和系统的资源使用有关 ,涉及到进程的权限.通过系统调用 geteuid我们可以得到进程的有效用户ID.和用户ID 相对应进程还有一个组ID和有效组ID系统调用 getgid和 getegid可以分别得到组ID和有效 组ID #include <sys/types.h> uid-t getuid (void) uid_t geteuid(void gid-t getgid (void) git_t getegid(void) 有时候我们还会对用户的其他信息感兴趣(登录名等等)这个时候我们可以调用 getpwul d来得到. struct passwd t char* pw__name;/*登录名称* char* pw_passwd;/*登录口令* uid-t pw_uid;/*用户ID* gid-t pw_ gid;/*用户组ID* char* pw gecos;/*用户的真名* char*pwdr;/*用户的目录* char* pw_ shel;/*用户的 SHELL* struct passwd *getpwuid(uid_t uid) 第6页共84页
Linux 操作系统下 c 语言编程入门 件的具体实现. 一个程序可能有许多进程,而每一个进程又可以有许多子进程.依次循环 下去,而产生子孙进程. 当程序被系统调用到内存以后,系统会给程序分配一定的资源(内 存,设备等等)然后进行一系列的复杂操作,使程序变成进程以供系统调用.在系统里面只 有进程没有程序,为了区分各个不同的进程,系统给每一个进程分配了一个 ID(就象我们的 身份证)以便识别. 为了充分的利用资源,系统还对进程区分了不同的状态.将进程分为新 建,运行,阻塞,就绪和完成五个状态. 新建表示进程正在被创建,运行是进程正在运行,阻 塞是进程正在等待某一个事件发生,就绪是表示系统正在等待 CPU 来执行命令,而完成表示 进程已经结束了系统正在回收资源. 关于进程五个状态的详细解说我们可以看《操作系 统》上面有详细的解说。 2。进程的标志 上面我们知道了进程都有一个 ID,那么我们怎么得到进程的 ID 呢?系统调用 getpid 可 以得到进程的 ID,而 getppid 可以得到父进程(创建调用该函数进程的进程)的 ID. #include <unistd> pid_t getpid(void); pid_t getppid(void); 进程是为程序服务的,而程序是为了用户服务的.系统为了找到进程的用户名,还为进程和 用户建立联系.这个用户称为进程的所有者.相应的每一个用户也有一个用户 ID.通过系统 调用 getuid 可以得到进程的所有者的 ID.由于进程要用到一些资源,而 Linux 对系统资源是 进行保护的,为了获取一定资源进程还有一个有效用户 ID.这个 ID 和系统的资源使用有关 ,涉及到进程的权限. 通过系统调用 geteuid 我们可以得到进程的有效用户 ID. 和用户 ID 相对应进程还有一个组 ID 和有效组 ID 系统调用 getgid 和 getegid 可以分别得到组 ID 和有效 组 ID #include <unistd> #include <sys/types.h> uid_t getuid(void); uid_t geteuid(void); gid_t getgid(void); git_t getegid(void); 有时候我们还会对用户的其他信息感兴趣(登录名等等),这个时候我们可以调用 getpwui d 来得到. 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 */ }; #include <pwd.h> #include <sys/types.h> struct passwd *getpwuid(uid_t uid); 第 6 页 共 84 页
Linux操作系统下c语言编程入门 下面我们学习一个实例来实践一下上面我们所学习的几个函数: #include <unistd. h> #include #indlude <stdio. h> int main(int argc, char **argv) pid_t my_pid, parent_pid; uid-t my_uid, my _euid struct passwd * my_info my_ pid=getpido: parent_pid=getppido: my_uid=getuidoz my_euid=geteuido my_gid=getgido my_egid=getegido my_info=getpwuid(my_uid); printf(" Process ID: %ld\n, my_pid); printf("Parent ID: %ld\n,parent_pid); printf("User ID: %ld\n",my_uid); printf(Effective User ID: %ld\n", my_euid); printf( Effective Group ID: ld\n my_egid): if(my_info printf( My Login Name: %s\n", my_info->pw_name); printf( My Password %s\n",my_info->pw_passwd); printf( My User ID: %ldn, my_info->pw_uid); printf( My Group ID: %ld\n", my_info->pw_gid); printf My Real Name: %s\n", my_info->pw_gecos printf( My Work Shell: %sn", my_info->pw_ shell) 3。进程的创建 创建一个进程的系统调用很简单.我们只要调用fork函数就可以了 #include <unistd. h> pid_t forko: 当一个进程调用了fok以后,系统会创建一个子进程这个子进程和父进程不同的地方只 有他的进程ID和父进程ID,其他的都是一样就象符进程克隆(one)自己一样当然创建 两个一模一样的进程是没有意义的为了区分父进程和子进程我们必须跟踪fork的返回 值.当fork掉用失败的时候(内存不足或者是用户的最大进程数己到)ok返回-1,否则f 第7页共84页
Linux 操作系统下 c 语言编程入门 下面我们学习一个实例来实践一下上面我们所学习的几个函数: #include <unistd.h> #include <pwd.h> #include <sys/types.h> #include <stdio.h> 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(“Process ID:%ld\n”,my_pid); printf(“Parent ID:%ld\n”,parent_pid); printf(“User ID:%ld\n”,my_uid); printf(“Effective User ID:%ld\n”,my_euid); printf(“Group ID:%ld\n”,my_gid); printf(“Effective Group ID:%ld\n”,my_egid): if(my_info) { printf(“My Login Name:%s\n” ,my_info->pw_name); printf(“My Password :%s\n” ,my_info->pw_passwd); printf(“My User ID :%ld\n”,my_info->pw_uid); printf(“My Group ID :%ld\n”,my_info->pw_gid); printf(“My Real Name:%s\n” ,my_info->pw_gecos); printf(“My Home Dir :%s\n”, my_info->pw_dir); printf(“My Work Shell:%s\n”, my_info->pw_shell); } } 3。进程的创建 创建一个进程的系统调用很简单.我们只要调用 fork 函数就可以了. #include <unistd.h> pid_t fork(); 当一个进程调用了 fork 以后,系统会创建一个子进程.这个子进程和父进程不同的地方只 有他的进程 ID 和父进程 ID,其他的都是一样.就象符进程克隆(clone)自己一样.当然创建 两个一模一样的进程是没有意义的.为了区分父进程和子进程,我们必须跟踪 fork 的返回 值. 当 fork 掉用失败的时候(内存不足或者是用户的最大进程数已到)fork 返回-1,否则 f 第 7 页 共 84 页
Linux操作系统下c语言编程入门 ork的返回值有重要的作用对于父进程fork返回子进程的ID,而对于fork子进程返回0我 们就是根据这个返回值来区分父子进程的.父进程为什么要创建子进程呢?前面我们己经 说过了Lnu是一个多用户操作系统在同一时间会有许多的用户在争夺系统的资源有时 进程为了早一点完成任务就创建子进程来争夺资源.一旦子进程被创建父子进程一起从 fork处继续执行,相互竞争系统的资源有时候我们希望子进程继续执行而父进程阻塞直 到子进程完成任务这个时候我们可以调用wait或者 waitpid系统调用. #include <sys/ wait.h> pid_t wait(int stat_loc); dt waitpid(pid_t pid int stat_loc, int options); wat系统调用会使父进程阻塞直到一个子进程结束或者是父进程接受到了一个信号如果 没有父进程没有子进程或者他的子进程已经结束了wat回立即返回成功时(因一个子进 程结束wait将返回子进程的ID,否则返回-1,并设置全局变量erno. stat loc是子进程的 退出状态.子进程调用exit,ext或者是 return来设置这个值.为了得到这个值 Linux定 义了几个宏来测试这个返回值 WIFEXITED:判断子进程退出值是非0 WEXITSTATUS:判断子进程的退出值(当子进程退出时非0). WIFSIGNALED:子进程由于有没有获得的信号而退出 WTERMSIG:子进程没有获得的信号号(在 WIFSIGNALED为真时才有意义) waitpid等待指定的子进程直到子进程返回如果pid为正值则等待指定的进程(pid)如果 为0则等待任何一个组ID和调用者的组ID相同的进程为-1时等同于wat调用小于-1时等 待任何一个组ID等于pd绝对值的进程. stat loc和wat的意义一样. options可以决定 父进程的状态可以取两个值 WNOHANG:父进程立即返回当没有子进程存在时. WUNTACHE D:当子进程结束时 waitpid返回,但是子进程的退出状态不可得到. 父进程创建子进程后,子进程一般要执行不同的程序为了调用系统程序我们可以使用系 统调用exec族调用exec族调用有着5个函数 #include <unistd. h> int execl(const char *path, const char *arg int execl(const char *file, const char *arg,.) int execle(const char *path, const char *arg,; int execv(const char *path, char *const argv D): int execvp(const char *file, char *const argv): exec族调用可以执行给定程序关于exec族调用的详细解说可以参考系统手册( man exec l).下面我们来学习一个实例注意编译的时候要加-m以便连接数学函数库 #include <unistd. h> #include <sys/types.h> #include <sys/wait.h> #include <errno. h> void main( void 第8页共84页
Linux 操作系统下 c 语言编程入门 ork 的返回值有重要的作用.对于父进程 fork 返回子进程的 ID,而对于 fork 子进程返回 0.我 们就是根据这个返回值来区分父子进程的. 父进程为什么要创建子进程呢?前面我们已经 说过了 Linux 是一个多用户操作系统,在同一时间会有许多的用户在争夺系统的资源.有时 进程为了早一点完成任务就创建子进程来争夺资源. 一旦子进程被创建,父子进程一起从 fork 处继续执行,相互竞争系统的资源.有时候我们希望子进程继续执行,而父进程阻塞直 到子进程完成任务.这个时候我们可以调用 wait 或者 waitpid 系统调用. #include <sys/types.h> #include <sys/wait.h> 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:父进程立即返回当没有子进程存在时. WUNTACHE D:当子进程结束时 waitpid 返回,但是子进程的退出状态不可得到. 父进程创建子进程后,子进程一般要执行不同的程序.为了调用系统程序,我们可以使用系 统调用 exec 族调用.exec 族调用有着 5 个函数. #include <unistd.h> 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 以便连接数学函数库. #include <unistd.h> #include <sys/types.h> #include <sys/wait.h> #include <stdio.h> #include <errno.h> #include <math.h> void main(void) { pid_t child; 第 8 页 共 84 页
Linux操作系统下c语言编程入门 int status printf( This will demostrate how to get child status\n) if(child=forko)==-1) printf( Fork Error: %s\n" strerror(errno); else if(child==O) int i; printf( I am the child: %ld\n", getpido): for(=0;i<1000000H++)sin(); printf('I exit with %d\n i while(((child=wait(&status))==-1)&(errno==EINTR)) if(child==-1) printf( Wait Error: %s\n strerror(erno)); else if(lstatus) printf("Child %ld terminated normally return status is zero\n else if(WIFEXITED(status) printf( Child %ld terminated normally return status is d\n child, WEXITSTATUS(status)); else if(WIFSIGNALED(status)) printf( Child %ld terminated due to signal %d znot caught \n child, WTERMSIG(status)) strerror函数会返回一个指定的错误号的错误信息的字符串 4。守护进程的创建 如果你在DOS时代编写过程序,那么你也许知道在DOS下为了编写一个常驻内存的程序 我们要编写多少代码了相反如果在Lnux下编写一个”常驻内存”的程序却是很容易的我 们只要几行代码就可以做到.实际上由于 Linux是多任务操作系统我们就是不编写代码 也可以把一个程序放到后台去执行的我们只要在命令后面加上&符号 SHELL就会把我们的 程序放到后台去运行的.这里我们”开发”一个后台检查邮件的程序.这个程序每个一个指 定的时间回去检查我们的邮箱,如果发现我们有邮件了,会不断的报警(通过机箱上的小喇 叭来发出声音)后面有这个函数的加强版本加强版本 后台进程的创建思想:首先父进程创建一个子进程然后子进程杀死父进程(是不是很无 情?).信号处理所有的工作由子进程来处理 #indlude <unistd.h> #include <sys/stat h #include <stdio. h> 第9页共84页
Linux 操作系统下 c 语言编程入门 int status; printf(“This will demostrate how to get child status\n”); if((child=fork())==-1) { printf(“Fork Error :%s\n”,strerror(errno)); exit(1); } else if(child==0) { int i; printf(“I am the child:%ld\n”,getpid()); for(i=0;i<1000000;i++) sin(i); i=5; printf(“I exit with %d\n”,i); exit(i); } while(((child=wait(&status))==-1)&(errno==EINTR)); if(child==-1) printf(“Wait Error:%s\n”,strerror(errno)); else if(!status) printf(“Child %ld terminated normally return status is zero\n”, child); else if(WIFEXITED(status)) printf(“Child %ld terminated normally return status is %d\n”, child,WEXITSTATUS(status)); else if(WIFSIGNALED(status)) printf(“Child %ld terminated due to signal %d znot caught\n”, child,WTERMSIG(status)); } strerror 函数会返回一个指定的错误号的错误信息的字符串. 4。守护进程的创建 如果你在 DOS 时代编写过程序,那么你也许知道在 DOS 下为了编写一个常驻内存的程序 我们要编写多少代码了.相反如果在 Linux 下编写一个”常驻内存”的程序却是很容易的.我 们只要几行代码就可以做到. 实际上由于 Linux 是多任务操作系统,我们就是不编写代码 也可以把一个程序放到后台去执行的.我们只要在命令后面加上&符号 SHELL 就会把我们的 程序放到后台去运行的. 这里我们”开发”一个后台检查邮件的程序.这个程序每个一个指 定的时间回去检查我们的邮箱,如果发现我们有邮件了,会不断的报警(通过机箱上的小喇 叭来发出声音). 后面有这个函数的加强版本加强版本 后台进程的创建思想: 首先父进程创建一个子进程.然后子进程杀死父进程(是不是很无 情?). 信号处理所有的工作由子进程来处理. #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <stdio.h> 第 9 页 共 84 页
Linux操作系统下c语言编程入门 #include <errno. h> #include <fcntl. h> /* Linux的默任个人的邮箱地址是/ar/spo/ma用户的登录名* #define MAIL /var/spool/ mail/ hoyt /*睡眠10秒钟*/ #define sleEp Time 10 main(void) pid_t child if(child=forko==-1) printf( Fork Error: %sn"strerror(errno)): exit(1); else if(child>o) while(1); if(kill (getppidO, SIGTERM)==-1) printf( Kill Parent Error: %s\nstrerror(errno)); while(1) {f{fd} if((mailfd=open(MAIL, O_RDONLY)I=-1) fprintf(stderr, " %S\007) sleep(sLEEP_TIMe) }} 你可以在默认的路径下创建你的邮箱文件然后测试一下这个程序当然这个程序还有很 多地方要改善的我们后面会对这个小程序改善的,再看我的改善之前你可以尝试自己改 善一下,比如让用户指定邮相的路径和睡眠时间等等相信自己可以做到的.动手吧,勇敢 的探险者 好了进程一节的内容我们就先学到这里了进程是一个非常重要的概念许多的程序都会 用子进程创建一个子进程是每一个程序员的基本要求! 第10页共84页
Linux 操作系统下 c 语言编程入门 #include <errno.h> #include <fcntl.h> #include <signal.h> /* Linux 的默任个人的邮箱地址是 /var/spool/mail/用户的登录名 */ #define MAIL “/var/spool/mail/hoyt” /* 睡眠 10 秒钟 */ #define SLEEP_TIME 10 main(void) { pid_t child; if((child=fork())==-1) { printf(“Fork Error:%s\n”,strerror(errno)); exit(1); } else if(child>0) while(1); if(kill(getppid(),SIGTERM)==-1) { printf(“Kill Parent Error:%s\n”,strerror(errno)); exit(1); } { int mailfd; while(1) { if((mailfd=open(MAIL,O_RDONLY))!=-1) { fprintf(stderr,”%s”,”\007”); close(mailfd); } sleep(SLEEP_TIME); } } } 你可以在默认的路径下创建你的邮箱文件,然后测试一下这个程序.当然这个程序还有很 多地方要改善的.我们后面会对这个小程序改善的,再看我的改善之前你可以尝试自己改 善一下.比如让用户指定邮相的路径和睡眠时间等等.相信自己可以做到的.动手吧,勇敢 的探险者. 好了进程一节的内容我们就先学到这里了.进程是一个非常重要的概念,许多的程序都会 用子进程.创建一个子进程是每一个程序员的基本要求! 第 10 页 共 84 页