Uni 多进程编程

上传人:m**** 文档编号:179067645 上传时间:2022-12-30 格式:DOCX 页数:22 大小:54.04KB
返回 下载 相关 举报
Uni 多进程编程_第1页
第1页 / 共22页
Uni 多进程编程_第2页
第2页 / 共22页
Uni 多进程编程_第3页
第3页 / 共22页
点击查看更多>>
资源描述
一.多进程程序的特点由于UNIX系统是分时多用户系统,CPU按时间片分配给各个用户使用,而在实质上应该说CPU按时间片 分配给各个进程使用, 每个进程都有自己的运行环境以使得在 CPU 做进程切换时不会忘记该进程已计算 了一半的半成品.以DOS的概念来说,进程的切换都是一次DOS中断处理过程,包括三个层次:用户数据的保存:包括正文段(TEXT),数据段(DATA,BSS),栈段(STACK),共享内存段(SHARED MEMORY )的保存.(2)寄存器数据的保存:包括PC(program counter,指向下一条要执行的指令的地址),PSW(processor status word,处理机状态字),SP(stackpointer,栈指针),PCBP(pointer of process control block,进程控制块指针), FP(frame pointer,指向栈中一个函数的local变量的首地址),AP(augument pointer,指向栈中函数调用的实参位 置),ISP(interrupt stack pointer,中断栈指针),以及其他的通用寄存器等.(3)系统层次的保存:包括proc,u,虚拟存储空间管理表格,中断处理栈.以便于该进程再一次得到CPU时间片时能正常运行下去. 既然系统已经处理好所有这些中断处理的过程, 我们做程序还有什么要担 心的呢? 我们尽可以使用系统提供的多进程的特点, 让几个程序精诚合作, 简 单而又高效地把结果给它搞出来.另外,UNIX系统本身也是用C语言写的多进程程序,多进程编程是UNIX的特 点,当我们熟悉了多进程编程后,将会对UNIX系统机制有一个较深的认识. 首先我介绍一下多进程程序的一些突出的特点:1. 并行化 一件复杂的事件是可以分解成若干个简单事件来解决的, 这在程序员 的大脑中早就形成了这种概念, 首先将问题分解成一个个小问题, 将小问 题再细分, 最后在一个合适的规模上做成一个函数. 在软件工程中也是这 么说的. 如果我们以图的方式来思考, 一些小问题的计算是可以互不干扰 的, 可以同时处理, 而在关键点则需要统一在一个地方来处理, 这样程序 的运行就是并行的, 至少从人的时间观念上来说是这样的. 而每个小问题 的计算又是较简单的.2. 简单有序 这样的程序对程序员来说不亚于管理一班人, 程序员为每个进程设计 好相应的功能, 并通过一定的通讯机制将它们有机地结合在一起, 对每个 进程的设计是简单的, 只在总控部分小心应付(其实也是蛮简单的), 就可 完成整个程序的施工.3. 互不干扰这个特点是操作系统的特点, 各个进程是独立的, 不会串位.4. 事务化比如在一个数据电话查询系统中, 将程序设计成一个进程只处理一次 查询即可, 即完成一个事务. 当电话查询开始时, 产生这样一个进程对付 这次查询; 另一个电话进来时, 主控程序又产生一个这样的进程对付, 每 个进程完成查询任务后消失. 这样的编程多简单, 只要做一次查询的程序 就可以了.二.常用的多进程编程的系统调用1. fork() 功能:创建一个新的进程.语法:#include#include pid_t fork();说明:本系统调用产生一个新的进程, 叫子进程, 是调用进程的一个复 制品. 调用进程叫父进程, 子进程继承了父进程的几乎所有的属 性:. 实际 UID,GID 和有效 UID,GID. 环境变量 . 附加 GID.调用exec()时的关闭标志. UID 设置模式比特位 . GID 设置模式比特位 . 进程组号 . 会话 ID. 控制终端 . 当前工作目录 . 根目录 . 文件创建掩码 UMASK. 文件长度限制 ULIMIT. 预定值, 如优先级和任何其他的进程预定参数, 根据种类不同 决定是否可以继承. 还有一些其它属性 . 但子进程也有与父进程不同的属性:. 进程号, 子进程号不同与任何一个活动的进程组号 . 父进程号 . 子进程继承父进程的文件描述符或流时 , 具有自己的一个拷贝 并且与父进程和其它子进程共享该资源. 子进程的用户时间和系统时间被初始化为 0. 子进程的超时时钟设置为 0. 子进程的信号处理函数指针组置为空 . 子进程不继承父进程的记录锁 .返回值: 调用成功则对子进程返回 0, 对父进程返回子进程号, 这也是 最方便的区分父子进程的方法. 若调用失败则返回-1 给父进程, 子进程不生成.例子:pid_t pid;if (pid=fork()0) /*父进程处理过程*/else if (pid=0) /*子进程处理过程*/exit(0); /*注意子进程必须用exit()退出运行*/else printf(fork errorn);exit(0);2. system() 功能:产生一个新的进程, 子进程执行指定的命令.语法:#include#includeint system(string)char *string;说明:本调用将参数string传递给一个命令解释器(一般为sh)执行,即 string被解释为一条命令,由sh执行该命令.若参数string为一 个空指针则为检查命令解释器是否存在.该命令可以同命令行命令相同形式, 但由于命令做为一个参数放 在系统调用中, 应注意编译时对特殊意义字符的处理. 命令的查 找是按PATH环境变量的定义的.命令所生成的后果一般不会对父 进程造成影响.返回值:当参数为空指针时, 只有当命令解释器有效时返回值为非零.若参数不为空指针, 返回值为该命令的返回状态(同 waitpid() 的返回值. 命令无效或语法错误则返回非零值,所执行的命令被 终止. 其他情况则返回-1.例子:char command81;int i;for (i=1;i0) wait(int *)0);/*父进程等待子进程的返回*/ else /*子进程处理过程*/ exit(0);7.waitpid() 功能:等待指定进程号的子进程的返回并修改状态 语法:#include#includepid_t waitpid(pid,stat_loc,options)pid_t pid;int *stat_loc,options;说明:当pid等于-l,option等于0时,该系统调用等同于wait (否则该 系统调用的行为由参数pid和options决定.pid指定了一组父进程要求知道其状态的子进程: -1要求知道任何一个子进程的返回状态.0:要求知道进程号为pid值的子进程的状态.0) waitpid(pid,&stat_loc,0);/*父进程等待进程号为pid的子进程的返回*/else /*子进程的处理过程*/exit(1);/*父进程*/printf(stat_loc is %dn,stat_loc);/*字符串stat_loc is 1将被打印出来*/8. setpgrp() 功能:设置进程组号和会话号.语法:#includepid_t setpgrp()说明:若调用进程不是会话首进程.将进程组号和会话号都设置为与它 的进程号相等.并释放调用进程的控制终端.返回值:调用成功后,返回新的进程组号. 例子:/*父进程处理*/if (fork()0) /*父进程处理*/else setpgrp();/*子进程的进程组号已修改成与它的进程号相同*/exit(0);9. exit() 功能:终止进程.语法:#includevoid exit(status)int status;说明:调用进程被该系统调用终止.引起附加的处理在进程被终止前全 部结束.返回值:无10. signal()功能:信号管理功能语法:#includevoid (*signal(sig,disp)(int)int sig;void (*disp)(int);void (*sigset(sig,disp)(int)int sig;void (*disp)(int);int sighold(sig) int sig;int sigrelse(sig) int sig;int sigignore(sig)int sig;int sigpause(sig)int sig; 说明:这些系统调用提供了应用程序对指定信号的简单的信号处理.signal。和sigset()用于修改信号定位.参数sig指定信号(除了 SIGKILL和SIGSTOP,这两种信号由系统处理,用户程序不能捕捉到). disp 指定新的信号定位,即新的信号处理函数指针.可以为 SIG_IGN,SIG_DFL 或信号句柄地址.若使用signal(),disp是信号句柄地址,sig不能为SIGILL,SIGTRAP 或SIGPWR,收到该信号时,系统首先将重置sig的信号句柄为SIG_DFL, 然后执行信号句柄.若使用 sigset(),disp 是信号句柄地址,该信号时,系统首先将该 信号加入调用进程的信号掩码中,然后执行信号句柄.当信号句柄 运行结束后,系统将恢复调用进程的信号掩码为信号收到前的状态.另外, 使用sigset()时,disp为SIG_HOLD,则该信号将会加入调用进程的 信号掩码中而信号的定位不变.sighold(將信号加入调用进程的信号掩码中.sigrelse()将信号从调用进程的信号掩码中删除.sigignore()将信号的定位设置为SIG_IGN.sigpause()将信号从调用进程的信号掩码中删除,同时挂起调用 进程直到收到信号.若信号SIGCHLD的信号定位为SIG_IGN,则调用进程的子进程在终 止时不会变成僵死进程.调用进程也不用等待子进程返回并做相 应处理.返回值:调用成功则signal。返回最近调用signal()设置的disp的值. 否则返回 SIG_ERR.例子一:设置用户自己的信号中断处理函数,以SIGINT信号为例:int flag=0;void myself()flag=1;printf(get signal SIGINTn);/*若要重新设置 SIGINT 信号中断处理函数为本函数则执行以*下步骤*/void (*a)();a=myself;signal(SIGINT,a);flag=2;main()while (1) sleep(2000); /*等待中断信号*/if (flag=1) printf(skip system call sleepn);exit(0);if (flag=2) printf(skip system call sleepn); printf(waiting for next signaln); 11. kill() 功能:向一个或一组进程发送一个信号.语法:#include#includeint kill(pid,sig);pid_t pid;int sig; 说明:本系统调用向一个或一组进程发送一个信号,该信号由参数 sig 指 定,为系统给出的信号表中的一个.若为 0(空信号)则检查错误但 实际上并没有发送信号,用于检查pid的有效性.pid指定将要被发送信号的进程或进程组.pid若大于0,则信号将 被发送到进程号等于pid的进程;若pid等于0则信号将被发送到所 有的与发送信号进程同在一个进程组的进程(系统的特殊进程除 外);若pid小于-1,则信号将被发送到所有进程组号与pid绝对值 相同的进程;若pid等于-1,则信号将被发送到所有的进程(特殊系 统进程除外). 信号要发送到指定的进程,首先调用进程必须有对该进程发送信 号的权限.若调用进程有合适的优先级则具备有权限.若调用进程 的实际或有效的UID等于接收信号的进程的实际UID或用setuid() 系统调用设置的UID,或sig等于SIGCONT同时收发双方进程的会话 号相同,则调用进程也有发送信号的权限.若进程有发送信号到pid指定的任何一个进程的权限则调用成功, 否则调用失败,没有信号发出.返回值:调用成功则返回 0,否则返回-1.例子:假设前一个例子进程号为324,现向它发一个SIGINT信号,让它做 信号处理:kill(pid_t)324,SIGINT);12. alarm() 功能:设置一个进程的超时时钟.语法:#includeunsigned int alarm(sec)unsigned int sec;说明:指示调用进程的超时时钟在指定的时间后向调用进程发送一个SIGALRM 信号.设置超时时钟时时间值不会被放入堆栈中,后一次 设置会把前一次(还未到超时时间)冲掉.若 sec 为 0,则取消任何以前设置的超时时钟.fork()会将新进程的超时时钟初始化为0.而当一个进程用exec()族系统调用新的执行文件时,调用前设置的超时时钟在调用后仍 有效.返回值:返回上次设置超时时钟后到调用时还剩余的时间秒数.例子:int flag=0;void myself()flag=1;printf(get signal SIGALRMn);/*若要重新设置SIGALRM信号中断处理函数为本函数则执行*以下步骤*/void (*a)();a=myself;signal(SIGALRM,a);flag=2;main()alarm(100); /*100秒后发超时中断信号*/while (1) sleep(2000); /*等待中断信号*/if (flag=1) printf(skip system call sleepn);exit(0);if (flag=2) printf(skip system call sleepn);printf(waiting for next signaln);13. msgsnd()功能:发送消息到指定的消息队列中.语法:#include#include#includeint msgsnd(msqid,msgp,msgsz,msgflg)int msqid;void *msgp;size_t msgsz;int msgflg;说明:发送一个消息到由msqid指定消息队列标识号的消息队列.参数 msgp 指向一个用户定义的缓冲区,并且缓冲区的第一个域应 为长整型,指定消息类型,其他数据放在缓冲区的消息中其他正文 区内.下面是消息元素定义:long mtype;char mtext;mtype 是一个整数,用于接收进程选择消息类型.mtext是一个长度为msgsz字节的任何正文,参数msgsz可从0到系 统允许的最大值间变化.msgflg 指定操作行为 :.若(msgflg&IPC_NOWAIT)是真的,消息并不是被立即发送而调用 进程会立即返回.若(msgflg&IPC_NOWAIT)不是真的,则调用进程会被挂起直到下 面情况之一发生:* 消息被发送出去.* 消息队列标志被系统删除.系统调用返回-1.* 调用进程接收到一个未被忽略的中断信号,调用进程继续 执行或被终止.调用成功后,对应指定的消息队列的相关结构做如下动作:.消息数(msg_qnum)加 1.消息队列最近发送进程号(msgspid)改为调用进程号.消息队列发送时间(msg_stime)改为当前系统时间.以上信息可用命令ipcs -a看到.返回值:成功则返回 0,否则返回-1.14. msgrcv()功能:从消息队列中取得指定类型的消息.语法:#include#include#includeint msgrcv(msqid,msgp,msgsz,msgtyp,msgflg)int msqid;void *msgp;int msgsz;long msgtyp;int msgflg;说明:本系统调用从由msqid指定的消息队列中读取一个由msgtyp指定 类型的消息到由msgp指向的缓冲区中,同样的,该缓冲区的结构如 前所述,包括消息类型和消息正文.msgsz为可接收的消息正文的 字节数.若接收到的消息正文的长度大于msgsz,则会被截短到 msgsz字节为止(当消息标志msgflg&MSG_NOERROR为真时),截掉的 部份将被丢失,而且不通知消息发送进程.msgtyp 指定消息类型:. 为 0 则接收消息队列中第一个消息 . 大于 0 则接收消息队列中第一个类型为 msgtyp 的消息 . 小于 0 则接收消息队列中第一个类型值不小于 msgtyp 绝对值且 类型值又最小的消息.msgflg 指定操作行为:.若(msgflg&IPC_NOWAIT)是真的,调用进程会立即返回,若没有接收到消息则返回值为-l,errno设置为ENOMSG.若(msgflg&IPC_NOWAIT)不是真的,则调用进程会被挂起直到下 面情况之一发生:* 队列中的消息的类型是有效的.*消息队列标志被系统删除系统调用返回-1.* 调用进程接收到一个未被忽略的中断信号,调用进程继续执行或被终止. 调用成功后,对应指定的消息队列的相关结构做如下动作:.消息数(msg_qnum)减 1.消息队列最近接收进程号(msg_lrpid)改为调用进程号.消息队列接收时间(msg_rtime)改为当前系统时间.以上信息可用命令ipcs -a看到. 返回值:调用成功则返回值等于接收到实际消息正文的字节数. 不成功则返回-1.15. msgctl()功能:消息控制操作语法:#include#include#includeint msgctl(msqid,cmd,buf)int msqid,cmd;struct msqid_ds *buf;说明:本系统调用提供一系列消息控制操作,操作动作由cmd定义,以下 cmd定义值表明了各操作动作的定义.IPC_STAT将msqid相关的数据结构中各个元素的当前值放入由buf 指向的结构中.IPC_SET将msqid相关的数据结构中的下列元素设置为由buf指 向的结构中的对应值.msg_perm.uidmsg_perm.gidmsg_perm.modemsg_qbytes本命令只能由有效UID等于msg_perm.cuid或msg_perm.uid的进程或有效 UID 有合适权限的进程操作.只有具有合适权限的 用户才能增加 msg_qbytes 的值.IPC_RMID:删除由msqid指示的消息队列将它从系统中删除并破坏相关的数据结构.本命令只能由有效 UID 等于 msg_perm.cuid 或 msg_perm.uid 的 进程或有效UID有合适权限的进程操作.返回值:调用成功则返回值为 0,否则为-1.16. msgget() 功能:取得一个消息队列.语法:#include#include#includeint msgget(key,msgflg)key_t key;int msgflg;说明:本系统调用返回与参数key相关的消息队列的标识符. 若以下事实成立,则与消息队列相关的标识符和数据结构将被创 建出来:. 若参数 key 等于 IPC_PRIVATE. 若参数 key 没有一个已存在的消息队列标识符与之相关,同时值 (msgflg&IPC_CREAT)为真.创建消息队列时,与新的消息队列标识符相关的数据结构将被初 始化为如下:.msg_perm.cuid和msg_perm.uid设置为调用进程的有效UID.msg_perm.cgid和msg_perm.gid设置为调用进程的有效GID.msg_perm.mode访问权限比特位设置为msgflg访问权限比特位. msg_qnum,msg_lspid,msg_lrpid,msg_stime,msg_rtime 设置为 0. msg_ctime 设置为当前系统时间. msg_qbytes 设置为系统允许的最大值.返回值:调用成功则返回一非 0 值,称为消息队列标识符;否则返回值为-1. 例子:本例将包括上述所有消息队列操作的系统调用:#define RKEY 0x9001L /*读消息队列的 KEY 值*/#define WKEY 0x9002L /*写消息队列的 KEY 值*/#define MSGFLG 0666 /*消息队列访问权限*/#define IPC_WAIT 0 /*等待方式在include文件中未定义*/int rmsqid; /*读消息队列标识符*/int wmsqid; /*写消息队列标识符*/struct msgbuf long mtype;char mtext200; buf; /*若读消息队列已存在就取得标识符,否则则创建并取得标识符*/if (rmsqid=msgget(RKEY,MSGFLG|IPC_CREAT)0) printf(get read message queue failedn);exit(1); /*若写消息队列已存在则失败,若不存在则创建并取得标识符*/ if (wmsqid=msgget(WKEY, MSGFLG|IPC_CREAT|IPC_TRUNC)0)printf(get %ld type message from queue:%sn, buf.mtype,buf.mtext);else printf(get message failedn); exit(3); buf.mtype=3Lif (msgsnd(wmsqid,&buf,sizeof(struct msgbuf)-sizeof(long), IPC_NOWAIT)0)printf(send message OKn);else printf(send message failedn); exit(4); msgctl(wmsqid,IPC_RMID,(struct msqid *)NULL);17. shmat() 功能:联接共享内存的操作.语法:#include#include#includevoid *shmat(shmid,shmaddr,shmflg) int shmid;void *shmaddr;int shmid;说明:将由shmid指示的共享内存联接到调用进程的数据段中被联接的 段放在地址,该地址由以下准则指定:.若shmaddr等于(void *)0,则段联接到由系统选择的第一个可 用的地址上.若 shmaddr 不等于(void *)0 同时(shmflg&SHM_RND)值为真,则 段联接到由(shmaddr-(shmaddr%SHMLBA)给出的地址上.若 shmaddr 不等于(void *)0 同时(shmflg&SHM_RND)值为假,则 段联接到由shmaddr指定的地址上.若(shmflg&sSHM_RDONLY)为真并且调用进程有读允许,则被联接 的段为只读;否则,若值不为真且调用进程有读写权限,则被联接 的段为可读写的.返回值:若调用成功则返回被联接的共享内存段在数据段上的启始地址.否则返回值为-1.18. shmdt() 功能:断开共享内存联接的操作.语法:#include#include#includevoid *shmdt(shmaddr)void *shmaddr;说明:本系统调用将由shmaddr指定的共享内存段从调用进程的数据段 脱离出去.返回值:若调用成功则返回值为 0,否则返回值为-1.19. shmget()功能:取得共享内存段语法:#include#include#includeint shmget(key,size,shmflg)key_t key;int size,shmflg;说明:本系统调用返回key相关的共享内存标识符.共享内存标识符和相关数据结构及至少size字节的共享内存段能正常创建,要求以下事实成立:. 参数 key 等于 IPC_PRIVATE.参数key没有相关的共享内存标识符,同时(shmflg&IPC_CREAT) 值为真.共享内存创建时,新生成的共享内存标识相关的数据结构被初始 化如下:.shm_perm.cuid和shm_perm.uid设置为调用进程的有效UID.shm_perm.cgid和shm_perm.gid设置为调用进程的有效GID. shm_perm.mode 访问权限比特位设置为 shmflg 访问权限比特位. shm_lpid,shm_nattch,shm_atime,shm_dtime 设置为 0. shm_ctime 设置为当前系统时间. shm_segsz 设置为 0.返回值:若调用成功则返回一个非0值,称为共享内存标识符,否则返回 值为-1.20. shmctl() 功能:共享内存控制操作.语法:#include#include#includeint shmctl(shmid,cmd,buf)int shmid,cmd;struct shmid_ds *buf;说明:本系统调用提供一系列共享内存控制操作操作行为由cmd指定.以下为cmd的有效值:.IPC_STAT将shmid相关的数据结构中各个元素的当前值放入由 buf 指向的结构中.IPC_SET将shmid相关的数据结构中的下列元素设置为由buf指 向的结构中的对应值.shm_perm.uidshm_perm.gidshm_perm.mode本命令只能由有效UID等于shm_perm.cuid或shm_perm.uid的 进程或有效 UID 有合适权限的进程操作.IPC_RMID:删除由shmid指示的共享内存将它从系统中删除并 破坏相关的数据结构.本命令只能由有效UID等于shm_perm.cuid或shm_perm.uid的 进程或有效 UID 有合适权限的进程操作.返回值:若调用成功则返回 0,否则返回-1.例子:本例包括上述所有共享内存操作系统调用:#include#include#include#define SHMKEY 74#define K 1024int shmid;cleanup()shmctl(shmid,IPC_RMID,0);exit(0);main()int *pint;char *addr1,*addr2;extern char *shmat();extern cleanup();for (i=0;i20;i+)signal(i,cleanup);shmid=shmget(SHMKEY,128*K,0777|IPC_CREAT);addr1=shmat(shmid,0,0);addr2=shmat(shmid,0,0);printf(addr1 0x%x addr2 0x%xn,addr1,addr2); pint=(int*)addr1;for (i=0;i256;i+)*pint+=i;pint=(int*)addr1;*pint=256;pint=(int*)addr2;for (i=0;i256;i+)printf(index %dtvalue%dn,i,*pint+);shmdt(addr1);shmdt(addr2);pause();21. semctl()功能:信号量控制操作.语法:#include#include#includeint semctl(semid,memnum,cmd,arg)int semid,semnum,cmd;union semun int val;struct semid_ds *buf;ushort *array;arg;说明:本系统调用提供了一个信号量控制操作,操作行为由cmd定义,这些命令是对由 semid 和 semnum 指定的信号量做操作的.每个命令都 要求有相应的权限级别:.GETVAL:返回semval的值,要求有读权限.SETVAL:设置semval的值到arg.val上.此命令成功执行后,semadj的值对应的所有进程的信号量全部被清除,要求有修 改权限.GETPID:返回sempid的值,要求有读权限.GETNCNT:返回semncnt的值,要求有读权限.GETZCNT:返回semzcnt的值,要求有读权限.以下命令在一组信号量中的各个semval上操作:.GETALL返回每个semval的值,同时将各个值放入由arg.array 指向的数组中当此命令成功执行后,semadj的值对应的所有 进程的信号量全部被清除,要求有修改权限.SETALL根据由arg.array指向的数组设置各个semval值.当此命令成功执行后,semadj的值对应的所有进程的信号量全部 被清除,要求有修改权限.以下命令在任何情况下都是有效的:.IPC_STAT将与semid相关的数据结构的各个成员的值放入由 arg.buf指向的结构中.要求有读权限.IPC_SET:设置semid相关数据结构的如下成员,设置数据从arg.buf指向的结构中读取:sem_perm.uidsem_perm.gidsem_perm.mode本命令只能由有效 UID 等于 sem_perm.cuid 或 sem_perm.uid 的 进程或有效UID有合适权限的进程操作.IPC_RMID:删除由semid指定的信号量标识符和相关的一组信号 量及数据结构.本命令只能由有效UID等于sem_perm.cuid或 sem_perm.uid 的进程或有效 UID 有合适权限的进程操作.返回值:若调用成功,则根据cmd返回以下值: GETVAL:semval 的值.GETPID:sempid 的值.GETNCNT:semncnt 的值.GETZCNT:semzcnt 的值.其他:0.若调用失败则返回-1.22. semget() 功能:取得一组信号量.语法:#include#include#includeint semget(key,nsems,semflg)key_t key;int nsems,semflg;说明:返回和key相关的信号量标识符.若以下事实成立,则与信号量标识符,与之相关的semid_ds数据结 构及一组 nsems 信号量将被创建 :. key 等于 IPC_PRIVATE.系统内还没有与key相关的信号量,同时(semflg&IPC_CREAT) 为真.创建时新的信号量相关的semid_ds数据结构被初始化如下:.在操作权限结构,sem_perm.cuid和sem_perm.uid设置等于调用 进程的有效 UID.在操作权限结构,sem_perm.cgid和sem_perm.gid设置等于调用 进程的有效 GID.访问权限比特位sem_perm.mode设置等于semflg的访问权限比 特位. sem_otime 设置等于 0,sem_ctime 设置等于当前系统时间. 返回值:若调用成功,则返回一非 0值,称为信号量标识符;否则返回-1.23. semop()功能:信号量操作.语法:#include#include#includeint semop(semid,sops,nsops)int semid;struct sembuf *sops;unsigned nsops;说明:本系统调用用于执行用户定义的在一组信号量上操作的行为集合. 该组信号量与 semid 相关.参数sops为一个用户定义的信号量操作结构数组指针. 参数 nsops 为该数组的元素个数.数组的每个元素结构包括如下成员: sem_num; /* 信号量数 */ sem_op; /* 信号量操作 */ sem_flg; /* 操作标志 */由本系统调用定义的每个信号量操作是针对由semid和sem_num指 定的信号量的变量sem_op指定三种信号量操作的一种:. 若 sem_op 为一负数并且调用进程具有修改权限,则下列情况之 一将会发生:*若semval不小于sem_op的绝对值,则sem_op的绝对值被减去 semval的值.若(semflg&SEM_UNDO)为真则sem_op的绝对值加 上调用进程指定的信号量的semadj值.* 若 semval 小于 sem_op 的绝对值同时(semflg&IPC_NOWAIT)为 真,则本调用立即返回.* 若 semval 小于 sem_op 的绝对值同时(semflg&IPC_NOWAIT)为 假,则本系统调用将增加指定信号量相关的 semncnt 值(加一), 将调用进程挂起直到下列条件之一被满足:(1) .semval值变成不小于sem_op的绝对值.当这种情况发 生时,指定的信号量相关的 semncnt 减一,若(semflg&SEM_UNDO)为真则sem_op的绝对值加上调用 进程指定信号量的 semadj 值.(2) .调用进程等待的semid已被系统删除.(3) .调用进程捕俘到信号,此时,指定信号量的 semncnt 值 减一,调用进程执行中断服务程序.若sem_op为一正值,同时调用进程具有修改权限,sem_op的值加 上semval的值,若(semflg&SEM_UNDO)为真,贝U sem_op减去调用 进程指定信号量的 semadj 值. 若 sem_op 为 0,同时调用进程具有读权限,下列情况之一将会发生:* 若 semval 为 0,本系统调用立即返回.*若semval不等于0且(semflg&IPC_NOWAIT)为真,本系统调用 立即返回.*若semval不等于0且(semflg&IPC_NOWAIT)为假,本系统调用 将把指定信号量的semzcnt 值加一,将调用进程挂起直到下列情况之一发生:(1) .semval值变为0时,指定信号量的semzcnt值减一.(2) .调用进程等待的semid已被系统删除.(3) .调用进程捕俘到信号,此时,指定信号量的semncnt值 减一,调用进程执行中断服务程序.返回值:调用成功则返回 0,否则返回-1. 例子:本例将包括上述信号量操作的所有系统调用:#include #include#include#define SEMKEY 75 int semid;unsigned int count;/*在文件 sys/sem.h 中定义的 sembuf 结构* struct sembuf * unsigned short sem_num;* short sem_op;* short sem_flg;* */struct sembuf psembuf,vsembuf; /*P 和 V 操作*/ cleanup()semctl(semid,2,IPC_RMID,0);exit(0);main(argc,argv)int argc;char *argv;int i,first,second;short initarray2,outarray2;extern cleanup();if (argc=1) for (i=0;i20;i+)signal(i,clearup); semid=semget(SEMKEY,2,0777|IPC_CREAT); initarray0=initarray1=1;semctl(semid,2,SETALL,initarray); semctl(semid,2,GETALL,outarray); printf(sem init vals %d%d n, outarray0,outarray1);pause(); /*睡眠到被一软件中断信号唤醒*/ else if (argv10=a) first=0;second=1; else first=1;second=0; semid=semget(SEMKEY,2,0777);psembuf.sem_op=-1;psembuf.sem_flg=SEM_UNDO;vsembuf.sem_op=1; vsembuf.sem_flg=SEM_UNDO; for (count=0;xcount+) psembuf.sem_num=first;semop(semid,&psembuf,1);psembuf.sem_num=second;semop(semid,&psembuf,1);printf(proc %d count %dn,getpid(),count);vsembuf.sem_num=second;semop(semid,&vsembuf,1);vsembuf.sem_num=first;semop(semid,&vsembuf,1);24. sdenter() 功能:共享数据段同步访问,加锁.语法:#includeint sdenter(addr,flags)char *addr;int flags;说明:用于指示调用进程即将可以访问共享数据段中的内容.参数addr为将一个sdget()调用的有效返回码.所执行的动作取决于flags的值:.SD_NOWAIT:若另一个进程已对指定的段调用本系统调用且还没 有调用sdleave(),并且该段并非用SD_UNLOCK标志创建,则调 用进程不是等待该段空闲而是立即返回错误码.SD_WRITE:指示调用进程希望向共享数据段写数据此时,另一 个进程用SD_RDONLY标志联接该共享数据段则不被允许. 返回值:调用成功则返回 0,否则返回-1.25. sdleave() 功能:共享数据段同步访问,解锁.语法:#includeint sdleave(addr,flags)char *addr;说明:用于指示调用进程已完成修改共享数据段中的内容. 返回值:调用成功则返回 0,否则返回-1.26. sdget()功能:联接共享数据段到调用进程的数据空间中.语法:#includechar *sdget(path,flags,size.mode)char *path;int flags; long size; int mode; 说明:本系统调用将共享数据段联接到调用进程的数据段中,具体动
展开阅读全文
相关资源
正为您匹配相似的精品文档
相关搜索

最新文档


当前位置:首页 > 建筑环境 > 建筑资料


copyright@ 2023-2025  zhuangpeitu.com 装配图网版权所有   联系电话:18123376007

备案号:ICP2024067431-1 川公网安备51140202000466号


本站为文档C2C交易模式,即用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知装配图网,我们立即给予删除!