libevent是一個使用很廣泛的網絡庫,今天想了解下它。於是去git clone了一份源碼,用vs2005的命令行:nmake -f makefile.nmake編譯之,順利編譯通過,生成三個靜態庫:libevent.lib,libevent_core.lib,libevent_extras.lib
一切都很順利,隨便寫了個test.cpp,代碼如下:
#include "event2/event.h" void test_libevent() { } int main(int argc,char* argv[]) { test_libevent(); return 0; }
結果編譯不通過,出現錯誤“\wspiapi.h(44) : error C2894: templates cannot be declared to have 'C' linkage”。
找了下原因,通過生成預處理文件發現wspiapi.h是被event2\util.h包含進去了。libevent所有頭文件包含均聲明 extern "C" , 因此wspiapi.h被以C頭文件的形式包含,但是wspiapi.h里面有模板 !
結果編譯器認為該用C的方式去處理這個頭文件,結果C代碼中出現了template,於是編譯器慫了,很尷尬的告訴程序員,C代碼里面不能包含C++的模板,報錯!
解決辦法:
方法一:把test.cpp改名字叫test.c,這樣用C來寫代碼;
方法二:按照防止頭文件重復包含的原理,在test.cpp最開頭的部分定義#define _WSPIAPI_H_ 與 #define _WINSOCKAPI_。
最終成果:
#define _WSPIAPI_H_ #define _WINSOCKAPI_ #include "event2/event.h" #include "event2/event_struct.h" #include <iostream> using namespace std; event ev; timeval tv; void time_cb(int fd,short eventid,void* argc) { cout<<"timer wakeup"<<endl; event_add(&ev,&tv); } void test_libevent() { event_base* eventbase = event_base_new(); tv.tv_sec = 1; tv.tv_usec = 0; event_assign(&ev,eventbase,-1,0,time_cb,NULL); event_add(&ev,&tv); event_base_dispatch(eventbase); event_base_free(eventbase); } int main(int argc,char* argv[]) { WORD wVersionRequested = MAKEWORD( 2, 1 ); WSADATA wsaData; WSAStartup(wVersionRequested,&wsaData); test_libevent(); WSACleanup(); return 0; }
需要注意的是,鏈接過程中會出現warning,要求添加/NODEFAULTLIB選項,這時候要去掉所有默認lib,添加:libevent.lib libevent_extras.lib libevent_core.lib Ws2_32.lib libcmtd.lib libcpmtd.lib
這問題還是有點坑爹的...
再PS一下,libevent默認編譯的是release版本的,如果需要編譯debug版本的,修改一下makefile.nmake:
#CFLAGS=$(CFLAGS) /Ox /W3 /wd4996 /nologo
修改為:
#CFLAGS=$(CFLAGS) /D_DEBUG /Od /W3 /wd4996 /nologo