Semaphore: for managing access to resources
Shared memory:for highly efficient data sharing between programs
Messaging:for an easy way of passing data between programs
1.Semaphores
一般来说一个thread可能同时占有多个资源,所以Linux中总是对一组sem进行操作, 不过大部分情况下我们只要对一两个sem进行操作。
#include <sys/sem.h>
int semget(key_t key,int num_sems,int sem_flags);
key_t 像文件名一样表示一个资源
num_sems 信号量的大小
函数返回int型sem_id
实际上semget函数跟fopen函数很类似
key_t类似文件名,是唯一的
sem_id则是fopen函数返回的FILE * file stream
所以可以有不同的process可以用不同的信号量对应同一个资源
int semctl(int sem_id,int sem_num,int command,...);
sem_num 是semaphore number,use it when work with arrays of semaphores
如果是0 表示the first and only semphore
第四个参数通常是下面的结构
struct semun{
int val;
struct semid_ds * buf;
unsigned short * array;
}
command 为SETVAL时就初始化sem的值为semun.val
command 为IPC_RMID时删除sem_id如果不再使用
int semop(int sem_id,struct sembuf * sem_ops,size_t num_sem_ops);
struct sembuf{
short sem_num;
short sem_op;
short sem_flay;
}
sem_num一般是0,除非有一组信号量
sem_op 通常是+1 或者-1 表示可以改变的值
sem_flg 通常是SEM_UNDO 这样process 没释放sem就结束的话,操作系统会自动release the semaphore
下面是一个例子
Shared Memory是能在两个Process之间高效的传输数据一种IPC
一个Process创建一块SM,其他Process可以将其作为自己地址空间的一部分访问。当一个Process想SM中写入时,其他的Process都会得知
#include <sys/shm.h>
int shmget(key_t key,size_t size,int shmflg);
size是内存大小
函数返回shm_id
void * shmat(int shm_id,const void *shm_addr,int shmflg);
当创建一个SM后,需要将它加入到其他进程的地址空间中
shm_addr通常为NULL,让系统自动简历映射
shmflg 如果是SHM_RND表示 SHM_RDONLY表示这块SM对于这个进程是只读的
int shmdt(const void *shm_addr);makes that memory unavailable to the current process
int shmctl(int shm_id,int cmd,struct shmid_ds *buf);
shmid_ds{
uid_t shm_perm.uid;
uid_t shm_perm.gid;
mode_t shm_perm.mode;
}
command可以为IPC_STAT IPC_SET IPC_RMID
message queue一般比pipe高效简单,但传递的数据大小仍然有限制
#include <sys/msg.h>
int msgget(key_t key,int msgflg);
int msgsnd(int msqid,const void *msg_ptr,size_t msg_sz,int msgflg);
添加一个message到message queue
int msgrcv(int msqid,void * msg_ptr,size_t msg_sz,long int msgtype,int msgflg);
从message queue中取message
int msgctl(int msqid,int command,struct msqid_ds *buf);
struct msqid_ds{
uid_t msg_perm.uid;
uid_t msg_perm.gid;
mode_t msg_perm.mode;
}