Chinaunix首页 | 论坛 | 博客
  • 博客访问: 494006
  • 博文数量: 133
  • 博客积分: 1235
  • 博客等级: 少尉
  • 技术积分: 1201
  • 用 户 组: 普通用户
  • 注册时间: 2010-09-08 19:59
文章分类

全部博文(133)

文章存档

2023年(12)

2022年(3)

2018年(2)

2017年(4)

2016年(4)

2015年(42)

2014年(1)

2013年(12)

2012年(16)

2011年(36)

2010年(1)

分类: LINUX

2011-11-21 10:46:44

一.进程体内各线程间的互斥和同步,使用如下API(无名信号量,基于内存的信号量)
(1)、sem_init
功能:         用于创建一个信号量,并初始化信号量的值。
头文件:      
函数原型:     int sem_init (sem_t* sem, int pshared, unsigned int value);
函数传入值:   sem:信号量。
                   pshared:决定信号量能否在几个进程间共享。由于目前LINUX还没有实现进
                               程间共享信息量,所以这个值只能取0。
(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却不必由执行过它的等待操作的同一线程执行.

阅读(1869) | 评论(1) | 转发(1) |
给主人留下些什么吧!~~

星期五啦2011-11-23 01:01:14

好帖子~