我们知道系统调用semget用来把Unix信号量集合的键值译成代表信号量集合的标识符,该集合中有count个元素,其存取权限定义与文件相同,由flags定义。若flags的IPC_CREAT位被置位,则当该集合不存在时系统就创建之。
因此各进程可都用置IPC_CREAT位的flags参数来获取信号量集合的标识符,不需要由某一进程事先创建。若flags为IPC_PRIDVATE则不管同键值的Unix信号量集合是否存在系统都建立之,并返回下一个可用的标识符。
系统调用semctl在一组信号量上做各种控制操作,诸如Unix信号量集合的初始化、删除和状态查询等。常用的操作及相关的命令格式如下:
①取消Unix信号量集合
- int semctl(sid,count,IPC_RMID,0)
- int sid; /*信号量集合标识符*/
- int count; /*信号量集合中元素个数*/
②设置信号量集合的初值(初始化)
Unix信号量集合刚建立时,各信号量的初值不确定,需要设定初值。初值的设定可用SETALL或SETVAL命令。若用SETALL命令,其格式为:
- int semctl(sid,count,SETALL,arg)
- int sid; /*信号量集合标识符*/
- int count; /*信号量集合中元素个数*/
- ushort *arg; /*命令参数*/
该命令把数组arg中的前count个值依次赋给集合中各信号量,一次可设定多个信号量的初值。
若用SETVAL命令,其格式为:
- int semctl(sid,semnum,SETVAL,arg)
- int sid; /*信号量集合标识符*/
- int semnum; /*信号量元素编号*/
- int arg; /*命令参数*/
该命令将arg的值赋给集合中第semnum个信号量,一次仅能设定一个信号量的初值。
③查询Unix信号量集合的当前值
查询信号量集合的当前值可用GETALL或GETVAL命令。若用GETALL命令,其格式为:
- int semctl(sid,count,GETALL,arg)
- int sid; /*信号量集合标识符*/
- int count; /*信号量集合中元素个数*/
- ushort *arg; /*命令参数*/
该命令把信号量集合中各信号量的当前值返回到数组arg中。
若用GETVAL命令,其格式为:
- int semctl(sid,semnum,GETVAL,0)
- int sid; /*信号量集合标识符*/
- int semnum; /*信号量元素编号*/
该命令把集合中第semnum个信号量的当前值作为调用的返回值。
④查询某个Unix信号量的等待进程数
当一个进程要执行信号量操作时若条件不具备则被阻塞,有关信号量的等待进程数也相应变化。
通过GETNCNT命令可查询等待信号量增值的进程数,其格式如下:
- int semctl(sid,semnum,GETNCNT,0)
- int sid; /*信号量集合标识符*/
- int semnum; /*信号量元素编号*/
该命令把等待第semnum个信号量增值的进程数作为调用的返回值。
通过GETZCNT命令可查询等待信号量值为0的进程数,其格式如下:
- int semctl(sid,semnum,GETZCNT,0)
- int sid; /*信号量集合标识符*/
- int semnum; /*信号量元素编号*/
该命令把等待第semnum个信号量值为0的进程数作为调用的返回值。
关于Unix信号量的介绍,我们就到这里了,至于其它的控制命令,因不常用而不再累述。
【编辑推荐】