使用Qt库时出现winsock2.h和windos.h包含顺序错误

来源:互联网 发布:java编程九九乘法表 编辑:程序博客网 时间:2024/05/16 01:14

同时使用Qt与mms.lite库时时,在互相包含头文件时出现了winsock2.h和windos.h包含顺序错误的出错信息(一大堆重定义错误)。查找mms.lite库发现在sysincs.h中已经有对这两个头文件包含顺序的处理,如下所示:

/* winsock2.h MUST ALWAYS BE BEFORE windows.h to override defs in */
/* winsock.h (included by windows.h). Any module that includes windows.h*/
/* before sysincs.h, must also include winsock2.h before windows.h. */
#include <winsock2.h>  /* must be before windows.h */
#include <windows.h>

在google上查Qt这方面的问题,得到解决方法如下:

编辑D:\Qt\Qt5.0.2\5.0.2\msvc2010_opengl\include\QtCore下的qt_windows.h

在#include <windows.h>之前添加一行:

#define WIN32_LEAN_AND_MEAN

原因解释:

先让我们看看winsock2.h的内容,在文件开头有如下宏定义:

#ifndef _WINSOCK2API_

#define _WINSOCK2API_

#define _WINSOCKAPI_    /* Prevent inclusion of winsock.h in windows.h */

_WINSOCK2API_ 很容易理解,这是最常见的防止头文件重复包含的保护措施。_WINSOCKAPI_的定义则是为了阻止对老文件winsock.h的包含,即是说,如果用户先包含了winsock2.h就不允许再包含winsock.h了,否则会导致类型重复定义。这是怎样做到的呢?很简单,因为winsock.h的头部同样存在如下的保护措施:

#ifndef _WINSOCKAPI_

#define _WINSOCKAPI_

再回过头来看winsock2.h,在上述内容之后紧跟着如下宏指令:

/*

* Pull in WINDOWS.H if necessary

*/

#ifndef _INC_WINDOWS

#include <windows.h>

#endif /* _INC_WINDOWS */

其作用是如果用户没有包含windows.h(_INC_WINDOWS在windows.h中定义)就自动包含它,以定义WinSock2.0所需的类型和常量等。

现在切换到windows.h,查找winsock,我们会惊奇的发现以下内容:

#ifndef WIN32_LEAN_AND_MEAN

#include <cderr.h>

#include <dde.h>

#include <ddeml.h>

#include <dlgs.h>

#ifndef _MAC

#include <lzexpand.h>

#include <mmsystem.h>

#include <nb30.h>

#include <rpc.h>

#endif

#include <shellapi.h>

#ifndef _MAC

#include <winperf.h> 



#if(_WIN32_WINNT >= 0x0400)

#include <winsock2.h>

#include <mswsock.h>

#else

#include <winsock.h>

#endif /* _WIN32_WINNT >=   0x0400 */

#endif

// 这里省略掉一部分内容

#endif /* WIN32_LEAN_AND_MEAN */

看到没?windows.h会反向包含winsock2.h或者winsock.h!相互间的包含便是万恶之源!

下面具体分析一下问题是怎么发生的。

错误情形1:我们在自己的工程中先包含winsock2.h再包含windows.h,如果WIN32_LEAN_AND_MEAN未定义且 _WIN32_WINNT大于或等于0x400,那么windows.h会在winsock2.h开头被自动引入,而windows.h又会自动引入 mswsock.h,此时,mswsock.h里所用的socket类型还尚未定义,因此会出现类型未定义错误。

错误情形2:先包含 windows.h再包含winsock2.h,如果WIN32_LEAN_AND_MEAN未定义且_WIN32_WINNT未定义或者其版本号小于 0x400,那么windows.h会自动导入旧有的winsock.h,这样再当winsock2.h被包含时便会引起重定义。

这里要说明的是,宏WIN32_LEAN_AND_MEAN的作用是减小win32头文件尺寸以加快编译速度,一般由AppWizard在stdafx.h中自动定义。_WIN32_WINNT的作用是开启高版本操作系统下的特殊函数,比如要使用可等待定时器(WaitableTimer),就得要求 _WIN32_WINNT的值大于或等于0x400。因此,如果你没有遇到上述两个问题,很可能是你没有在这些条件下进行网络编程。

0 0
原创粉丝点击