close and shutdown?

背景:我们要讲述的是网络编程中常用的两个API:

       #include <unistd.h>

       int close(int fd);

       #include <sys/socket.h>

       int shutdown(int sockfd, int how);

以及TCP的半连接,半打开。


shutdown函数的行为依赖第二个参数区分:

SHUT_RD :reception(接受)的disallow(拒绝),也就是第一个参数的文件描述符的接受数据被阻止。
SHUT_WR :transmissions(传播,发送)的disallow,也就是发送被阻止。
SHUT_RDWR :兼而有之。

其实这些都是宏,我们完全可以使用数字,更直观:

返回值:On success, zero is returned.  On error, -1 is returned, and errno is set appropriately.

具体也可以使用man 2 shutdown查看手册。
具体实现都是从内核层面执行。当然了,shutdown之后,接收缓冲区中的数据仍然可以被处理。

close是关闭掉对应的文件描述符并释放与描述符相关的资源,收不了也接不了。

注意,shutdown并不会关闭掉对应的文件描述符并释放与描述符相关的资源,只是对于可用性进行内核层面阻止,你要free这个fd,还是要使用close。当然了,这是Linux,如果是Windows,close要换成closesocket();
也就是说:shutdown() 提供了更细粒度(你可以关闭部分或者全关闭)的控制,而 close() 完全关闭套接字并释放资源。前者允许你在不关闭套接字的情况下,停止某个方向的数据传输。例如,你可以通过 SHUT_WR 选项来从内核层面停止该sockfd的发送数据能力,但仍然允许接收数据。即即shutdown() 适用于在应用层控制连接的关闭行为时使用。
另外如果还需要在TCP协议栈层面深入理解close and shutdown,详见:https://stackoverflow.com/questions/4160347/close-vs-shutdown-socket/23483487#23483487

我们再给出一点代码:
server.cc

#include <iostream>
#include <sys/socket.h>
#include <netinet/in.h>
#include <unistd.h>
#include <cstring>int main()
{// 创建一个 TCP 套接字int server_fd = socket(AF_INET, SOCK_STREAM, 0);if (server_fd == -1){std::cerr << "Socket creation failed!" << std::endl;return -1;}// 设置服务器地址sockaddr_in server_addr{};server_addr.sin_family = AF_INET;server_addr.sin_addr.s_addr = INADDR_ANY;server_addr.sin_port = htons(12345); // 监听端口 12345// 绑定套接字到地址if (bind(server_fd, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1){std::cerr << "Bind failed!" << std::endl;close(server_fd);return -1;}// 开始监听连接if (listen(server_fd, 1) == -1){std::cerr << "Listen failed!" << std::endl;close(server_fd);return -1;}std::cout << "Server is listening on port 12345..." << std::endl;// 等待客户端连接int client_fd = accept(server_fd, nullptr, nullptr);if (client_fd == -1){std::cerr << "Accept failed!" << std::endl;close(server_fd);return -1;}std::cout << "Client connected!" << std::endl;// 接收客户端发来的数据char buffer[1024];ssize_t bytes_received = recv(client_fd, buffer, sizeof(buffer), 0);if (bytes_received > 0){buffer[bytes_received] = '\0'; // 添加字符串结束符std::cout << "Received data: " << buffer << std::endl;}// 使用 shutdown() 关闭发送和接收功能if (shutdown(client_fd, SHUT_RDWR) == -1){std::cerr << "Shutdown failed!" << std::endl;close(client_fd);close(server_fd);return -1;}std::cout << "Shutdown the connection. No further send/recv operations allowed." << std::endl;// 尝试发送数据const char *msg = "Hello from server!";ssize_t bytes_sent = send(client_fd, msg, strlen(msg), 0);if (bytes_sent == -1){std::cerr << "Send failed!" << std::endl;}else{std::cout << "Sent data: " << msg << std::endl;}// 最后关闭连接close(client_fd);close(server_fd);return 0;
}

client.cc

telnet localhost 12345

 server收到一条数据后,将调用shutdown,届时,将无法通信。


TCP 的半连接是指在 TCP 三次握手过程中发生的一种状态。当主机 A 向主机 B 发起连接请求时,B 响应了这个请求,但 A 并没有完成第三次握手,这种情况被称为半连接。

与此相关的概念是半连接攻击,也称为 SYN 攻击。在这种攻击中,攻击者不断发送连接请求,但不完成后续的握手过程,导致主机 B 分配的内存资源被长期占用,最终可能导致资源耗尽。


TCP的半打开?

半打开:两方建立TCP连接,然后一方关闭,而另一方并不知情,这样的连接称之为半打开连接。处于半打开的连接,如果双方不进行数据通信,确实发现不了问题,仍处于连接状态的一方不会检测另外一方已经出现异常。怎么解决?
心跳机制,具体使用定时器实现,比如让服务器每隔一段时间发送报文请求回复。当主机收到这个包后,因为本地应用程序已经关闭了它的套接字,所以主机此时会给服务端回复一个RST包:报文中没有负载且TCP头部标志中的RST位被设置的一个数据包。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/482428.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java设计模式——职责链模式:解锁高效灵活的请求处理之道

嘿&#xff0c;各位 Java 编程大神和爱好者们&#xff01;今天咱们要一同深入探索一种超厉害的设计模式——职责链模式。它就像一条神奇的“处理链”&#xff0c;能让请求在多个对象之间有条不紊地传递&#xff0c;直到找到最合适的“处理者”。准备好跟我一起揭开它神秘的面纱…

Javascript中DOM事件监听 (鼠标事件,键盘事件,表单事件)

#DOM&#xff08;Document Object Model&#xff09;事件监听是一种机制&#xff0c;它允许 JavaScript 代码在 HTML 文档中的元素上监听特定的事件。当这些事件发生时&#xff0c;与之关联的 JavaScript 函数&#xff08;也称为事件处理函数&#xff09;就会被执行。这使得网页…

TiDB 无统计信息时执行计划如何生成

作者&#xff1a; weiyinghua 原文来源&#xff1a; https://tidb.net/blog/4c49ac0d 一、Pseudo 统计信息总体生成规则 TiDB 在表无统计信息时&#xff0c;不会进行动态采样&#xff0c;而是用静态的、预设规则以及经验假设来生成计划。用函数 PseudoTable 创建一个伪统…

服务器密码错误被锁定怎么解决?

当服务器密码错误多次导致账号被锁定时&#xff0c;解决方法需要根据服务器的操作系统&#xff08;如 Linux 或 Windows &#xff09;和具体服务器环境来处理。以下是常见的解决办法&#xff1a; 一、Linux 服务器被锁定的解决方法 1. 使用其他用户账号登录 如果有其他未被…

认识redis 及 Ubuntu安装redis

文章目录 一. redis概念二. redis应用场景二. redis的特性四. 使用Ubuntu安装redis 一. redis概念 redis 是在内存中存储数据的中间件, 用在分布式系统 redis是客户端服务器结构的程序, 客户端服务器之间通过网络来通信 二. redis应用场景 redis可用作数据库 类似MySQL, 但…

LabVIEW内燃机气道试验台测控系统

基于LabVIEW软件开发的内燃机气道试验台测控系统主要应用于内燃机气道的性能测试和数据分析&#xff0c;通过高精度的测控技术&#xff0c;有效提升内燃机的测试精度和数据处理能力。 项目背景 随着内燃机技术的发展&#xff0c;对其气道性能的精准测量需求日益增加。该系统通…

Rust vs Java:后端开发应该选哪个?

后端技术的发展迅速。根据JetBrains 2024年开发者调查,尽管Java仍然占据约34.5%的市场份额,但Rust在高性能应用中的应用逐渐增多。过去四年中,Rust在企业中的采用增长了240%(根据Stack Overflow 2024开发者调查)。随着组织更加注重效率和可扩展性,选择Rust还是Java已成为…

触觉智能亮相OpenHarmony人才生态大会2024

11月27日&#xff0c;OpenHarmony人才生态大会2024在武汉隆重举行。本次大会汇聚了政府领导、学术大咖、操作系统技术专家、高校及企业代表&#xff0c;围绕新时代背景下的操作系统人才培养进行了深入探讨&#xff0c;分享高校、企业在产学研融合方面的先进经验&#xff0c;全面…

springboot366高校物品捐赠管理系统(论文+源码)_kaic

毕 业 设 计&#xff08;论 文&#xff09; 高校物品捐赠管理系统设计与实现 摘 要 传统办法管理信息首先需要花费的时间比较多&#xff0c;其次数据出错率比较高&#xff0c;而且对错误的数据进行更改也比较困难&#xff0c;最后&#xff0c;检索数据费事费力。因此&#xff…

深入理解AIGC背后的核心算法:GAN、Transformer与Diffusion Models

深入理解AIGC背后的核心算法&#xff1a;GAN、Transformer与Diffusion Models 前言 随着人工智能技术的发展&#xff0c;AIGC&#xff08;AI Generated Content&#xff0c;人工智能生成内容&#xff09;已经不再是科幻电影中的幻想&#xff0c;而成为了现实生活中的一种新兴力…

企业网站面临的爬虫攻击及安全防护策略

在当今数字化时代&#xff0c;企业网站不仅是展示企业形象的窗口&#xff0c;更是进行商业活动的重要平台。然而&#xff0c;企业网站在日常运营中面临着多种类型的爬虫攻击&#xff0c;这些攻击不仅会对网站的正常访问造成影响&#xff0c;还可能窃取敏感数据&#xff0c;给企…

STM32的CAN波特率计算

公式&#xff1a; CAN波特率 APB总线频率 / &#xff08;BRP分频器 1&#xff09;/ (SWJ BS1 BS2) SWJ一般为1。 例如STM32F407的&#xff0c;CAN1和CAN2都在在APB1下&#xff0c;频率是42000000 如果想配置成1M波特率&#xff0c;则计算公式为&#xff1a;

《操作系统 - 清华大学》6 -3:局部页面置换算法:最近最久未使用算法 (LRU, Least Recently Used)

文章目录 1. 最近最久未使用算法的工作原理2. 最近最久未使用算法示例3.LRU算法实现3.1 LRU的页面链表实现3.2 LRU的活动页面栈实现3.3 链表实现 VS 堆栈实现 1. 最近最久未使用算法的工作原理 最近最久未使用页面置换算法&#xff0c;简称 LRU&#xff0c; 算法思路&#xff…

数据集-目标检测系列- 海边漫步锻炼人检测数据集 person >> DataBall

数据集-目标检测系列- 海边漫步锻炼人检测数据集 person >> DataBall DataBall 助力快速掌握数据集的信息和使用方式&#xff0c;会员享有 百种数据集&#xff0c;持续增加中。 需要更多数据资源和技术解决方案&#xff0c;知识星球&#xff1a; “DataBall - X 数据球…

【赵渝强老师】PostgreSQL的段、区和块

PostgreSQL的逻辑存储结构主要是指数据库集群、数据库、表空间、段、区、块等&#xff1b;同时PostgreSQL的逻辑存储结构也包括数据库中的各种数据库对象&#xff0c;如&#xff1a;表、索引、视图等等。所有数据库对象都有各自的对象标识符oid&#xff08;object identifiers&…

【YOLO系列复现】二、基于YOLOv6的目标检测:YOLOv6训练自己的数据集(史诗级详细教程)

官方模型&#xff1a;YOLOv6/README_cn.md at main meituan/YOLOv6 目录 1、模型和环境准备 1.1 模型下载 1.2 依赖环境安装 1.3 权重文件下载 1.4 环境测试 2、配置文件和数据集准备 2.1 准备数据集 2.2 配置文件准备 2.3 BUG修改 3、模型训练 3.1 模型训练 3.2 …

Flink常见面试题

1、Flink 的四大特征&#xff08;基石&#xff09; 2、Flink 中都有哪些 Source&#xff0c;哪些 Sink&#xff0c;哪些算子&#xff08;方法&#xff09; 预定义Source 基于本地集合的source&#xff08;Collection-based-source&#xff09; 基于文件的source&#xff08;…

【C语言】扫雷游戏(一)

我们先设计一个简单的9*9棋盘并有10个雷的扫雷游戏。 1&#xff0c;可以用数组存放&#xff0c;如果有雷就用1表示&#xff0c;没雷就用0表示。 2&#xff0c;排查(2,5)这个坐标时&#xff0c;我们访问周围的⼀圈8个位置黄色统计周围雷的个数是1。排查(8,6)这个坐标时&#xf…

【博主推荐】C#中winfrom开发常用技术点收集

文章目录 前言1.打开文件夹并选中文件2.窗体之间传参3.异步调用&#xff1a;让数据处理不影响页面操作4.创建一个多文档界面(MDI) 应用程序5.在WinForms中使用数据绑定6.在WinForms中后台使用控件的事件处理7.在WinForms中窗体跳转的几种方式8.后台处理方法中&#xff0c;调用窗…

Matlab 绘制雷达图像完全案例和官方教程(亲测)

首先上官方教程链接 polarplothttps://ww2.mathworks.cn/help/matlab/ref/polarplot.html 上实例 % 定义角度向量和径向向量 theta linspace(0, 2*pi, 5); r1 [1, 2, 1.5, 2.5, 1]; r2 [2, 1, 2.5, 1.5, 2];% 绘制两个雷达图 polarplot(theta, r1, r-, LineWidth, 2); hold …