LNK2019:未解析的外部符号

时间:2011-08-29 16:06:19

标签: c++ winapi winsock

我已经看到很多其他类似的问题,但我无法在他们的帮助下解决这个问题。我知道这是一个链接问题,但从我所看到的,我已经把连接理顺了。

我正在写一个聊天服务器/客户端(在this article的帮助下)。

我已经定义了一个类来保存服务器函数,并且有一个处理所有包含的头文件。

这是头文件:

#include <windows.h>
#include <winsock.h>
#include <stdio.h>
#include <tchar.h>
#include <strsafe.h>
#include "resource1.h"


class ChatServer
{
    public: int InitServer(HINSTANCE hInst);

    public: void ReportError(int errorCode, const char *whichFunc);
};

这是实际的服务器“class”:

#include "server.h"
#define NETWORK_ERROR -1
#define NETWORK_OK     0
//Keeps stuff for the server    
int ChatServer::InitServer(HINSTANCE hInst)
    {
        WORD sockVersion;
        WSADATA wsaData;
        int nret;

        sockVersion = MAKEWORD(1,1); //Version 1.1

        //Init winsock
        WSAStartup(sockVersion, &wsaData);

        //Create listening socket
        SOCKET listeningSocket;

        //AFINET - Go over TCP
        //SOCK_STREAM - Stream oriented socket
        //IPPROTO_TCP - Use tcp rather than udp
        listeningSocket = socket( AF_INET, SOCK_STREAM, IPPROTO_TCP);

        if(listeningSocket == INVALID_SOCKET)
        {
            nret = WSAGetLastError(); //Get error detail
            ReportError(nret, "socket()");

            WSACleanup();

            return NETWORK_ERROR;
        }

        SOCKADDR_IN serverInfo;

        serverInfo.sin_family = AF_INET;
        serverInfo.sin_addr.s_addr = INADDR_ANY;
        serverInfo.sin_port = htons(1337); 

        //Bind the socket to local server address.
        nret = bind(listeningSocket, (LPSOCKADDR)&serverInfo, sizeof(struct sockaddr));

        if(nret == SOCKET_ERROR)
        {
            nret = WSAGetLastError();
            ReportError(nret, "bind()");
            WSACleanup();
            return NETWORK_ERROR;
        }

        //Make socket listen
        nret = listen(listeningSocket, 10); //Up to 10 connections at the same time.

        if(nret = SOCKET_ERROR)
        {
            nret = WSAGetLastError();
            ReportError(nret, "listen()");
            WSACleanup();
            return NETWORK_ERROR;
        }

        //Wait for client
        SOCKET theClient;
        theClient = accept(listeningSocket, NULL, NULL);

        if(theClient == INVALID_SOCKET)
        {
            nret = WSAGetLastError();
            ReportError(nret, "accept()");
            WSACleanup();
            return NETWORK_ERROR;
        }

        //Send and receive from the client, and finally,
        closesocket(theClient);
        closesocket(listeningSocket);

        //shutdown
        WSACleanup();
        return NETWORK_OK;
    }



void ChatServer::ReportError(int errorCode, const char *whichFunc)

    {

       char errorMsg[92];                   // Declare a buffer to hold
                                            // the generated error message
       ZeroMemory(errorMsg, 92);            // Automatically NULL-terminate the string
       // The following line copies the phrase, whichFunc string, and integer errorCode into the buffer
       sprintf(errorMsg, "Call to %s returned error %d!", (char *)whichFunc, errorCode);



       MessageBox(NULL, errorMsg, "socketIndication", MB_OK);

    }

最后,带有程序入口方法的main.cpp文件调用“ChatServer :: InitServer(g_hInst)”。它非常大,所以我省略了它,但是如果它需要我也会发布它。

我得到的错误消息与下面的错误消息类似,但它们都表示与winsockets API相关的api函数存在问题:

Error   3   error LNK2019: unresolved external symbol _closesocket@4 referenced in function "public: int __thiscall ChatServer::InitServer(struct HINSTANCE__ *)" (?InitServer@ChatServer@@QAEHPAUHINSTANCE__@@@Z)  

正如我之前所说,我认为这个问题与编译器误解如何处理应该链接到winsock.h的“closesocket”等函数有关。

感谢您的任何建议,并感谢阅读所有这些胡言乱语:)

3 个答案:

答案 0 :(得分:9)

这些链接器错误始终相同:您使用的是链接器无法找到的符号。您需要告诉链接器链接包含这些符号的库。

  

正如我之前所说,我认为这个问题与此有关   编译器误解了如何处理函数   应该链接到winsock.h的“closesocket”。

不,closesocket没有链接到winsock.h。 winsock.h是头文件,而不是库。它可能包含closesocket的声明,这对编译器来说没问题,但链接器确实需要知道该函数的代码在哪里,以便它可以将程序链接到它。

你可能应该使用winsock2.h而不是winsock.h。 Windows套接字API 2.0的历史可以追溯到1994年,版本1已经过时了。

您可以在this function's documentation的底部看到它所在的库是ws2_32.lib。

答案 1 :(得分:2)

您的课程可以这样定义:

class ChatServer
{
public:
    int InitServer(HINSTANCE hInst);
    void ReportError(int errorCode, const char *whichFunc);
};

但是,错误是由于不包括库输入引起的。在项目输入行上添加对winsock库的依赖也考虑使用winsock2.h。

供参考:http://msdn.microsoft.com/en-us/library/ms737629(v=vs.85).aspx

答案 2 :(得分:2)

仅包含.h文件不会导致实现函数的实际代码被链接..h文件告诉编译器函数存在于某处,以及它们的签名是什么。您仍然必须使实际实现该功能的库可用于链接器。