首页 > 系统相关 >突破编程_C++_网络编程(Windows 套接字(API 接口(2)))

突破编程_C++_网络编程(Windows 套接字(API 接口(2)))

时间:2024-04-09 18:58:05浏览次数:24  
标签:sockaddr addr Windows 编程 C++ int 接字 servaddr 函数

1 TCP 连接管理

1.1 connect 函数

connect 函数用于发起一个 TCP 连接请求到远程服务器。这个函数通常用于客户端套接字,以建立与服务器的连接。

(1)函数原型

int connect(SOCKET s, const struct sockaddr *name, int namelen);

(2)参数说明

  • s:一个已创建但未连接的套接字描述符。
  • name:指向包含目标服务器地址信息的sockaddr结构体的指针。
  • namelen:name参数指向的地址结构的长度。

(3)返回值

  • 如果成功,返回0。
  • 如果失败,返回SOCKET_ERROR,并可以通过WSAGetLastError函数获取错误代码。

(4)示例代码

SOCKET sockfd;  
struct sockaddr_in serv_addr;  
  
// 创建套接字...  
sockfd = socket(AF_INET, SOCK_STREAM, 0);  
if (sockfd < 0) {  
    // 错误处理...  
}  
  
// 设置服务器地址信息...  
memset(&serv_addr, 0, sizeof(serv_addr));  
serv_addr.sin_family = AF_INET;  
serv_addr.sin_addr.s_addr = inet_addr("192.168.1.100"); // 服务器IP地址  
serv_addr.sin_port = htons(12345); // 服务器端口号  
  
// 发起连接请求...  
if (connect(sockfd, (struct sockaddr*)&serv_addr, sizeof(serv_addr)) < 0) {  
    // 连接失败处理...  
} else {  
    // 连接成功处理...  
}

1.2 listen 函数

listen 函数用于使套接字进入监听状态,等待客户端连接。这个函数通常用于服务器端套接字。

(1)函数原型

int listen(SOCKET s, int backlog);

(2)参数说明

  • s:一个已绑定但未连接的套接字描述符。
  • backlog:指定等待连接队列的最大长度。

(3)返回值

如果成功,返回 0。
如果失败,返回 SOCKET_ERROR,并可以通过 WSAGetLastError 函数获取错误代码。

(4)示例代码

SOCKET sockfd;  
  
// 创建并绑定套接字...  
sockfd = socket(AF_INET, SOCK_STREAM, 0);  
if (sockfd < 0) {  
    // 错误处理...  
}  
  
// 绑定地址和端口...  
struct sockaddr_in serv_addr;  
// 设置serv_addr...  
if (bind(sockfd, (struct sockaddr*)&serv_addr, sizeof(serv_addr)) < 0) {  
    // 绑定失败处理...  
}  
  
// 进入监听状态...  
if (listen(sockfd, 5) < 0) {  
    // 监听失败处理...  
} else {  
    // 等待客户端连接...  
}

1.3 accept 函数

accept 函数用于接受一个来自客户端的连接请求,并返回一个新的套接字用于与该客户端通信。这个函数通常用于服务器端套接字,在 listen 函数之后调用。

(1)函数原型

SOCKET accept(SOCKET s, struct sockaddr *addr, int *addrlen);

(2)参数说明

  • s:一个处于监听状态的套接字描述符。
  • addr(可选):指向一个sockaddr结构体的指针,用于接收客户端的地址信息。
  • addrlen(可选):指向一个整数的指针,用于接收addr参数指向的地址结构的实际长度。

(3)返回值

  • 如果成功,返回一个新的套接字描述符,用于与客户端通信。
  • 如果失败,返回INVALID_SOCKET,并可以通过WSAGetLastError函数获取错误代码。

(4)示例代码

SOCKET listenfd, connfd;  
struct sockaddr_in cli_addr;  
socklen_t clilen;  
  
// 假设listenfd是一个已绑定并处于监听状态的套接字...  
  
// 接受客户端连接请求...  
clilen = sizeof(cli_addr);  
connfd = accept(listenfd, (struct sockaddr*)&cli_addr, &clilen);  
if (connfd < 0) {  
    // 接受连接失败处理...  
} else {  
    // 与客户端通信...  
}

在这个例子中,accept 函数会阻塞调用线程,直到有客户端发起连接请求。当有一个客户端成功连接时,accept 函数会返回一个新的套接字描述符 connfd,这个新的套接字描述符用于与这个特定的客户端进行通信。同时,如果提供了 addr 和 addrlen 参数,accept 函数会填充这些参数以返回客户端的地址信息。

在实际应用中,服务器通常会在一个单独的线程中调用 accept 函数,以便在等待新的连接请求时能够继续处理现有的连接。这可以通过创建线程池或使用异步I/O技术来实现。

此外,值得注意的是,accept 函数返回的新套接字描述符 connfd 与原始的监听套接字描述符listenfd 是独立的。你可以使用 connfd 来读取和写入数据,而 listenfd 则继续用于接受新的连接请求。

最后,为了确保资源得到正确释放,当不再需要套接字时,应使用 closesocket 函数来关闭它们。这包括原始的监听套接字以及通过 accept 函数返回的每个新套接字。

2 数据发送与接收

2.1 send 函数

send 函数用于向已连接的套接字发送数据。在 TCP 编程中,需要先调用 connect 函数来建立连接,然后使用 send 函数来发送数据。

(1)函数原型

int send(SOCKET s, const char *buf, int len, int flags);

(2)参数说明

  • s:已连接的套接字描述符。
  • buf:指向包含要发送数据的缓冲区的指针。
  • len:要发送数据的字节数。
  • flags:控制发送操作的标志,通常设置为0。

(3)返回值

  • 如果成功,返回发送的字节数。这不一定等于请求发送的字节数,特别是当套接字设置为非阻塞模式时。
  • 如果失败,返回SOCKET_ERROR,并可以通过WSAGetLastError函数获取错误代码。

(4)示例代码

#include <winsock2.h>  
#include <stdio.h>  
#include <string.h>  
  
#pragma comment(lib, "ws2_32.lib")  
  
int main() {  
    WSADATA wsaData;  
    SOCKET sockfd;  
    struct sockaddr_in servaddr;  
    char sendbuf[] = "Hello, TCP server!";  
    int sendlen = strlen(sendbuf);  
  
    // 初始化Winsock库  
    if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) {  
        printf("WSAStartup failed.\n");  
        return 1;  
    }  
  
    // 创建套接字  
    if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {  
        perror("socket creation failed");  
        return 1;  
    }  
  
    // 设置服务器地址信息  
    memset(&servaddr, 0, sizeof(servaddr));  
    servaddr.sin_family = AF_INET;  
    servaddr.sin_port = htons(12345); // 假设服务器端口是12345  
    servaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); // 假设服务器IP是127.0.0.1  
  
    // 连接到服务器  
    if (connect(sockfd, (struct sockaddr*)&servaddr, sizeof(servaddr)) < 0) {  
        perror("connection failed");  
        return 1;  
    }  
  
    // 发送数据  
    if (send(sockfd, sendbuf, sendlen, 0) != sendlen) {  
        perror("send failed");  
        return 1;  
    }  
    printf("Data sent successfully.\n");  
  
    // 关闭套接字  
    closesocket(sockfd);  
    WSACleanup();  
  
    return 0;  
}

2.2 sendto 函数

sendto 函数用于通过 UDP 套接字向指定地址发送数据报。由于 UDP 是无连接的,因此需要提供目标地址信息。

(1)函数原型

int sendto(SOCKET s, const char *buf, int len, int flags, const struct sockaddr *to, int tolen);

(2)参数说明

  • s:UDP套接字描述符。
  • buf:指向包含要发送数据的缓冲区的指针。
  • len:要发送数据的字节数。
  • flags:控制发送操作的标志,通常设置为0。
  • to:指向包含目标地址信息的sockaddr结构体的指针。
  • tolen:to参数指向的地址结构的长度。

(3)返回值

  • 如果成功,返回发送的字节数。
  • 如果失败,返回SOCKET_ERROR,并可以通过WSAGetLastError函数获取错误代码。

(4)示例代码

#include <winsock2.h>  
#include <stdio.h>  
#include <string.h>  
  
#pragma comment(lib, "ws2_32.lib")  
  
int main() {  
    WSADATA wsaData;  
    SOCKET sockfd;  
    struct sockaddr_in servaddr;  
    char sendbuf[] = "Hello, UDP server!";  
    int sendlen = strlen(sendbuf);  
  
    // 初始化Winsock库  
    if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) {  
        printf("WSAStartup failed.\n");  
        return 1;  
    }  
  
    // 创建UDP套接字  
    if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {  
        perror("socket creation failed");  
        return 1;  
    }  
  
    // 设置服务器地址信息  
    memset(&servaddr, 0, sizeof(servaddr));  
    servaddr.sin_family = AF_INET;  
    servaddr.sin_port = htons(12345); // 假设服务器端口是12345  
    servaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); // 假设服务器IP是127.0.0.1  
  
    // 发送数据到服务器  
    if (sendto(sockfd, sendbuf, sendlen, 0, (struct sockaddr*)&servaddr, sizeof(servaddr)) != sendlen) {  
        perror("sendto failed");  
        return 1;  
    }  
    printf("Data sent successfully.\n");  
  
    // 关闭套接字  
    closesocket(sockfd);  
    WSACleanup();  
  
    return 0;  
}

2.3 recv 函数

recv 函数用于从已连接的套接字接收数据。在 TCP 编程中,需要先调用 accept 函数来接受连接,然后使用 recv 函数来接收数据。

(1)函数原型

recv函数用于从已连接的套接字接收数据。在TCP编程中,你通常会先调用accept函数来接受连接,然后使用recv函数来接收数据。

(2)参数说明

  • s:已连接的套接字描述符。
  • buf:指向用于存储接收数据的缓冲区的指针。
  • len:缓冲区的大小,即最多可以接收的字节数。
  • flags:控制接收操作的标志,通常设置为0。

(3)返回值

  • 如果成功,返回接收到的字节数。这可以是 0(表示连接已正常关闭),也可以是请求的字节数或更少的字节数。
  • 如果失败,返回 SOCKET_ERROR,并可以通过 WSAGetLastError 函数获取错误代码。

(4)示例代码

#include <winsock2.h>  
#include <stdio.h>  
#include <string.h>  

#pragma comment(lib, "ws2_32.lib")  

#define BUFSIZE 1024  

int main() {
	WSADATA wsaData;
	SOCKET listenfd, connfd;
	struct sockaddr_in servaddr, cliaddr;
	char buffer[BUFSIZE];
	int n;
	// 初始化Winsock库  
	if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) {
		printf("WSAStartup failed.\n");
		return 1;
	}

	// 创建套接字  
	if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
		perror("socket creation failed");
		return 1;
	}

	// 设置服务器地址信息  
	memset(&servaddr, 0, sizeof(servaddr));
	servaddr.sin_family = AF_INET;
	servaddr.sin_addr.s_addr = INADDR_ANY;
	servaddr.sin_port = htons(12345); // 假设服务器端口是12345  

	// 绑定套接字到地址  
	if (bind(listenfd, (struct sockaddr*)&servaddr, sizeof(servaddr)) < 0) {
		perror("bind failed");
		return 1;
	}

	// 开始监听连接  
	if (listen(listenfd, 5) < 0) {
		perror("listen failed");
		return 1;
	}

	// 接受客户端连接  
	int clilen = sizeof(cliaddr);
	if ((connfd = accept(listenfd, (struct sockaddr*)&cliaddr, &clilen)) < 0) {
		perror("accept failed");
		return 1;
	}

	// 接收数据  
	n = recv(connfd, buffer, BUFSIZE - 1, 0);
	if (n < 0) {
		perror("recv failed");
		return 1;
	}
	buffer[n] = '\0';
	printf("Received: %s\n", buffer);

	// 关闭套接字  
	closesocket(connfd);
	closesocket(listenfd);
	WSACleanup();

	return 0;
}

2.4 recvfrom 函数

recvfrom 函数用于从 UDP 套接字接收数据报,并获取发送方的地址信息。由于 UDP 是无连接的,因此每次接收数据时都需要获取发送方的地址。

(1)函数原型

int recvfrom(SOCKET s, char *buf, int len, int flags, struct sockaddr *from, int *fromlen);

(2)参数说明

  • s:UDP套接字描述符。
  • buf:指向用于存储接收数据的缓冲区的指针。
  • len:缓冲区的大小,即最多可以接收的字节数。
  • flags:控制接收操作的标志,通常设置为0。
  • from(可选):指向一个sockaddr结构体的指针,用于接收发送方的地址信息。
  • fromlen(可选):指向一个整数的指针,用于接收from参数指向的地址结构的实际长度。

(3)返回值

如果成功,返回接收到的字节数。
如果失败,返回 SOCKET_ERROR,并可以通过 WSAGetLastError 函数获取错误代码。

(4)示例代码

#include <winsock2.h>  
#include <stdio.h>  
#include <string.h>  

#pragma comment(lib, "ws2_32.lib")  

#define BUFSIZE 1024  

int main() {
	WSADATA wsaData;
	SOCKET sockfd;
	struct sockaddr_in servaddr, cliaddr;
	char buffer[BUFSIZE];
	int n;
	int clilen;

	// 初始化Winsock库  
	if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) {
		printf("WSAStartup failed.\n");
		return 1;
	}

	// 创建UDP套接字  
	if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
		perror("socket creation failed");
		return 1;
	}

	// 设置服务器地址信息  
	memset(&servaddr, 0, sizeof(servaddr));
	servaddr.sin_family = AF_INET;
	servaddr.sin_addr.s_addr = INADDR_ANY;
	servaddr.sin_port = htons(12345); // 假设服务器端口是12345  

	// 绑定套接字到地址  
	if (bind(sockfd, (struct sockaddr*)&servaddr, sizeof(servaddr)) < 0) {
		perror("bind failed");
		return 1;
	}

	// 接收数据  
	clilen = sizeof(cliaddr);
	n = recvfrom(sockfd, buffer, BUFSIZE - 1, 0, (struct sockaddr*)&cliaddr, &clilen);
	if (n < 0) {
		perror("recvfrom failed");
		return 1;
	}
	buffer[n] = '\0';
	printf("Received from %s:%d: %s\n", inet_ntoa(cliaddr.sin_addr), ntohs(cliaddr.sin_port), buffer);

	// 关闭套接字  
	closesocket(sockfd);
	WSACleanup();

	return 0;
}

标签:sockaddr,addr,Windows,编程,C++,int,接字,servaddr,函数
From: https://blog.csdn.net/h8062651/article/details/137542398

相关文章

  • 【每周例题】力扣 C++ 移除元素
    移除元素题目移除元素 思路分析1.涉及到容器,那么就很直接的想法,遍历容器,找出与val相同的数,移除,然后利用函数输出长度与移除后的数组2.移除部分我们使用指针去处理,用指针遍历数组,符合移除条件的利用erase函数移除注:这里使用到了一个万能头文件,参加蓝桥杯的同学可以试试运用......
  • 在Windows环境下使用NodeJS的fast-glob不正确执行的问题
    fast-glob是NodeJS中的一个非常高效的文件遍历工具,通过它在文件系统中方便的指定和筛选文件,它采用UnixBashshell使用的规则返回与一组定义的指定模式匹配的路径名,并进行了一些简化,同时以任意顺序返回结果。它支持同步、Promise和StreamAPI。在Windows环境下使用NodeJS中的......
  • 【每周例题】蓝桥杯 C++ 多数
    多数元素题目多数元素思路分析一.第一个想法,暴力遍历,然后会发现容易超时,那么更进一步想:哈希表使用哈希表存储每个数出现的次数,即使用哈希映射(HashMap)来存储每个元素以及出现的次数。对于哈希映射中的每个键值对,键表示一个元素,值表示该元素出现的次数加入后,遍历所有键值对,......
  • C++程序分享--常见编程面试题:判断字符串是否为回文串
    关注我,持续分享逻辑思维&管理思维;可提供大厂面试辅导、及定制化求职/在职/管理/架构辅导;有意找工作的同学,请参考博主的原创:《面试官心得--面试前应该如何准备》,《面试官心得--面试时如何进行自我介绍》, 《做好面试准备,迎接2024金三银四》。【图解《程序员面试常见的十大算法......
  • Windows系统下汇编环境的搭建
    Windows系统下汇编环境的搭建最近在学习assembly64时,需要对程序进行编写->生成汇编代码->调试->执行.本文聚焦于如果在Windows环境下,尽可能精简并且完整的构建一个汇编环境.基于Windows11,WSLUbuntu22.04,vscode,其他的系统/WSL发行版本.您可以以本文作为简......
  • Windows系统下汇编环境的搭建
    Windows系统下汇编环境的搭建最近在学习assembly64时,需要对程序进行编写->生成汇编代码->调试->执行.本文聚焦于如果在Windows环境下,尽可能精简并且完整的构建一个汇编环境.基于Windows11,WSLUbuntu22.04,vscode,其他的系统/WSL发行版本.您可以以本文作为简......
  • 在Windows电脑上上传iOS应用至App Store
     引言......
  • C/C++与Python:各自的优势与前景展望
    在讨论C/C++和Python这两种编程语言的前景时,我们必须认识到每种语言都有其独特的定位和应用场景,并不存在绝对意义上的“谁更有前景”。它们分别在不同的领域发挥着重要作用,而且在未来的技术发展过程中,二者都将继续保持其不可替代的地位。C/C++:底层与性能优势C和C++是两种历史......
  • C++继承之protected继承
    1概述  通过继承机制,可以利用已有的对象类型来定义新对象类型。所定义的新对象类型不仅仅拥有新定义的成员,而且还同时拥有旧的成员。我们称已存在的用来派生新类的类为基类,又称为父类。由已存在的类派生出的新类称为派生类,又称为子类。。2对象访问作用域作用域:publi......
  • Windows程序设计作业1-周黑鸭工厂
     一.作业内容    使用C#编码(涉及类、接口、委托等关键知识点),实现对周黑鸭工厂的产品生产统一管理,主要产品包括鸭脖和鸭翅。武汉工厂能生生产鸭脖和鸭翅,南京工厂只能生产鸭翅,长沙工厂只能生产鸭脖。    具体要求如下:定义接口IProductionFactory,包含生产......