转载请注明出处: http://blog.csdn.net/luotuo44/article/details/38350633
Libevent提供给用户的可见多线程API都在thread.h文件中。在这个文件提供的API并不多。基本上都是一些定制函数,像前面几篇博文说到的,可以为Libevent定制用户自己的多线程函数。
开启多线程:
Libevent默认是不开启多线程的,也没有锁、条件变量这些东西。这点和前面博客说到的"没有定制就用Libevent默认提供",有所不同。只有当你调用了evthread_use_windows_threads()或者evthread_use_pthreads()或者调用evthread_set_lock_callbacks函数定制自己的多线程、锁、条件变量才会开启多线程功能。其实,前面的那两个函数其内部实现也是定制,在函数的内部,Libevent封装的一套Win32线程、pthreads线程。然后调用evthread_set_lock_callbacks函数,进行定制。
thread.h文件只提供了定制线程的接口,并没有提供使用线程接口。这点很像前面说到的Libevent日志和内存分配。其实这也很好理解。因为都是你提供定制的线程函数。你都能提供了,你肯定有办法使用,没必要要Libevent提供一些API给你使用。
如果用户为libevent开启了多线程,那么libevent里面的函数就会变成线程安全的。此时主线程在使用event_base_dispatch,别的线程是可以线程安全地使用event_add把一个event添加到主线程的event_base中。具体的工作原理可以参考《evthread_notify_base通知主线程》。
锁和条件变量结构体:
Libevent允许用户定制自己的锁和条件变量。其实现原理和前面说到的日志和内存分配一样,都是内部有一个全局变量。定制自己的锁和条件变量,就是对这个全局变量进行赋值。
锁结构:
//thread.h文件
struct evthread_lock_callbacks {
//版本号,设置为宏EVTHREAD_LOCK_API_VERSION
int lock_api_version;
//支持的锁类型,有普通锁,递归锁,读写锁三种
unsigned supported_locktypes;
//分配一个锁变量(指针类型),因为不同的平台锁变量是不同的类型
//所以用这个通用的void*类型
void *(*alloc)(unsigned locktype);
void (*free)(void *lock, unsigned locktype);
int (*lock)(unsigned mode, void *lock);
int (*unlock)(unsigned mode, void *lock);
};
目前Libevent支持的locktype