p { margin-bottom: 0.21cm; }
信号量:通过 PV 操作输出 ABAB····
修改:initsem.c
#include "pv.h"
int initsem(key_t key , int initVal )
{
int status = 0, semid;
if((semid = semget(key,1,SEMPERM|IPC_CREAT|IPC_EXCL)) == -1)
{
if(errno == EEXIST)
{
semid = semget(key,1,0);
// semun arg;
// arg.val = initValue;
// printf("%d ,%d \n", semctl(semid,0,GETVAL,arg),arg.val);
// arg.val = initValue;
// semctl(semid,0,SETVAL,arg);
}
else
{
semun arg;
arg.val = initVal ;
status = semctl(semid,0,SETVAL,arg);
}
if(semid ==-1 || status == -1)
{
perror("initsem failed\n");
return -1;
}
return semid;
}
// 主函数
#include "pv.h"
int main()
{
key_t semkey_A=0x456;
key_t semkey_B=0x789;
int semid_A,semid_B;
if ((semid_A=initsem(semkey_A,1))<0) exit(1);
if ((semid_B=initsem(semkey_B,0))<0) exit(1);
if (fork()==0)
{
int i;
for (i=0;i<100;i++)
{
p(semid_A);
printf("A\n");
v(semid_B);
}
}
else
{
int j;
for (j=0;j<100;j++)
{
p(semid_B);
printf("B\n");
v(semid_A);
}
// waitpid();
}
return 0;
}
上面程序会出现第一次运行的时候可以正确的输出 ABABABAB`````````, 但是最后不能退出程序,第二次的时候会输出全 A 全 B , 原因为信号量一直保 留在内存中。然后我的解决方案为每次都初始化信号量的值。通过semctl(semid,0,SETVAL,arg) 设置 , 如上面的程序被注释掉的 。
还有一个问题就是第二个程序也就是可以正常输出 ABABAB 的程序,我把输出的 \n 去掉会出现 aaaaaaaabbbbbbbb 的情况,然后通过资料查明, \n 对输出是有影响的。如果 printf 不加 \n, 且 IO 缓冲区没满的话,只能等到进程退出的时候才会刷新缓冲区。( \n 的话会刷新缓冲区),输出 aaaa 再 bbbb 是因为子进程退出刷新了缓冲区,然后父进程再退出刷新自己的缓冲区。