订阅者对一个或多个频道感兴趣,只需接收感兴趣的消息,不需要知道什么样的发布者发布的。
这种发布者和订阅者的解耦合可以带来更大的扩展性和更加动态的网络拓扑。
使用C语言信号量需要包含: #include <semaphore.h>
结构体为: sem_t
- 函数:
-
int sem_init(sem_t *sem, int pshared, unsigned int value);
- 对sem_t 指定的信号量进行初始化。
- pshared 为共享选择 为0时表示它是当前进程的局部信号量,否则其他进程能够共享这个信号(这个参数受版本的影响),一般不支持进程间共享信号量,pshared传递一个非零将会使函数返回ENOSYS错误。
- value 为整数型初始值
-
int sem_post(sem_t * sem);
- 给信号量的值加上一个1,它是一个原子操作即同时对同一个信号量做加“1”操作的两个线程是不会冲突的;而同时对同一个文件进行读、加和写操作的两个程序就有可能会引起冲突。信号量的值永远会正确地加一个“2”--因为有两个线程试图改变它。
-
int sem_wait(sem_t *sem); 和 int sem_trywait(sem_t *sem);
- 函数sem_trywait,它是sem_wait的非阻塞搭档。sem_trywait是一个立即返回函数,不会因为任何事情阻塞。根据其返回值得到不同的信息。如果返回值为0,说明信号量在该函数调用之前大于0,但是调用之后会被该函数自动减1,至于调用之后是否为零则不得而知了。如果返回值为EAGAIN说明信号量计数为0。
-
int sem_getvalue(sem_t *sem, int *valp);
- 获得信号量sem的值,并保存到valp中。
-
int sem_destroy (sem_t *sem);
- 这个函数也使用一个信号量指针做参数,归还自己占据的一切资源。在清理信号量的时候如果还有线程在等待它,用户就会收到一个错误。 然而在linux的线程中,其实是没有任何资源关联到信号量对象需要释放的,因此在linux中,销毁信号量对象的作用仅仅是测试是否有线程因为该信号量在等待。如果函数返回0说明没有,正常注销信号量,如果返回EBUSY,说明还有线程正在等待该信号量的信号。
-
使用步骤
- 声明信号量sem_t sem1;
- 初始化信号量sem_init(&sem1,0,1);
- sem_post和sem_wait函数配合使用来达到线程同步
- 释放信号量int sem_destroy (&sem1);
-
2. 统一事件源。libevent对i/o事件、信号和定时事件提供统一的处理。
3. 线程安全。libevent使用libevent_pthreads库来提供线程安全支持。
4. 基于reactor模式的实现。
- API:
- struct event_base *event_base_new(void);
- 分配并且返回一个新的具有默认设置的 event_base。函数会检测环境变量,返回一个到 event_base 的指针。如果发生错误,则返回 NULL。选择各种方法时,函数会选择 OS 支持的最快方法。
- void event_base_free(struct event_base *base);
- 有申请就有释放
- int event_base_dispatch(struct event_base *base);
- event_base_dispatch ()等同于没有设置标志的 event_base_loop ( )。所以event_base_dispatch ()将一直运行,直到没有已经注册的事件了,或者调用 event_base_loopbreak()或者 event_base_loopexit()为止。 如果想在移除所有已注册的事件之前停止活动的事件循环,可以调用两个稍有不同的函数 。 int event_base_loopexit(struct event_base *base,const struct timeval *tv); int event_base_loopbreak(struct event_base *base);
- struct event_base *event_base_new(void);
- API:
- redisAsyncContext *redisAsyncConnect(const char *ip, int port);
- 异步连接redis服务器
- redisLibeventAttach
- libevent 事件注册
- int redisAsyncSetConnectCallback(redisAsyncContext *ac, redisConnectCallback *fn);
- 用于设置异步上下文中的建链回调函数。
- void redisAsyncDisconnect(redisAsyncContext *);
- 断开异步连接
- void redisAsyncFree(redisAsyncContext *);
- 释放内存
- int redisAsyncCommand(redisAsyncContext *ac, redisCallbackFn *fn, void *privdata, const char *format, ...);
- redisAsyncCommand函数,是异步API中用于向Redis发送命令的函数。该函数与同步API中发送命令的函数redisCommand类似,同样支持printf式的可变参数。 这里的fn和privdata分别表示收到命令回复后要调用的回调函数及其参数。因为Redis是单线程处理命令,因此当客户端使用异步API与事件库的结合之后,命令就自动的管道化了。也就是客户端在单线程模式下,发送命令的顺序和接收回复的顺序是一致的。因此,当发送命令时,就会将回调函数fn和参数privdata封装成回调结构redisCallback,并将该结构记录到单链表或者字典中。当收到回复后,就会依次得到链表或者字典中的redisCallback结构,调用其中的回调函数。
- redisAsyncContext *redisAsyncConnect(const char *ip, int port);
- CRedisPublisher
- init
- 初始化libevent, 信号量
- uninit
- 释放空间
- connect
- 异步连接到redis服务器
- 将(libevent)事件绑定到redis服务上
- 创建事件处理线程
- 设置连接回调
- 设置断开连接回调
- 给信号值加一
- disconnect
- 断开连接
- 释放回调
- publish
- 发送消息
- connect_callback
- 连接成功
- disconnect_callback
- 断开连接
- command_callback
- 执行命令回调
- event_thread
- 事件处理线程
- event_proc
- 事件处理
- init
- CRedisPublisher
- init
- 初始化libevent, 信号量
- uninit
- 释放空间
- connect
- 异步连接到redis服务器
- 将(libevent)事件绑定到redis服务上
- 创建事件处理线程
- 设置连接回调
- 设置断开连接回调
- 给信号值加一
- disconnect
- 断开连接
- 释放回调
- subscribe
- 接收订阅的消息
- connect_callback
- 连接成功
- disconnect_callback
- 断开连接
- command_callback
- 执行命令回调
- event_thread
- 事件处理线程
- event_proc
- 事件处理
- init