#include <stdio.h> //这个是sem_init.c
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
union semun {
int val; /* cmd为SETVAL需要的整数值 */
struct semid_ds *buf; /* cmd填IPC_STAT, IPC_SET需要的缓冲区首地址 */
unsigned short *array; /* cmd为GETALL, SETALL时需要的数组首地址 */
struct seminfo *__buf; /* cmd填IPC_INFO时需要的缓冲区首地址 (Linux-specific) */
};
int main(int argc, const char *argv[])
{
key_t key = ftok(".", 176);
//创建信号量集合,也就是道具集合,这里包含1个道具
int semid = semget(key, 1, IPC_CREAT|0666);
//设置信号量的值,也就是道具的数量
union semun arg;
arg.val = 1;
semctl(semid, 0, SETVAL, arg);
return 0;
}
//本函数是为了验证2个进程同时写同一个文件,用信号量和不用信号量的明显区别
//代码需要配合sem_init.c使用,先运行sem_init.c,再开2个终端几乎同时运行本程序
//代码现象是加信号量和不加信号量的结果是完全不一样的,不加会发生乱序/覆盖!!!
//这个是semone.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
union semun {
int val; /* cmd为SETVAL需要的整数值 */
struct semid_ds *buf; /* cmd填IPC_STAT, IPC_SET需要的缓冲区首地址 */
unsigned short *array; /* cmd为GETALL, SETALL时需要的数组首地址 */
struct seminfo *__buf; /* cmd填IPC_INFO时需要的缓冲区首地址 (Linux-specific) */
};
int main(int argc, const char *argv[])
{
key_t key = ftok(".", 176);
//创建信号量集合,也就是道具集合,这里包含1个道具
int semid = semget(key, 1, IPC_CREAT|0666);
//拿放道具
struct sembuf sops = {
.sem_num = 0, //操作第0类道具
.sem_op = -1, //道具数量减1
.sem_flg = SEM_UNDO, //进程结束自动还原
};
//semop(semid, &sops, 1); //占用1次,术语叫P操作
//打开普通文件,这个文件得事先创建好!
FILE *fp = fopen("./test.txt", "a+"); //注意打开方式选a+不要选r+
int i = 20;
while(i--){
//写普通文件,写入PID和i以做区分!
fprintf(fp, "PID: %d, i: %d\n", getpid(), i);
fflush(fp);
printf("PID: %d write 'test.txt' onece\n", getpid());
sleep(1);
}
fclose(fp); //关闭刷新普通文件
sops.sem_op = 1; //道具数量+1
//semop(semid, &sops, 1); //释放1次,术语叫V操作
printf("semop()2 is success\n");
return 0;
}