亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C > 正文

使用C語言的fork()函數在Linux中創建進程的實例講解

2020-01-26 14:32:32
字體:
來源:轉載
供稿:網友

在Linux中創建一個新進程的唯一方法是使用fork()函數。fork()函數是Linux中一個非常重要的函數,和以往遇到的函數有一些區別,因為fork()函數看起來執行一次卻返回兩個值。

fork()函數用于從已存在的進程中創建一個新進程。新進程稱為子進程,而園進程稱為父進程。使用fork()函數得到的子進程是父進程的一個復制品,它從父進程處繼承了整個進程的地址空間,包括進程的上下文、代碼段、進程堆棧、內存信息、打開的文件描述符、符號控制設定、進程優先級、進程組號、當前工作目錄、根目錄、資源限制和控制終端等,而子進程所獨有的只有它的進程號、資源使用和計時器等。

因為子進程幾乎是父進程的完全復制,所以父子兩進程會運行同一個程序。這就需要用一種方式來區分它們,并使它們照此運行,否則,這兩個進程不可能做不同的事。實際上是在父進程中執行fork()函數時,父進程會復制一個子進程,而且父子進程的代碼從fork()函數的返回開始分別在兩個地址空間中同時運行,從而使兩個進程分別獲得所屬fork()函數的返回值,其中在父進程中的返回值是子進程的進程號,而在子進程中返回0。因此,可以通過返回值來判斷該進程的父進程還是子進程。

同時可以看出,使用fork()函數的代價是很大的,它復制了父進程中的代碼段、數據段和堆棧段里的大部分內容,使得fork()函數的系統開銷比較大,而且執行速度也不是很快。

fork()函數語法

2016622171656749.jpg (495×165)

fork()函數出錯可能有兩種原因:
1、當前的進程數已經達到了系統規定的上限,這時errno的值被設置為EAGAIN
2、系統內存不足,這時errno的值被設置為ENOMEM

示例

下面的是csapp.h頭文件,后面的討論中均只用該頭文件來完成程序的編寫。 

/* $begin csapp.h */#ifndef __CSAPP_H__#define __CSAPP_H__#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <string.h>#include <ctype.h>#include <setjmp.h>#include <signal.h>#include <sys/time.h>#include <sys/types.h>#include <sys/wait.h>#include <sys/stat.h>#include <fcntl.h>#include <sys/mman.h>#include <errno.h>#include <math.h>#include <pthread.h>#include <semaphore.h>#include <sys/socket.h>#include <netdb.h>#include <netinet/in.h>#include <arpa/inet.h>/* Default file permissions are DEF_MODE & ~DEF_UMASK *//* $begin createmasks */#define DEF_MODE  S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH#define DEF_UMASK S_IWGRP|S_IWOTH/* $end createmasks *//* Simplifies calls to bind(), connect(), and accept() *//* $begin sockaddrdef */typedef struct sockaddr SA;/* $end sockaddrdef *//* Persistent state for the robust I/O (Rio) package *//* $begin rio_t */#define RIO_BUFSIZE 8192typedef struct {  int rio_fd;        /* Descriptor for this internal buf */  int rio_cnt;        /* Unread bytes in internal buf */  char *rio_bufptr;     /* Next unread byte in internal buf */  char rio_buf[RIO_BUFSIZE]; /* Internal buffer */} rio_t;/* $end rio_t *//* External variables */extern int h_errno;  /* Defined by BIND for DNS errors */ extern char **environ; /* Defined by libc *//* Misc constants */#define  MAXLINE   8192 /* Max text line length */#define MAXBUF  8192 /* Max I/O buffer size */#define LISTENQ 1024 /* Second argument to listen() *//* Our own error-handling functions */void unix_error(char *msg);void posix_error(int code, char *msg);void dns_error(char *msg);void app_error(char *msg);/* Process control wrappers */pid_t Fork(void);void Execve(const char *filename, char *const argv[], char *const envp[]);pid_t Wait(int *status);pid_t Waitpid(pid_t pid, int *iptr, int options);void Kill(pid_t pid, int signum);unsigned int Sleep(unsigned int secs);void Pause(void);unsigned int Alarm(unsigned int seconds);void Setpgid(pid_t pid, pid_t pgid);pid_t Getpgrp();/* Signal wrappers */typedef void handler_t(int);handler_t *Signal(int signum, handler_t *handler);void Sigprocmask(int how, const sigset_t *set, sigset_t *oldset);void Sigemptyset(sigset_t *set);void Sigfillset(sigset_t *set);void Sigaddset(sigset_t *set, int signum);void Sigdelset(sigset_t *set, int signum);int Sigismember(const sigset_t *set, int signum);/* Unix I/O wrappers */int Open(const char *pathname, int flags, mode_t mode);ssize_t Read(int fd, void *buf, size_t count);ssize_t Write(int fd, const void *buf, size_t count);off_t Lseek(int fildes, off_t offset, int whence);void Close(int fd);int Select(int n, fd_set *readfds, fd_set *writefds, fd_set *exceptfds,     struct timeval *timeout);int Dup2(int fd1, int fd2);void Stat(const char *filename, struct stat *buf);void Fstat(int fd, struct stat *buf) ;/* Memory mapping wrappers */void *Mmap(void *addr, size_t len, int prot, int flags, int fd, off_t offset);void Munmap(void *start, size_t length);/* Standard I/O wrappers */void Fclose(FILE *fp);FILE *Fdopen(int fd, const char *type);char *Fgets(char *ptr, int n, FILE *stream);FILE *Fopen(const char *filename, const char *mode);void Fputs(const char *ptr, FILE *stream);size_t Fread(void *ptr, size_t size, size_t nmemb, FILE *stream);void Fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream);/* Dynamic storage allocation wrappers */void *Malloc(size_t size);void *Realloc(void *ptr, size_t size);void *Calloc(size_t nmemb, size_t size);void Free(void *ptr);/* Sockets interface wrappers */int Socket(int domain, int type, int protocol);void Setsockopt(int s, int level, int optname, const void *optval, int optlen);void Bind(int sockfd, struct sockaddr *my_addr, int addrlen);void Listen(int s, int backlog);int Accept(int s, struct sockaddr *addr, socklen_t *addrlen);void Connect(int sockfd, struct sockaddr *serv_addr, int addrlen);/* DNS wrappers */struct hostent *Gethostbyname(const char *name);struct hostent *Gethostbyaddr(const char *addr, int len, int type);/* Pthreads thread control wrappers */void Pthread_create(pthread_t *tidp, pthread_attr_t *attrp,       void * (*routine)(void *), void *argp);void Pthread_join(pthread_t tid, void **thread_return);void Pthread_cancel(pthread_t tid);void Pthread_detach(pthread_t tid);void Pthread_exit(void *retval);pthread_t Pthread_self(void);void Pthread_once(pthread_once_t *once_control, void (*init_function)());/* POSIX semaphore wrappers */void Sem_init(sem_t *sem, int pshared, unsigned int value);void P(sem_t *sem);void V(sem_t *sem);/* Rio (Robust I/O) package */ssize_t rio_readn(int fd, void *usrbuf, size_t n);ssize_t rio_writen(int fd, void *usrbuf, size_t n);void rio_readinitb(rio_t *rp, int fd); ssize_t  rio_readnb(rio_t *rp, void *usrbuf, size_t n);ssize_t  rio_readlineb(rio_t *rp, void *usrbuf, size_t maxlen);/* Wrappers for Rio package */ssize_t Rio_readn(int fd, void *usrbuf, size_t n);void Rio_writen(int fd, void *usrbuf, size_t n);void Rio_readinitb(rio_t *rp, int fd); ssize_t Rio_readnb(rio_t *rp, void *usrbuf, size_t n);ssize_t Rio_readlineb(rio_t *rp, void *usrbuf, size_t maxlen);/* Client/server helper functions */int open_clientfd(char *hostname, int portno);int open_listenfd(int portno);/* Wrappers for client/server helper functions */int Open_clientfd(char *hostname, int port);int Open_listenfd(int port); #endif /* __CSAPP_H__ *//* $end csapp.h */

fork()函數示例一

#include "csapp.h" int main(){ pid_t pid; int x=1; pid=fork(); if(pid==0) {  printf("child :x=%d/n",++x);  exit(0); } printf("parent:x=%d/n",--x); exit(0);}

例如上面的程序,由于fork()函數比較特殊,執行一次,返回兩次。返回兩次分別是在父進程和子進程中各返回一個值,在子進程中返回為0,在父進程中返回進程ID,一般為正整數即非零。這樣就能根據返回值來確定其在哪個進程中了。如上面的程序,子進程中pid=0,所以執行if語句,子進程會共享父進程的文本/數據/bss段/堆以及用戶棧,子進程隨后正常終止并且返回碼為0,因此子進程不執行后續的共享代碼塊,因此本程序的輸出結果是

parent:x=0
child :x=2

fork()函數示例二

#include "csapp.h"int main(){ if(fork()==0) {  printf("a"); } else {  printf("b");  waitpid(-1,NULL,0); } printf("c"); exit(0);}

此程序是用來檢驗子進程與父進程的關系。同樣再次強調一遍,fork()函數用于新建子進程,子進程具有與父進程相同的用戶級虛擬地址空間,包括文本/數據/bss段/堆/用戶棧,子進程可以讀寫任意父進程打開的文件,它們的最大區別是它們有不同的PID。fork函數調用一次,返回兩次,一次在父進程中,其返回子進程的PID;在子進程中,fork返回0,因為子進程的PID總是非零的,返回值就提供了一個明確的方法來辨別是在父進程中執行還是在子進程中執行。waitpid()函數是等待子進程終止,若無錯誤,則返回值為正數。因為在在子進程中,fork()返回0,因此先輸出a,并且其共享父進程的代碼段,故又輸出c;而在父進程中,fork()返回值非零,所以執行else語句,故輸出bc。因此本程序的輸出結果為
acbc

fork()函數示例三

#include "csapp.h"int main(){ int x=1; if(fork()==0)  printf("printf1:x=%d/n",++x); printf("printf2:x=%d/n",--x); exit(0);}

本程序再次演示子進程與父進程的區別。程序中,在子進程中,子進程共享數據x=1,并且fork()返回0,因此if語句被執行,輸出printf1:x=2,接著共享后面一部分代碼段,因此再輸出printf2:x=1;而對于父進程,fork()返回非零,因此不會執行if語句段,而執行后面的代碼,即輸出printf2:x=0.因此本程序輸出結果為(子進程與父進程順序不唯一)

printf2:x=0printf1:x=2printf2:x=1

fork()函數示例四

#include "csapp.h"#define N 3int main(){ int status,i; pid_t pid;  for(i=0;i<N;i++)  if((pid=fork())==0) //新建子進程   exit(100+i);  while((pid=waitpid(-1,&status,0))>0) { //如果子進程是正常終止的,就返回進程的進程號PID  if(WIFEXITED(status)) //返回退出狀態   printf("child %d terminated normally with exit status =%d/n",pid,WEXITSTATUS(status));  else   printf("child %d erminated abnormally/n",pid); } if(errno!=ECHILD)  printf("waitpid error/n"); exit(0);}

本代碼主要是測試進程的終止,即waitpid案例程序。定義生成兩個進程,本例子是不按照特定順序來回收僵死子進程,本程序返回結果為

child 28693 terminated normally with exit status =100child 28694 terminated normally with exit status =101child 28695 terminated normally with exit status =102

fork()函數示例五

#include "csapp.h"#define N 2int main(){ int status,i; pid_t pid[N],retpid; for(i=0;i<N;i++)  if((pid[i]=fork())==0)   exit(100+i);//退出并返回狀態碼 i=0; while((retpid=waitpid(pid[i++],&status,0))>0) {  if(WIFEXITED(status))   printf("child %d terminated normally with exit status=%d/n",retpid,WEXITSTATUS(status));  else   printf("child %d terminated abnormally/n",retpid); } if(errno !=ECHILD)  printf("waitpid error!"); exit(0);}

按照創建進程的順序來回收這些僵死進程,注意程序中的pid[i++]是按序的標志,本程序運行結果為

child 29846 terminated normally with exit status=100child 29847 terminated normally with exit status=101

fork()函數示例六

#include "csapp.h"/*推測此程序會輸出什么樣的結果*/int main(){ int status; pid_t pid; printf("Hello/n"); pid=fork(); printf("%d/n",!pid); if(pid!=0) {  if(waitpid(-1,&status,0)>0) {   if(WIFEXITED(status)!=0)  printf("%d/n",WEXITSTATUS(status));  } } printf("Bye/n"); exit(2);}

首先父進程會輸出Hello,子進程新建成功,在子進程中,pid為0,輸出1,并且子進程無法執行if語句,但是子進程仍然可以輸出Bye,并且正常退出并返回狀態碼為2。而在父進程中,pid為非零的正數,因此先輸出0,然后執行if語句,由于子進程已經正常退出,故輸出狀態碼2,并且最后執行公共代碼塊,輸出Bye并正常退出。因此,總共輸出的結果如下所示:

Hello01Bye2Bye

當然,順序不唯一,還有一種可能的結果是

Hello1Bye02Bye

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
精品国内产的精品视频在线观看| 伊人亚洲福利一区二区三区| 国产欧美日韩精品专区| 精品久久香蕉国产线看观看亚洲| 欧美日韩国产中字| 免费不卡在线观看av| 亚洲日本成人网| 久久久久久久久久久久久久久久久久av| 日韩一区视频在线| 国产精品高潮视频| 欧美亚洲在线观看| 欧美在线视频观看免费网站| 欧美国产日韩二区| xvideos亚洲人网站| 国产精品亚洲第一区| 久久精品99国产精品酒店日本| 欧美日韩第一页| 最近2019中文字幕大全第二页| 久久免费精品日本久久中文字幕| 欧美一区二区色| 久久精品久久久久久国产 免费| 7777kkkk成人观看| 亚洲图片欧美日产| 青青青国产精品一区二区| 成人免费视频在线观看超级碰| 国产精品女人网站| 久久亚洲成人精品| 国模私拍一区二区三区| 亚洲国产三级网| 成人午夜激情网| 亚洲日韩中文字幕在线播放| 亚洲精品少妇网址| 国产精品aaaa| 亚洲天堂av高清| 亚洲一区二区三区四区在线播放| 国产在线观看不卡| 久久国产精品影片| 国产欧美最新羞羞视频在线观看| 68精品国产免费久久久久久婷婷| 亚洲人成网站在线播| 日韩精品视频在线观看免费| 国产成人一区二区在线| 亚洲老板91色精品久久| 国产精品极品美女粉嫩高清在线| 亚洲丝袜一区在线| 欧美激情视频在线观看| 成人av.网址在线网站| 国产在线a不卡| 国产成+人+综合+亚洲欧美丁香花| 国产精品欧美激情在线播放| 91九色视频导航| 成人精品久久一区二区三区| 美女啪啪无遮挡免费久久网站| 亚洲欧美一区二区三区四区| 国产视频自拍一区| 日韩免费av一区二区| 欧洲s码亚洲m码精品一区| 亚洲美女av电影| 亚洲精品xxx| 精品国产乱码久久久久久婷婷| 欧美理论电影在线播放| 亚洲国产成人av在线| 国产久一一精品| 久久久精品国产| 久久精品国产成人| 亚洲国产成人精品久久| 91免费人成网站在线观看18| 亚洲理论片在线观看| 欧美日韩国产精品一区二区三区四区| 久久久久成人精品| 国产精品xxxxx| 日韩av片永久免费网站| 欧美中文字幕在线播放| 国产小视频国产精品| 亚洲在线免费看| 欧美激情视频一区二区三区不卡| 中文字幕日韩专区| 久久天天躁夜夜躁狠狠躁2022| 久久久久久久久久久久久久久久久久av| 亚洲国产一区二区三区在线观看| 欧美成人午夜激情| 久久久久久亚洲精品中文字幕| 亚洲自拍高清视频网站| 色yeye香蕉凹凸一区二区av| 一本色道久久综合亚洲精品小说| 日韩欧美亚洲国产一区| 欧美国产日产韩国视频| 午夜精品一区二区三区在线视| 亚洲美女视频网| xvideos国产精品| 亚洲成色777777在线观看影院| 久久免费视频在线观看| 欧美色播在线播放| 国产精品成人va在线观看| 国产午夜精品全部视频播放| 91av在线精品| 51午夜精品视频| 91精品久久久久久久久久久久久| 一级做a爰片久久毛片美女图片| 国产欧美精品在线播放| 国产91久久婷婷一区二区| 黄网动漫久久久| 久久久综合免费视频| 久久精品亚洲国产| 久久久久久久97| 亚洲综合日韩中文字幕v在线| 久久久精品国产一区二区| 亚洲欧美成人在线| 清纯唯美日韩制服另类| 91精品国产免费久久久久久| 国产精品r级在线| 久久精品亚洲精品| 精品性高朝久久久久久久| 日本不卡视频在线播放| 国产91网红主播在线观看| 亚洲国产精品va在线| 亚洲天堂成人在线视频| 狠狠色狠色综合曰曰| 国产精品一区二区3区| 亚洲最新中文字幕| 国产区亚洲区欧美区| 欧美日韩国产综合新一区| 久久天天躁夜夜躁狠狠躁2022| 国产精品视频中文字幕91| 国产日产久久高清欧美一区| 日本伊人精品一区二区三区介绍| 久久成人18免费网站| 欧美激情xxxx性bbbb| 亚洲图片在区色| 日韩va亚洲va欧洲va国产| 国产精品久久久久影院日本| 91精品久久久久久久久| 欧美一级视频一区二区| 亚洲久久久久久久久久久| 91在线国产电影| 奇米影视亚洲狠狠色| 欧美孕妇孕交黑巨大网站| 狠狠久久五月精品中文字幕| 亚洲综合av影视| 在线电影中文日韩| 欧美亚洲国产日韩2020| 亚洲热线99精品视频| 精品一区二区三区电影| 久久久精品国产| 亚洲第一色中文字幕| 91青草视频久久| 国产精品视频不卡| 成人网在线免费看| 久久免费国产精品1| 疯狂做受xxxx高潮欧美日本| 成人激情视频免费在线| 久久久久久亚洲精品中文字幕| 国产视频精品xxxx| 欧美视频裸体精品| 91精品国产免费久久久久久| 精品人伦一区二区三区蜜桃免费| 欧美极品第一页| 久久理论片午夜琪琪电影网| 在线观看国产成人av片| 亚洲精品国精品久久99热| 亚洲国产成人精品一区二区| 成人美女免费网站视频| 亚洲国产精品热久久| 欧美精品制服第一页|