C语言中的Windows套接字编程

6
我正在参加一门网络课程,教授直接朗读教科书。无需多言,我不知道自己在做什么。我们的学期项目是从教材中复制代码并创建一个客户端-服务器网络。只需从书本上复制代码而不进行任何修改。
教材中的代码存在错误(缺少分号,额外的括号),但我至少成功编译了代码。然而,我遇到了一堆链接错误。
例如: Error 1 error LNK2019: unresolved external symbol impsendto@24 referenced in function _main C:\Users\Documents\Visual Studio 2010\Projects\Client_Server\Client_Server\Client_Server\Server.obj Client_Server 我查阅了错误代码,认为代码试图链接到头文件中不存在的定义。相比语法错误,我很难解决LNK错误。但像我之前说过的一样,我不知道该如何解决这个问题。我正在发送服务器端的代码,客户端端也遇到了同样的错误。
include <stdio.h>
include <string.h>
include <WinSock2.h>
include <WinSock.h>
include <stdint.h>
include <time.h>

int main(void) {

int s;      
int len;
char  buffer[256];  
struct sockaddr_in servAddr; 
struct sockaddr_in clntAddr; 

int clntAddrLen; //length of client socket addre

//Build local (server) socket add

memset(&servAddr, 0, sizeof(servAddr));
servAddr.sin_family = AF_INET;
servAddr.sin_port = htons(21);
servAddr.sin_addr.s_addr = htonl(INADDR_ANY);

   //create socket
if((s=socket(PF_INET, SOCK_DGRAM, 0) <0 ))
{
   perror("Error: Socket Failed!");
    exit(1);
}

//bind socket to local address and port
if((bind(s,(struct sockaddr*)&servAddr, sizeof(servAddr))<0))
{
    perror("Error:bind failed!");
    exit(1);
}

for(;;)
{
len = recvfrom(s,buffer, sizeof(buffer),0,(struct sockaddr*)&clntAddr, &clntAddrLen);

    //send string
    sendto(s, buffer, len, 0, (struct sockaddr*)&clntAddr, sizeof(clntAddr));
}

}

希望能够得到一些有用的提示、信息链接或建议。我尝试阅读教材,但是完全不知所措。此外,这是本学期唯一与代码相关的作业。其他所有的作业都是使用数据包嗅探器收集数据包。老师直接让我们复制并运行第 X 页上的代码。

3个回答

7

如果要使用winsock,您需要链接Ws2_32.lib库。在使用任何其他winsock函数之前,您还必须调用WSAStartup(这不会导致当前错误,但在解决缺少库问题后会引起问题)。


你可能也不需要同时包含 <winsock2.h> 和 <winsock.h>。只需在源文件中将 #include <winsock2.h> 作为第一个头文件,并在 Visual Studio 的链接设置中指定 ws2_32.lib 即可。 - selbie
我添加了库文件,可以在调试模式下构建,但没有创建可执行文件。当我尝试在发布模式下进行调试时,遇到了相同的链接错误。 - networkingNoob

5

首先,我将尝试使用您最近的评论提供帮助:假设您正在使用Visual Studio(我认为这是开始Windows编程的最佳选择,因为Microsoft关注Windows基本库的最新版本,并且它们与有用的MSDN支持兼容)。

如果您遇到以下错误: 1>asdf.obj : error LNK2001: unresolved external symbol _imp_WSAStartup@8 表示ws2_32.lib未正确链接。要解决此问题,请在解决方案资源管理器中右键单击项目,选择 链接器 -> 输入 ,并将ws2_32.lib添加到其他依赖项中。该库是Windows SDK的一部分(我猜它与大多数Visual Studio版本一起安装),请确保文件存在于计算机上。

现在,如何以现代风格创建正确的项目,而无需遵循古老的教程:

您需要添加的库是Winsock2.h。 Winsock.h是旧版本(已废弃),没有必要在新应用程序中使用它。要开始使用套接字,您需要调用函数WSAStartup,在开始时必须初始化struct WSADATA。 基本代码如下:

#include <Winsock2.h>
int main()
{
WSADATA mywsadata; //your wsadata struct, it will be filled by WSAStartup
WSAStartup(0x0202,&mywsadata); //0x0202 refers to version of sockets we want to use.
//here goes your code with socket related things
return 0;
}

如果需要更多帮助,您可以访问这里

注:由于问题很旧,我不确定作者是否会发现我的答案有用,所以我想帮助其他看到这个问题的用户。


5
以下是一个简单的套接字程序(简单的HTTP客户端),可以在Windows和Linux上运行。如果您正在使用“Windows上的gcc”,则需要使用以下命令进行编译:
gcc prog_name.c -lws2_32

代码:

#include <stdio.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

#if defined(_WIN32) || defined(_WIN64)
    #include <winsock2.h>
#else
    #include <sys/socket.h>
    #include <arpa/inet.h>
#endif

#define MSG_SIZE 1024
#define REPLY_SIZE 65536

int main(int argc, char *argv[])
{
    int s = -1;
    struct sockaddr_in server;
    char message[MSG_SIZE] = {0}, server_reply[REPLY_SIZE] = {0};
    int recv_size = 0;

#if defined(_WIN32) || defined(_WIN64)    
    WSADATA wsa;
    if (WSAStartup(MAKEWORD(2,2),&wsa) != 0) {
        printf("\nError: Windows socket subsytsem could not be initialized. Error Code: %d. Exiting..\n", WSAGetLastError());
        exit(1);
    }
#endif
    
    //Create a socket
    if((s = socket(AF_INET, SOCK_STREAM, 0)) < 0)    {
        printf("Error: Could not create socket: %s. Exiting..\n", strerror(errno));
        exit(1);
    }

    // Fill in server's address
    memset(&server, 0, sizeof(server));
    server.sin_addr.s_addr = inet_addr("172.217.160.238"); // google.com
    server.sin_family = AF_INET;
    server.sin_port = htons(80);

    // Connect to server
    if (connect(s, (struct sockaddr *)(&server), sizeof(server)) < 0) {
        printf("Error: Could not connect to server: %s. Exiting..\n", strerror(errno));
        exit(1);
    }
    
    // Send HTTP request
    strcpy(message, "GET / HTTP/1.1\r\n\r\n");
    if(send(s, message, strlen(message), 0) < 0) {
        printf("Error: Could not send http request to server: %s. Exiting..\n", strerror(errno));
        exit(1);
    }
    
    // Receive a reply from the server
    printf("\nWaiting for server reply..\n");
    if((recv_size = recv(s, server_reply, REPLY_SIZE, 0)) < 0) {
        printf("Error: Something wrong happened while getting reply from server: %s. Exiting..\n", strerror(errno));
        exit(1);
    }

    server_reply[REPLY_SIZE - 1] = 0;

    printf("\nServer Reply:\n\n");
    printf("%s\n", server_reply);

    // Close the socket
#if defined(_WIN32) || defined(_WIN64)  
    closesocket(s);
    WSACleanup();
#else
    close(s);
#endif

    exit(0);
} // end of main

太好了!我遇到了同样的问题。由于某种原因,我在连接过程中遇到了几个问题(我猜是在链接过程中),而这个方法对我有效。 - undefined

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接