有时候为了简便,在包含头文件时,使用匿名名称空间来避免重复包含的引起的multiple definition of这类错误。
但是,要注意的是,如果两个文件包含同一个匿名名称空间的类,然后通过这个类生成对象时。他们的适配类型并不同!!!!
也就是如果在一个类中有一个成员函数使用名称空间里面的类的类型做参数,在另外一个文件包含匿名名称空间里面的类使用此对象传入参数到此函数时,类型并不匹配。就可能会出现一个undefined reference to `ez_poll::setRecvCallback(std::tr1::function<void ()((anonymous namespace)::buffer*)>)' 如此的错误:下是我写的一个程序出错的例子
文件ez_buffer_cpp.h
namespace { //添加namespace会导致错误
class buffer
{
private:
deque<vector<char>*> _blocks;
public:
buffer();
~buffer();
char *getData();
bool addData(const char *message);
bool consume(size_t num);
void append(vector<char> *data);
};
}
文件epoll.h
#include "ez_buffer_cpp.h" //注意 包含了前一个文件
using namespace std::tr1;
class ez_poll{
private:
int server_fd;
struct epoll_event *events;
buffer _readBuffer;
buffer _sendBuffer;
int epoll_fd;
int _maxRevSize;
int _maxSendSize;
pthread_t thread;
function<void(buffer*)> _recvCallback;
static bool setnonblocking(int sockfd); //有了static声明就不用在函数中使用static
public:
ez_poll();
~ez_poll();
bool init(int port);
static void run(void *arg); //在类中封装多线程必须使用static
bool thread_run();
void thread_join();
void setRecvCallback(function<void(buffer*)> callback);
};
文件main.cpp
//test
#include "epoll.h" //这里包含了epoll.h间接的包含了ez_buffer_cpp.h 但是这次包ez_buffer_cpp.h由于是匿名名称空间,所以会随机生成名称空间造成buffer类与ez_poll成员类的参数不同!所以调用会出错
//using namespace std::tr1;
void fun(buffer *data)
{
std::cout<<"fdsf"<<std::endl;
}
int main()
{
ez_poll server;
server.init(8848);
server.thread_run();
std::cout<<"what the fox say"<<std::endl;
std::tr1::function<void(buffer*)> qq(fun);
server.setRecvCallback(qq); //这里会出错
for(;;)
{
std::cout<<"running......"<<std::endl;
sleep(1);
}
server.thread_join();
}
好的解决方式还是使用#ifndef.....,并且把类的声明和实现写在两个文件中