同共享内存一样,系统中同样需要为信号量集定制一系列专有的操作函数(semget,semctl等)。系统命令ipcs可查看当前的系统IPC的状态,在命令后使用-s参数。使用函数semget可以创建或者获得一个信号量集ID,函数原型如下:
#include <sys/shm.h>
int semget( key_t key, int nsems, int flag)
函数中参数key用来变换成一个标识符,每一个IPC对象与一个key相对应。当新建一个共享内存段时,使用参数flag的相应权限位对ipc_perm结构中的mode域赋值,对相应信号量集的shmid_ds初始化的值如表1所示。
shmid_ds结构初始化值表 ipc_perm结构数据 初 值 ipc_perm结构数据 初 值 Sem_otime 0 Sem_nsems Nsems Sem_ctime 系统当前值 参数nsems是一个大于等于0的值,用于指明该信号量集中可用资源数(在创建一个信号量时)。当打开一个已存在的信号量集时该参数值为0。函数执行成功,则返回信号量集的标识符(一个大于等于0的整数),失败,则返回–1。函数semop用以操作一个信号量集,函数原型如下:
#include <sys/sem.h>
int semop( int semid, struct sembuf semoparray[], size_t nops )
函数中参数semid是一个通过semget函数返回的一个信号量标识符,参数nops标明了参数semoparray所指向数组中的元素个数。参数semoparray是一个struct sembuf结构类型的数组指针,结构sembuf来说明所要执行的操作,其定义如下:
struct sembuf{
unsigned short sem_num
short sem_op
short sem_flg
}
在sembuf结构中,sem_num是相对应的信号量集中的某一个资源,所以其值是一个从0到相应的信号量集的资源总数(ipc_perm.sem_nsems)之间的整数。sem_op指明所要执行的操作,sem_flg说明函数semop的行为。sem_op的值是一个整数,如表2所示,列出了详细sem_op的值及所对应的操作。
sem_op值详解 Sem_op 操 作 正数 释放相应的资源数,将sem_op的值加到信号量的值上 0 进程阻塞直到信号量的相应值为0,当信号量已经为0,函数立即返回。如果信号量的值不为0,则依据sem_flg的IPC_NOWAIT位决定函数动作。sem_flg指定IPC_NOWAIT,则semop函数出错返回EAGAIN。sem_flg没有指定IPC_NOWAIT,则将该信号量的semncnt值加1,然后进程挂起直到下述情况发生。信号量值为0,将信号量的semzcnt的值减1,函数semop成功返回;此信号量被删除(只有超级用户或创建用户进程拥有此权限),函数smeop出错返回EIDRM;进程捕捉到信号,并从信号处理函数返回,在此情况将此信号量的semncnt值减1,函数semop出错返回EINTR 负数 请求sem_op的绝对值的资源。如果相应的资源数可以满足请求,则将该信号量的值减去sem_op的绝对值,函数成功返回。当相应的资源数不能满足请求时,这个操作与sem_flg有关。sem_flg指定IPC_NOWAIT,则semop函数出错返回EAGAIN。sem_flg没有指定IPC_NOWAIT,则将该信号量的semncnt值加1,然后进程挂起直到下述情况发生:当相应的资源数可以满足请求,该信号的值减去sem_op的绝对值。成功返回;此信号量被删除(只有超级用户或创建用户进程拥有此权限),函数smeop出错返回EIDRM:进程捕捉到信号,并从信号处理函数返回,在此情况将此信号量的semncnt值减1,函数semop出错返回EINTR
表示有4个等待进程。
信号量的当前值如果是正值N,该值表示有N个可用资源。
如果为0,则表示所有资源全部被分配,同时没有进程处于等待状态
如果为负数N,则表示全部资源分配完毕,且还有N个进程处于等待该资源的状态。
例如:若信号S的初值为3,当前值为-2,则表示有( 2 )个等待进程,当信号量的值小于0时,其绝对值表示系统中因请求该类资源而被阻塞的进程个数。
扩展资料:
参数nsems是一个大于等于0的值,用于指明该信号量集中可用资源数(在创建一个信号量时)。当打开一个已存在的信号量集时该参数值为0。函数执行成功,则返回信号量集的标识符(一个大于等于0的整数),失败,则返回–1。函数semop用以操作一个信号量集,函数原型如下:
#include <sys/sem.h>
int semop( int semid, struct sembuf semoparray[], size_t nops )
函数中参数semid是一个通过semget函数返回的一个信号量标识符,参数nops标明了参数semoparray所指向数组中的元素个数。
参考资料来源:百度百科-信号量
#include#include
#include
#include types.h>
#include msg.h>
#include
#include ipc.h>
void msg_show_attr(int msg_id, struct msqid_ds msg_info)
{
int ret = -1
sleep(1)
ret = msgctl(msg_id, IPC_STAT, &msg_info)
if( -1 == ret)
{
printf(获消息信息失败\n)
return
}
printf(\n)
printf(现队列字节数:%d\n,msg_info.msg_cbytes)
printf(队列消息数:%d\n,msg_info.msg_qnum)
printf(队列字节数:%d\n,msg_info.msg_qbytes)
printf(发送消息进程pid:%d\n,msg_info.msg_lspid)
printf(接收消息进程pid:%d\n,msg_info.msg_lrpid)
printf(发送消息间:%s,ctime(&(msg_info.msg_stime)))
printf(接收消息间:%s,ctime(&(msg_info.msg_rtime)))
printf(变化间:%s,ctime(&(msg_info.msg_ctime)))
printf(消息UID:%d\n,msg_info.msg_perm.uid)
printf(消息GID:%d\n,msg_info.msg_perm.gid)
}
int main(void)
{
int ret = -1
int msg_flags, msg_id
key_t key
struct msgmbuf{
int mtype
char mtext[10]
}
struct msqid_ds msg_info
struct msgmbuf msg_mbuf
int msg_sflags,msg_rflags
char *msgpath = /ipc/msg/
key = ftok(msgpath,’a')
if(key != -1)
{
printf(功建立KEY\n)
}
else
{
printf(建立KEY失败\n)
}
msg_flags = IPC_CREAT
msg_id = msgget(key, msg_flags|0666)
if( -1 == msg_id)
{
printf(消息建立失败\n)
return 0
}
msg_show_attr(msg_id, msg_info)
msg_sflags = IPC_NOWAIT
msg_mbuf.mtype = 10
memcpy(msg_mbuf.mtext,测试消息,sizeof(测试消息))
ret = msgsnd(msg_id, &msg_mbuf, sizeof(测试消息), msg_sflags)
if( -1 == ret)
{
printf(发送消息失败\n)
}
msg_show_attr(msg_id, msg_info)
msg_rflags = IPC_NOWAIT|MSG_NOERROR
ret = msgrcv(msg_id, &msg_mbuf, 10,10,msg_rfla
共享内存示例代码:
#include
#include sem.h>
#include ipc.h>
#include
typedef int sem_t
union semun {
int val
struct semid_ds *buf
unsigned short *array
} arg
sem_t CreateSem(key_t key, int value)
{
union semun sem
sem_t semid
sem.val = value
semid = semget(key,value,IPC_CREAT|0666)
if (-1 == semid)
{
printf(create semaphore error\n)
return -1
}
semctl(semid,0,SETVAL,sem)
return semid
}
/*
struct sembuf{
ushort sem_num
short sem_op
short sem_flg
}
*/
void SetvalueSem(sem_t semid, int value)
{
union semun sem
sem.val = value
semctl(semid,0,SETVAL,sem)
return
}
int GetvalueSem(sem_t semid)
{
union semun sem
return semctl(semid,0,GETVAL,sem)
return sem.val
}
void DestroySem(sem_t semid)
{
union semun sem
sem.val = 0
semctl(semid,0,IPC_RMID,sem)
}
int Sem_P(sem_t semid)
{
struct sembuf sops={0,+1,IPC_NOWAIT}
return (semop(semid,&sops,1))
}
int Sem_V(sem_t semid)
{
struct sembuf sops={0,-1,IPC_NOWAIT}
return (semop(semid,&sops,1))
}
static char msg[]=共享内存\n
int main(void)
{
key_t key
int semid,shmid
char i,*shms,*shmc
struct semid_ds buf
int value = 0
char buffer[80]
pid_t p
key = ftok(/ipc/sem/,’a')
shmid = shmget(key,1024,IPC_CREAT|0604)
semid = CreateSem(key,1)
p = fork()
if(p >0)
{
/* 父进程 */
/* 建立共享内存 */
shms = (char *)shmat(shmid,0,0)
memcpy(shms, msg, strlen(msg)+1)
sleep(10)
Sem_P(semid)
shmdt(shms)
DestroySem(semid)
}
else if(p == 0)
{
shmc = (char *)shmat(shmid,0,0)
Sem_V(semid)
printf(共享内存值:%s\n,shmc)
shmdt(sg_
欢迎分享,转载请注明来源:夏雨云
评论列表(0条)