(2)其他函数。
int sem_wait (sem_t* sem);
int sem_trywait (sem_t* sem);
int sem_post (sem_t* sem);
int sem_getvalue (sem_t* sem);
int sem_destroy (sem_t* sem);
功能:sem_wait和sem_trywait相当于P操作,它们都能将信号量的值减一,两者的区别在
于若信号量的值小于零时,sem_wait将会阻塞进程,而sem_trywait则会立即返回。
sem_post相当于V操作,它将信号量的值加一,同时发出唤醒的信号给等待的进程(或线程)。
sem_getvalue 得到信号量的值。
sem_destroy 摧毁信号量。
如果某个基于内存的信号灯是在不同进程间同步的,该信号灯必须存放在共享内存区中,这要只要该共享内存区存在,该信号灯就存在。
二.不同进程间的互斥和同步,使用如下API(有名信号量)
有名信号量:可以用于线程间或进程间同步
创建打开有名信号量
sem_t *sem_open(const char *name, int oflag);
sem_t *sem_open(const char *name, int oflag,
mode_t mode, unsigned int value);
成功返回信号量指针;失败返回SEM_FAILED,设置errno
name是文件路径名,但不能写成/tmp/a.sem这样的形式,因为在linux下,sem都是在/dev/shm目录下,可写成"/mysem"或"mysem",创建出来的文件都是"/dev/shm/sem.mysem",mode设置为0666
value设置为信号量的初始值.所需信号灯等已存在条件下指定O_CREAT|O_EXCL却是个错误。
关闭信号量,进程终止时,会自动调用它
int sem_close(sem_t *sem);
成功返回0;失败返回-1,设置errno
删除信号量,立即删除信号量名字,当其他进程都关闭它时,销毁它
int sem_unlink(const char *name);
成功返回0;失败返回-1,设置errno
等待信号量,测试信号量的值,如果其值小于或等于0,那么就等待(阻塞);一旦其值变为大于0就将它减1,并返回
int sem_wait(sem_t *sem);
int sem_trywait(sem_t *sem);
成功返回0;失败返回-1,设置errno
当信号量的值为0时,sem_trywait立即返回,设置errno为EAGAIN
如果被某个信号中断,sem_wait会过早地返回,设置errno为EINTR
挂出信号量,给它的值加1,然后唤醒正在等待该信号量的进程或线程
int sem_post(sem_t *sem);
成功返回0;失败返回-1,不会改变它的值,设置errno
该函数是异步信号安全的,可以在信号处理程序里调用它
获取信号量的值
int sem_getvalue(sem_t *sem, int *sval);
成功返回0,sval设置为信号量的值;失败返回-1,设置errno
如果当前信号量已上锁,sval的值或者为0,或者为某个负数,它的绝对值等于正在等待该信号量的进程和线程数,linux采用了前者
int sem_timedwait(sem_t *sem, const struct timespec *abs_timeout);
成功返回0;失败返回-1
信号处理程序会中断它,设置errno为EINTR
abs_timeout为绝对时间
struct timespec {
time_t tv_sec; /* Seconds */
long tv_nsec; /* Nanoseconds [0 .. 999999999] */
};
posix有名信号灯使用的几点注意 :1.Posix有名信号灯的值是随内核持续的。也就是说,一个进程创建了一个信号灯,这个进程结束后,这个信号灯还存在,并且信号灯的值也不会改动.2。当持有某个信号灯锁的进程没有释放他就终止时,内核并不给该信号灯解锁。
信号量和互斥锁的区别之一是:互斥锁必须总是由锁住它的线程解锁,信号量的post却不必由执行过它的等待操作的同一线程执行.