文件重定义冲突的分析与解决

引言

有时候看到论坛上有人问编译时重定义现象的问题。这个问题与头文件包含冲突有关,改一改相关头文件包含就会让问题消失。我在这里要以 <windows.h> <winsock.h> <winsock2.h>这3个头文件为例分析为什么会产生这种原因;然后给出一种不会产生冲突的做法;最后就此例的分析给出结论。本文对于已经知道冲突原因的读者是多余的,但对于一时没有时间去解的人还是有一定的参考价值的。

一个重命名的例子

先写一个socket基本操作类:

// SocketBase.h
#ifndef _SOCKET_BASE_H_
#define _SOCKET_BASE_H_
#include <winsock2.h>
#pragma comment(lib, "ws32_2.lib")
class Socket
{
};
#endif
写一个TCP监听类,它从Socket派生:
// TcpListener.h
  #ifndef _TCP_LISTENER_H_
#define _TCP_LISTENER_H_
#include "SocketBase.h"
class TcpListener : Socket
{
};
#endif
在基于MFC的工程中用 TcpListener 监听客户连接,同时这个地方须要用到 Windows 某些头文件。
#include "stdafx.h"  
#include "TcpListener.h"
void fun()
{
  TcpListener* listen = new TcpListener;
  ……
}
下面是头文件包含关系:
Stdafx.h -> windows.h -> winsock.h winsock2.h
TcpListener.h -> SocketBase.h -> winsock2.h
  编译,出现N多重定义错误。这个错误与stdafx.h中的 windows.h 和TcpListener.h 的 winsock2.h 有关,下面说两种消除错误的方法。

针对本工程中消除编译

错误产生的原因是 windows.h 中有:
  #include <winsock.h>
  #include <winsock2.h>
  产生重定义的是 windows.h 中的 winsock.h 相关定义与 TcpListener.h 中 winsock2.h 相关定义冲突。相同头文件是不会冲突的,因为有 "#ifndef #define …. #endif";如果windows中只包含一个winsock2.h就不会产生重定义了。

现在我们只要把 SocketBase.h 中的 "include <winsock2.h> 和 #pragma…"注释了就编译通过了。但经过注释的 SocketBase.cpp 与 TcpListener.cpp 单独编译就通不过了。这种只是针对特定的环境下解决问题,我们得想一个比较专业的办法。

一个可被接受的解决方法

Winsock2.h 与 windows.h 中的 winsock.h 相关项的重定义要在 SocketBase.h 中避免。在 SocketBase.h加一些编译条件就可以做到这一点,经过修改的 SockBase.h 如下:

#ifndef _SOCKET_BASE_H_
#define _SOCKET_BASE_H_
#ifndef _WINSOCKAPI_      // 没有包含winsock.h
#define _WINSOCKAPI_      // 避免再包含winsock.h
  #ifndef _WINSOCK2API_   // 没有包含winsock2.h
  #define _WINSOCK2API_   // 避免再包含winsock2.h
    #include <winsock2.h>
    #pragma comment(lib, "ws32_2.lib")
  #endif
#endif
class Socket
{
};
#endif
经过这样修改,就能做到 winsock.h 与 winsock2.h 中的相关项重定义了。

结论

通过对上述例子的分析解决,同样在其它类似的问题中适用。如果有更简单的方法避免重定义的情况出现,请告诉我一下。

参考

Win32 sdk中的 windows.h;
Win32 sdk中的 winsock.h;
Win32 sdk中的 winsock2.h;

时间: 2024-11-05 14:38:51

文件重定义冲突的分析与解决的相关文章

关于 WinSock2.h 与 WinSock.h 文件重定义冲突的 解决办法

事实上一直都用的是Winsock1的,还从来没有用过Winsock2,今天开始使用它,但是出现了一大堆的错误,都是那个C2011和C2375之类的错误.当时就想,是不是2和1发生了冲突了?但是我没有包含过1的头文件啊?! 后来一查,在Windows.h里面发现了这么一段代码: #ifndef _MAC#include <winperf.h>#include <winsock.h>#endif 我就一个倒.... 原来是这里包含了.... 对付这个的终极解决办法是: #define 

win8 1-win8.1+vs2012 d3d11生成的时候出现大量宏重定义

问题描述 win8.1+vs2012 d3d11生成的时候出现大量宏重定义 如图, 网上有解决办法说是把vc++包含目录中的$(WindowsSDK_IncludePath) 放在 $(DXSDK_DIR)Include 之前, 但是我这么做不仅没有解决, 还在引入SDK Sample中的"d3dx11Effect.h"之后引入了新的错误: 请大神指教这个问题究竟如何解决, 谢谢! include: Common里面是用vs2012重新编译过的Effects库 解决方案 看下有没有提供

解决C++中重定义的方法总结_C 语言

C++由于头文件重复包含了所定义的变量或者常量,编译器就会报重复定义的错误.如果你碰见这样的问题可以考虑重下面几个方面去解决: 1.在出现重定义错误的头文件加上:#ifndef FileName_H_#define FileName_H_ ....(头文件内容)#endif注意如果FileName_H_这个名字已经被使用,将会出现未定义问题(这里不讨论),这是你保证FileName_H_唯一就可以. 2.在出现重定义错误的头文件加上这一句:#pragma once 就可以解决(VS建立的类都会默

Java 编程技术中汉字问题的分析及解决,文件操作

编程|汉字|解决|问题 在基于 Java 语言的编程中,我们经常碰到汉字的处理及显示的问题.一大堆看不懂的 乱码肯定不是我们愿意看到的显示效果,怎样才能够让那些汉字正确显示呢?Java 语言 默认的编码方式是UNICODE ,而我们中国人通常使用的文件和数据库都是基于 GB2312 或者 BIG5 等方式编码的,怎样才能够恰当地选择汉字编码方式并正确地处理汉字的编 码呢?本文将从汉字编码的常识入手,结合 Java 编程实例,分析以上两个问题并提出 解决它们的方案. 现在 Java 编程语言已经广

c++-C++ 头文件已经防止重定义了,但是还是出现了错误

问题描述 C++ 头文件已经防止重定义了,但是还是出现了错误 test.cpp: #include"head.h" int i=3; main.cpp #include"head.h" #include using namespace std; extern int i; int main(){ cout << i<< endl; system("pause"); return 0; } head.h #ifndef HE

IBM HTTP Server Plugin默认plugin-key.kdb 密钥数据库文件默认个人证书密码过期分析与解决方法

IBM HTTP Server Plugin默认plugin-key.kdb 密钥数据库文件默认个人证书密码过期分析与解决方法    http://www.Java2Class.net/IBM/WebSphere/IBM_IHS_PLG_Personal_Certificate_Password_Expired_2012.04.26.pdf

vc-VC 两个文件包涵同一个头文件 提示变量重定义

问题描述 VC 两个文件包涵同一个头文件 提示变量重定义 总共四个文件.分别是定义main函数的text.cpp定义了一个类 的 a.cpp和a.h还有一个b.h text.cpp代码 #include ""a.h"" #include ""b.h""int main(){ //some code} a.h代码 #pragma onceclass Ca{ //some code}; a.cpp代码 #include "

socket-【C++】符号重定义,头文件包含问题

问题描述 [C++]符号重定义,头文件包含问题 #pragma once #include "Common.h" #include "WinSock.h" class CSockOperation { public: int Send(SOCKET socket,const char* buf,int len);//失败返回错误值 int Recv(SOCKET socket,char* lpBuf,int nBufLen);//失败返回错误值 int CSockO

c语言编程-调用文件重命名函数rename失败,errno的值为17,如何解决?

问题描述 调用文件重命名函数rename失败,errno的值为17,如何解决? void onMenuDele(char *id) { FILE *p=fopen("f:employee.txt","r"); //打开文件 if(p==NULL) { printf("Sorry!employee.txt cannot open!n"); return ; } FILE *fp2=fopen("f:\employ.txt",&q