2、System V 信号量

#include <stdio.h>
#include <unistd.h>
#include <limits.h>
#include <errno.h>
#include <string.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <sys/ipc.h>
#include <sys/sem.h>

#define PERMS S_IRUSR|S_IWUSR

void init_semaphore_struct(struct sembuf *sem,int semnum,
int semop,int semflg)
{
/* 初始话信号灯结构 */
/*信号灯的主要用途是保护临界资源(在一个时刻只被一个进程所拥有).*/
sem->sem_num=semnum;
sem->sem_op=semop;
sem->sem_flg=semflg;
}

/*
struct sembuf
{
short sem_num; // 使用那一个信号 
short sem_op; // 进行什么操作 
short sem_flg; // 操作的标志
};
*/

int del_semaphore(int semid)
{
/* 信号灯并不随程序的结束而被删除,如果我们没删除的话(将1 改为0)
可以用ipcs 命令查看到信号灯,用ipcrm 可以删除信号灯的
*/
/*
int semctl(int semid,int semnum,int cmd,union semun arg);
semctl 对信号量进行一系列的控制.
semid 是要操作的信号标志,
semnum 是信号的个数,
cmd 是操作的命令.经常用的两个值是:SETVAL(设置信号量的值)和IPC_RMID(删除信号灯).
arg 是一个给cmd 的参数.
*/
#if 1
return semctl(semid,1,IPC_RMID);//删除一个信号灯
#endif
}

/*
int semop(int semid,struct sembuf *spos,int nspos);
semop 是对信号进行操作的函数.
semid 是信号标志,
spos 是一个操作数组表明要进行什么操作,
nspos 表明数组的个数.
如果sem_op 大于0,那么操作将sem_op 加入到信号量的值中,并唤醒等待信号增加的进程.
如果为0,当信号量的值是0 的时候,函数返回,否则阻塞直到信号量的值为0. 如果小于0,
函数判断信号量的值加上这个负值.如果结果为0 唤醒等待信号量为0 的进程,如果小与0
函数阻塞.如果大于0,那么从信号量里面减去这个值并返回..
*/
int main(int argc,char **argv)
{
char buffer[MAX_CANON],*c;
int i,n;
int semid,semop_ret,status;
pid_t childpid;
struct sembuf semwait,semsignal;
if((argc!=2)||((n=atoi(argv[1]))<1))
{
	fprintf(stderr,"Usage:%s number\n\a",argv[0]);
	exit(1);
}
fprintf(stderr,"pid=%ld\n",getpid());

/* 使用IPC_PRIVATE 表示由系统选择一个关键字来创建 */
/* 创建以后信号灯的初始值为0 */
/*
int semget(key_t key,int nsems,int semflg);
.semget 创建一个信号量.成功时返回信号的ID,
key 是一个关键字,可以是用ftok 创建的也可以是IPC_PRIVATE 表明由系统选用一个关键字.
nsems 表明我们创建的信号个数.
semflg 是创建的权限标志,和我们创建一个文件的标志相同
*/
if((semid=semget(IPC_PRIVATE,1,PERMS))==-1)//创建一个信号
{
	fprintf(stderr,"[%d]:Acess Semaphore Error:%s\n\a",
	getpid(),strerror(errno));
	exit(1);
}
/* semwait 是要求资源的操作(-1) */
init_semaphore_struct(&semwait,0,-1,0);
/* semsignal 是释放资源的操作(+1) */
init_semaphore_struct(&semsignal,0,1,0);
/* 开始的时候有一个系统资源(一个标准错误输出) */
if(semop(semid,&semsignal,1)==-1)
{
	fprintf(stderr,"[%d]:Increment Semaphore Error:%s\n\a",
	getpid(),strerror(errno));

	if(del_semaphore(semid)==-1)
	fprintf(stderr,"[%d]:Destroy Semaphore Error:%s\n\a",
	getpid(),strerror(errno));

	exit(1);
}
/* 创建一个进程链 */

for(i=0;i<n;i++)
{
	if(childpid=fork())
		break;
	else
		printf("i=%d,childpid=%d\n",i,childpid);
}
sprintf(buffer,"[i=%d]-->[Process=%d]-->[Parent=%d]-->[Child=%d]\n",i,getpid(),getppid(),childpid);
c=buffer;
/* 这里要求资源,进入原子操作 */
while(((semop_ret=semop(semid,&semwait,1))==-1)&&(errno==EINTR));
if(semop_ret==-1)
{
	fprintf(stderr,"[%d]:Decrement Semaphore Error:%s\n\a",
	getpid(),strerror(errno));
}
else
{
	while(*c!='\0')fputc(*c++,stderr);
	/* 原子操作完成,赶快释放资源 */
	while(((semop_ret=semop(semid,&semsignal,1))==-1)&&(errno==EINTR));
	if(semop_ret==-1)
	fprintf(stderr,"[%d]:Increment Semaphore Error:%s\n\a",
	getpid(),strerror(errno));
}
/* 不能够在其他进程反问信号灯的时候,我们删除了信号灯 */
while((wait(&status)==-1)&&(errno==EINTR));
/* 信号灯只能够被删除一次的 */
if(i==1)
	if(del_semaphore(semid)==-1)
	fprintf(stderr,"[%d]:Destroy Semaphore Error:%s\n\a",
	getpid(),strerror(errno));
exit(0);
}


[root@localhost sem]# gcc -o semOp semOp.c
[root@localhost sem]# ./semOp 5
pid=22500
i=0,childpid=0
i=1,childpid=0
i=2,childpid=0
i=3,childpid=0
[i=3]-->[Process=22503]-->[Parent=22502]-->[Child=22504]
[i=2]-->[Process=22502]-->[Parent=22501]-->[Child=22503]
[i=1]-->[Process=22501]-->[Parent=22500]-->[Child=22502]
[i=0]-->[Process=22500]-->[Parent=8740]-->[Child=22501]
i=4,childpid=0
[i=4]-->[Process=22504]-->[Parent=22503]-->[Child=22505]
[i=5]-->[Process=22505]-->[Parent=22504]-->[Child=0]
[root@localhost sem]# ipcs


------ Shared Memory Segments --------
key        shmid      owner      perms      bytes      nattch     status
0x02012088 262144     root      666        4          2
0x01012155 294913     root      666        4          1
0x00000000 32931844   root      700        32768      2          dest
0x00000000 32964613   root      700        131072     2          dest


------ Semaphore Arrays --------
key        semid      owner      perms      nsems
0x0101fbdd 0          root      600        1
0x16012088 98305      root      666        1
0xde012088 131074     root      666        1
0x0101fa03 163843     root      600        1
0x0b012155 196612     root      666        1
0x6f012155 229381     root      666        1


------ Message Queues --------
key        msqid      owner      perms      used-bytes   messages


[root@localhost sem]#  

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!
提供的源码资源涵盖了小程序应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!
提供的源码资源涵盖了Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值