繁体   English   中英

C ++重定义头文件(winsock2.h)

[英]C++ Redefinition Header Files (winsock2.h)

如何防止包含头文件两次? 问题是我包括了 MyClass.h中然后我将MyClass.h包含在许多文件中,因此它包含多次并发生重定义错误。 怎么预防?

我正在使用#pragma而不是包含警卫,我想这很好。

MyClass.h:

// MyClass.h
#pragma once

#include <winsock2.h>

class MyClass
{

// methods
public:
 MyClass(unsigned short port);
 virtual ~MyClass(void);
};

编辑:我得到的错误很少

c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(91) : warning C4005: 'AF_IPX' : macro redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(460) : see previous definition of 'AF_IPX'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(124) : warning C4005: 'AF_MAX' : macro redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(479) : see previous definition of 'AF_MAX'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(163) : warning C4005: 'SO_DONTLINGER' : macro redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(402) : see previous definition of 'SO_DONTLINGER'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(206) : error C2011: 'sockaddr' : 'struct' type redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(485) : see declaration of 'sockaddr'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(384) : error C2143: syntax error : missing '}' before 'constant'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(384) : error C2143: syntax error : missing ';' before 'constant'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(384) : error C2059: syntax error : 'constant'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(437) : error C2143: syntax error : missing ';' before '}'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(437) : error C4430: missing type specifier - int assumed. Note: C++ does not support default-int
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(437) : error C4430: missing type specifier - int assumed. Note: C++ does not support default-int
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(518) : warning C4005: 'IN_CLASSA' : macro redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(287) : see previous definition of 'IN_CLASSA'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(524) : warning C4005: 'IN_CLASSB' : macro redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(293) : see previous definition of 'IN_CLASSB'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(530) : warning C4005: 'IN_CLASSC' : macro redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(299) : see previous definition of 'IN_CLASSC'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(541) : warning C4005: 'INADDR_ANY' : macro redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(304) : see previous definition of 'INADDR_ANY'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(543) : warning C4005: 'INADDR_BROADCAST' : macro redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(306) : see previous definition of 'INADDR_BROADCAST'
c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(577) : error C2011: 'sockaddr_in' : 'struct' type redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(312) : see declaration of 'sockaddr_in'
c:\program files\microsoft sdks\windows\v6.0a\include\winsock2.h(132) : error C2011: 'fd_set' : 'struct' type redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(68) : see declaration of 'fd_set'
c:\program files\microsoft sdks\windows\v6.0a\include\winsock2.h(167) : warning C4005: 'FD_SET' : macro redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(102) : see previous definition of 'FD_SET'
c:\program files\microsoft sdks\windows\v6.0a\include\winsock2.h(176) : error C2011: 'timeval' : 'struct' type redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(111) : see declaration of 'timeval'
c:\program files\microsoft sdks\windows\v6.0a\include\winsock2.h(232) : error C2011: 'hostent' : 'struct' type redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(167) : see declaration of 'hostent'
c:\program files\microsoft sdks\windows\v6.0a\include\winsock2.h(245) : error C2011: 'netent' : 'struct' type redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(180) : see declaration of 'netent'
c:\program files\microsoft sdks\windows\v6.0a\include\winsock2.h(252) : error C2011: 'servent' : 'struct' type redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(187) : see declaration of 'servent'
c:\program files\microsoft sdks\windows\v6.0a\include\winsock2.h(264) : error C2011: 'protoent' : 'struct' type redefinition
        c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(199) : see declaration of 'protoent'

包括当该问题引起的<windows.h><winsock2.h> 尝试安排<winsock2.h> _WINSOCKAPI_ <winsock2.h>之后包含<windows.h>的包含<winsock2.h>或首先定义_WINSOCKAPI_

#define _WINSOCKAPI_    // stops windows.h including winsock.h
#include <windows.h>
// ...
#include "MyClass.h"    // Which includes <winsock2.h>

另请参见

正如其他人所说,问题是在WinSock2.h之前包含windows.h 因为windows.h包含winsock.h 您不能同时使用WinSock2.hwinsock.h

解决方案:

  • windows.h之前包含WinSock2.h 对于预编译头文件,您应该在那里解决它。 在简单项目的情况下,很容易。 但是在大项目中(特别是在编写可移植代码时,没有预编译头文件),这可能非常困难,因为当包含WinSock2.h的头文件时, windows.h可能已经包含在其他一些头文件/实现文件中。

  • windows.h或项目范围之前定义WIN32_LEAN_AND_MEAN 但它会排除你可能需要的许多其他东西,你应该自己包括它。

  • windows.h或项目范围之前定义_WINSOCKAPI_ 但是当你包含WinSock2.h你会得到宏重新定义警告。

  • winsock.h足以满足您的项目时(在大多数情况下),使用windows.h而不是WinSock2.h 这可能会导致更长的编译时间,但可以解决任何错误/警告。

哦 - Windows的丑陋......包含的顺序在这里很重要。 你需要在windows.h之前包含winsock2.h。 由于windows.h可能包含在预编译头文件(stdafx.h)中,因此您需要包含winsock2.h:

#include <winsock2.h>
#include <windows.h>

通过使用“标题保护”:

#ifndef MYCLASS_H
#define MYCLASS_H

// This is unnecessary, see comments.
//#pragma once

// MyClass.h

#include <winsock2.h>

class MyClass
{

// methods
public:
    MyClass(unsigned short port);
    virtual ~MyClass(void);
};

#endif

当我试图拉出第三方软件包时,我遇到了这个问题,这个软件包显然包含了某些东西,其中有一些标题。 在项目级别定义_WINSOCKAPI_比通过他们的汤和修复有问题的包括更容易(更不用说更易于维护)。

在VS 2015中,以下内容将起作用:

#define _WINSOCKAPI_

虽然以下不会:

#define WIN32_LEAN_AND_MEAN

我不会只使用FILENAME_H而是

#ifndef FILENAME_H_AF06570D_B36E_4B82_8F97_C456AF4A38FD
#define FILENAME_H_AF06570D_B36E_4B82_8F97_C456AF4A38FD

//code stuff
#endif // FILENAME_H_AF06570D_B36E_4B82_8F97_C456AF4A38FD

我一直使用后缀guid。 几年前我遇到了一个非常糟糕的代码库,它有不同的头文件,文件名相同,包含guard。 有问题的文件定义了一个具有相同名称的类。 如果仅使用名称空间。 有些项目编成的一些没有。 使用独特的防护是区分标题及其内容的解决方案的一部分。

在Windows上使用Visual Studio使用guidgen.exe,在Linux上使用uuidgen -t。

我检查了递归包含,我发现了包含(递归)一些#include "windows.h"和“#include”Winsock.h“的头文件, and write a #include”Winsock2.h“ . in this files, i added #include“Winsock2.h”`作为第一个包括。

只是耐心一点,看看包含一个一个并建立这个顺序,首先#include "Winsock2.h"然后#include "windows.h"

我找到了这个链接windows.h和winsock2.h ,它有一个对我有用的替代方案:

#define _WINSOCKAPI_    // stops windows.h including winsock.h
#include <windows.h>
#include <winsock2.h>

我无法找到问题发生的地方但是通过添加#define我能够构建而无需弄清楚它。

我遇到了同样的问题,到目前为止我发现了这个问题:

从这个输出片段 -

c:\program files\microsoft sdks\windows\v6.0a\include\ws2def.h(91) : warning C4005: 'AF_IPX' : macro redefinition
c:\program files\microsoft sdks\windows\v6.0a\include\winsock.h(460) : see previous definition of 'AF_IPX'

- 似乎ws2def.h和winsock.h都已包含在您的解决方案中。

如果你查看文件ws2def.h,它会从以下注释开始 -

/*++

Copyright (c) Microsoft Corporation. All rights reserved.

Module Name:

    ws2def.h

Abstract:

    This file contains the core definitions for the Winsock2
    specification that can be used by both user-mode and 
    kernel mode modules.

    This file is included in WINSOCK2.H. User mode applications
    should include WINSOCK2.H rather than including this file
    directly. This file can not be included by a module that also
    includes WINSOCK.H.

Environment:

    user mode or kernel mode

--*/

注意最后一行 - “包含WINSOCK.H的模块不能包含此文件”

仍在尝试在不更改代码的情况下纠正问题。

如果这是有道理的,请告诉我。

你应该使用头部防护。

将这些行放在头文件的顶部

#ifndef PATH_FILENAME_H
#define PATH_FILENAME_H

在底部

#endif

#pragma once基于文件名的完整路径。 所以你可能有的是在不同的目录中有两个相同的MyClass.h或Winsock2.h副本。

#pragma once很好,即使在MS编译器上也是如此,许多其他编译器都不支持它。 正如许多其他人所提到的那样,使用包含警卫是可行的方法。 不要#pragma once使用#pragma once - 它会让你的生活更轻松。

#include guards是执行此操作的标准方法。 #pragma once不是,这意味着并非所有编译器都支持它。

在我的项目(我使用VS 2008 SP1)工作下一个解决方案:

头文件:

//myclass.h
#pragma once
#define _WINSOCKAPI_
#include <windows.h>

Cpp类:

//myclass.cpp
#include "Util.h"
#include "winsock2class.h"
#pragma comment(lib, "Ws2_32.lib")

其中#include“winsock2class.h”表示实现winsock2.h的类:

//winsock2class.h
#include <winsock2.h>
#include <windows.h>
#pragma comment(lib, "Ws2_32.lib")

我实际上遇到了一个问题,我必须将winsock2.h定义为第一个包含,它似乎还有来自其他包的其他问题。 希望这对遇到同样问题的人有帮助,不仅仅是windows.h,还包括所有问题。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM