C 中的 Windows 套接字编程

Posted

技术标签:

【中文标题】C 中的 Windows 套接字编程【英文标题】:Windows Socket Programming in C 【发布时间】:2011-10-09 20:15:11 【问题描述】:

我正在参加一个网络课程,教授正在向全班朗读这本书。不用说我不知道​​我在做什么。我们的学期项目是从我们的教科书中复制代码并建立一个客户端-服务器网络。从字面上复制书中的代码,无需修改。

这本书的代码有错误(缺少分号,多余的括号),但我设法至少编译了代码。但是,我遇到了一堆链接错误。

示例: 错误1错误LNK2019:函数_main C:\Users\Documents\Visual Studio 2010\Projects\Client_Server\Client_Server\Client_Server\Server.obj Client_Server中引用的未解析的外部符号impsendto@24

我查看了错误代码,我认为代码试图链接到头文件中不存在的定义。我很难修复 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 页上复制并运行代码。

【问题讨论】:

【参考方案1】:

您需要链接库 Ws2_32.lib 才能使用 winsock。在使用任何其他 winsock 函数之前,您还必须调用 WSAStartup(这不会导致您当前的错误,但会在您修复缺少的库问题后给您带来问题)。

【讨论】:

您可能也不需要同时包含 只需 #include 作为源中的第一个头文件并指定 ws2_32.lib在 Visual Studio 的链接设置中。 我添加了库文件,我可以在调试中构建,但它没有创建可执行文件。当我尝试在发布模式下调试时,我遇到了相同的链接错误。【参考方案2】:

首先,我将尝试帮助您使用您的最后一条评论:让我们假设您正在使用 Visual Studio(我认为最好的选择是启动 windows 编程的 winsock,因为 Microsoft 关心 Windows 基本库是最新的,它们是与有用的 msdn 支持兼容)。

如果您遇到如下错误: 1>asdf.obj : 错误 LNK2001: 无法解析的外部符号 _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;

如需更多帮助,请访问here

注意:由于问题很旧,我不确定它的作者是否会发现我的回答有帮助,我想帮助其他查看此问题的用户

【讨论】:

【参考方案3】:

下面是一个简单的socket程序(简单的http客户端),可以在Windows和Linux上运行。如果您使用的是“gcc on windows”,则需要使用以下命令进行编译:

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

【讨论】:

以上是关于C 中的 Windows 套接字编程的主要内容,如果未能解决你的问题,请参考以下文章

用于 UDP 客户端的 Windows C 套接字编程

在 Windows 上进行 C 套接字编程时,我需要替换哪些等效的标头、函数和数据类型?

套接字编程c ++ windows,看不到从服务器接收到的消息[关闭]

ioctl() 用于 C 中的套接字编程

c_cpp C语言中的套接字编程教程

c语言套接字编程中的listen()队列长度?