8

下午好。我已经开始学习 c++,并且我正在编译我的项目。如果你发现一些错误的代码,如果你告诉我,我会很高兴。

我有以下定义:

实用程序.h

#include "stdafx.h"
#include <string>
using namespace std;

class Utils
{
public:
static string GetStringFromInt (int number);
};

实用程序.cpp

#include "Utils.h"
#include <sstream>
#include <string>
using namespace std;

 string Utils::GetStringFromInt (int number)
{

    stringstream ss;
    ss << number;
    return ss.str();
}

平.h

#include "stdafx.h"
#include <string>

using namespace std;

class Ping
{
public:
    static int PingIt(int argc, char* argv[],string &mstime,string &ttl);   
};

ping.cpp

#include "Ping.h"
#include <string>
#include "icmpdefs.h"
#include <string>
#include <iostream>
#include <sstream>
#include <WinSock.h>
#include <Windows.h>
#include "Utils.h"
#pragma comment(lib,"wsock32.lib")
using namespace std;

int Ping::PingIt(int argc, char* argv[],string &mstime,string &ttl)
{


    // Check for correct command-line args
    if (argc < 2) {
        cerr << "usage: ping <host>" << endl;
        return 1;
    }

    // Load the ICMP.DLL
    HINSTANCE hIcmp = LoadLibrary("ICMP.DLL");
    if (hIcmp == 0) {
        cerr << "Unable to locate ICMP.DLL!" << endl;
        return 2;
    }

    // Look up an IP address for the given host name
    struct hostent* phe;
    if ((phe = gethostbyname(argv[1])) == 0) {
        cerr << "Could not find IP address for " << argv[1] << endl;
        return 3;
    }

    // Get handles to the functions inside ICMP.DLL that we'll need
    typedef HANDLE (WINAPI* pfnHV)(VOID);
    typedef BOOL (WINAPI* pfnBH)(HANDLE);
    typedef DWORD (WINAPI* pfnDHDPWPipPDD)(HANDLE, DWORD, LPVOID, WORD,
            PIP_OPTION_INFORMATION, LPVOID, DWORD, DWORD); // evil, no?
    pfnHV pIcmpCreateFile;
    pfnBH pIcmpCloseHandle;
    pfnDHDPWPipPDD pIcmpSendEcho;
    pIcmpCreateFile = (pfnHV)GetProcAddress(hIcmp,
            "IcmpCreateFile");
    pIcmpCloseHandle = (pfnBH)GetProcAddress(hIcmp,
            "IcmpCloseHandle");
    pIcmpSendEcho = (pfnDHDPWPipPDD)GetProcAddress(hIcmp,
            "IcmpSendEcho");
    if ((pIcmpCreateFile == 0) || (pIcmpCloseHandle == 0) || 
            (pIcmpSendEcho == 0)) {
        cerr << "Failed to get proc addr for function." << endl;
        return 4;
    }

    // Open the ping service
    HANDLE hIP = pIcmpCreateFile();
    if (hIP == INVALID_HANDLE_VALUE) {
        cerr << "Unable to open ping service." << endl;
        return 5;
    }

    // Build ping packet
    char acPingBuffer[64];
    memset(acPingBuffer, '\xAA', sizeof(acPingBuffer));
    PIP_ECHO_REPLY pIpe = (PIP_ECHO_REPLY)GlobalAlloc(
            GMEM_FIXED | GMEM_ZEROINIT,
            sizeof(IP_ECHO_REPLY) + sizeof(acPingBuffer));
    if (pIpe == 0) {
        cerr << "Failed to allocate global ping packet buffer." << endl;
        return 6;
    }
    pIpe->Data = acPingBuffer;
    pIpe->DataSize = sizeof(acPingBuffer);      

    // Send the ping packet
    DWORD dwStatus = pIcmpSendEcho(hIP, *((DWORD*)phe->h_addr_list[0]), 
            acPingBuffer, sizeof(acPingBuffer), NULL, pIpe, 
            sizeof(IP_ECHO_REPLY) + sizeof(acPingBuffer), 5000);
    if (dwStatus != 0) {
        cout << "Addr: " <<
                int(LOBYTE(LOWORD(pIpe->Address))) << "." <<
                int(HIBYTE(LOWORD(pIpe->Address))) << "." <<
                int(LOBYTE(HIWORD(pIpe->Address))) << "." <<
                int(HIBYTE(HIWORD(pIpe->Address))) << ", " <<
                "RTT: " << int(pIpe->RoundTripTime) << "ms, " <<
                "TTL: " << int(pIpe->Options.Ttl) << endl;

        mstime = Utils::GetStringFromInt((pIpe->RoundTripTime));
        ttl = Utils::GetStringFromInt(int(pIpe->Options.Ttl));
    }
    else {
        cerr << "Error obtaining info from ping packet." << endl;
    }

    // Shut down...
    GlobalFree(pIpe);
    FreeLibrary(hIcmp);
    return dwStatus;
}

当我编译项目时,我得到:

错误 1 ​​错误 C2653:“Ping”:不是类或命名空间名称 c:\users\claderasm\documents\visual studio 2010\projects\landetestconsole\landecplusconsole\ping.cpp 14 1 LandeCplusConsole

我读过有时会抛出此错误,因为您没有在第一个#include 中包含“stdafx.h”,但我已经更改了它。

如果你能告诉我更多的话,我会很高兴

4

1 回答 1

9

我无法使用您提供的代码重现您的错误,但我在 VS2008 上进行了尝试,它提出的一些警告让我认为这很可能是由于您的预编译头文件未包含在源代码中。我可以看到还有一些其他问题肯定会在以后给您带来问题:

  • 不要在 .h 中包含预编译的标头。(好吧,除非绝对必要,否则您甚至应该避免在 .h 中包含任何内容)。预编译的标头(至少在视觉上做事的方式)意味着首先包含在每个 cpp 文件中(不是 .h)。如果您不这样做,它将针对您拥有的每个包含发出警告:

    警告 C4627 : '#include "Ping.h"': 在查找预编译头文件时跳过使用 <- 你去你的 Point 类不再定义!

    最后是一个错误致命错误 C1010:在查找预编译头时文件意外结束。您是否忘记将 '#include "stdafx.h"' 添加到您的源代码中?

    这实际上是我在 VS2008 上编译代码时收到的消息,也许在 VS2010 上,除了这些之外,您还有关于 Point 未定义的错误。当整理出预编译的头文件问题时,它编译得很好(见下一点)

  • 使用预编译头文件并不意味着用于构建它的 .h 将自动包含在所有源代码中。为此,您必须更改一些项目设置:右键单击您的项目 -> Properties,在左侧面板中展开Configuration Properties -> C/C++ -> Advanced。在右侧的列表中,您应该看到Force Includes。在此处键入 stdafx.h,然后瞧,您不必手动将其放入添加到项目中的每个新 .cpp 中。请注意,您必须对所有配置执行此操作(顶部的组合框写有“配置:活动(调试)”

抱歉,还是VS2008的屏幕,希望VS2010也一样 在此处输入图像描述

  • 保护你的标题。当同一 .h 的多个包含发生时(并且会发生),您应该在标题上加注以避免您的类的多个定义。您可以通过两种方式进行操作:define 方法和 pragma once 方法。pragma once 不是标准的,但在 Visual 上编译速度更快,因此您最终可以混合使用这两种方式。

myheader.h 使用编译指示一次:

#pragma once
class MyClass
{
    //<some definitions>
};

myheader.h 使用定义:

#ifndef __MYHEADER_H
#define __MYHEADER_H
class MyClass
{
    //<some definitions>
};
#endif

myheader.h 同时使用:

#pragma once
#ifndef __MYHEADER_H
#define __MYHEADER_H
class MyClass
{
    //<some definitions>
};
#endif
  • 已经说过了,但要避免在标题中使用“使用”,因为它会传播。我自己避免在任何地方使用“使用”。
于 2012-11-08T07:00:56.750 回答