libevent在windows下用visual studio编译时出现error C2894错误的原因与解决方法

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

posted @ 2014-04-17 17:01  *神气*  阅读(4006)  评论(0编辑  收藏  举报