【Linux】Socket阻塞和非阻塞、同步与异步

创作不易,本篇文章如果帮助到了你,还请点赞 关注支持一下♡>𖥦<)!!
主页专栏有更多知识,如有疑问欢迎大家指正讨论,共同进步!
🔥Linux系列专栏:Linux基础 🔥

给大家跳段街舞感谢支持!ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ

在这里插入图片描述


目录

  • 阻塞
  • 非阻塞
  • 同步和异步

阻塞

阻塞模式下,send函数返回时,只代表需要发送的数据已经被拷贝到内核的发送缓冲区

并且如果发送缓冲区有足够的空间,数据可能已经开始从发送缓冲区发送到网络上。但是并不代表数据已经成功地被接收方接收

当发送缓冲区空间不够大的时候,等到发送缓冲区空间足够大再发送

调用recv函数并且套接字没有数据可读时,recv函数会阻塞当前线程,直到有数据可读或者出现错误。

如果 recv 函数返回,返回值代表内核的接收缓冲区中的数据已经被拷贝到用户空间

/* Windows下UDP阻塞socket*/#include <iostream>
#include <WinSock2.h>
using namespace std;#pragma comment(lib,"Ws2_32.lib")int main()
{//1.加载库,相当于将dll拷贝到exe路径下WORD wVersion = MAKEWORD(2, 2);WSADATA wsadata;int errcode = 0;errcode = WSAStartup(wVersion, &wsadata);//1.1判断WSAStartup是否成功if (errcode != 0){cout << "WSAStartup error:" << errcode << endl;return -1;}//1.2判断版本是否正确if (LOBYTE(wsadata.wVersion)!= 2 || HIBYTE(wsadata.wVersion) != 2){cout << "WSAStartup wVersion error" << endl;WSACleanup();			return -1;	}else //库和版本都加载成功{cout << "WSAStartup success!" << endl;}//2.创建套接字SOCKET sock;sock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);//2.1判断套接字是否创建成功if (sock == INVALID_SOCKET)		//INVALID无效的{cout << "set socket error:" << WSAGetLastError() << endl;WSACleanup();return -1;}else   //套接字创建成功{cout << "set socket success!" << endl;}//3.绑定ip和端口号struct sockaddr_in addrServer;addrServer.sin_family = AF_INET;addrServer.sin_addr.S_un.S_addr = INADDR_ANY;addrServer.sin_port = htons(8080);	//htons将端口号转换为大端模式errcode = bind(sock, (sockaddr*)&addrServer, sizeof(addrServer));//3.1判断绑定是否成功if (errcode == SOCKET_ERROR){cout << "bind error:" << WSAGetLastError() << endl;//关闭套接字closesocket(sock);//卸载库WSACleanup();return -1;}else{cout << "bind success!" << endl;}//4.循环接收发送消息int recvNum = 0;int sendNum = 0;char recvbuf[1024] = "";char sendbuf[1024] = "";struct sockaddr_in addrClient;int addrClientSize = sizeof(addrClient);while (true){//接收消息recvNum = recvfrom(sock, recvbuf, sizeof(recvbuf), 0, (sockaddr*)&addrClient, &addrClientSize);//判断接收成功if (recvNum > 0){//打印收到的数据//192.168.1.1——十进制四等分字符串类型ip地址:inet_addr//ulong类型ip地址:inet_ntoacout << "来自ip:" << inet_ntoa(addrClient.sin_addr) << " 的消息: " << recvbuf << endl;}else if (recvNum == 0)		//连接关闭{cout << "connection closed" << endl;break;}else                        //接收错误{cout << "recvfrom error:" << WSAGetLastError() << endl;break;}//发送消息gets_s(sendbuf);sendNum = sendto(sock, sendbuf, sizeof(sendbuf), 0, (sockaddr*)&addrClient, addrClientSize);//判断发送成功if (sendNum != 0){cout << "send success!" << endl;}else{cout << "sendto error:" << WSAGetLastError() << endl;}}//5.关闭套接字closesocket(sock);//6.卸载库WSACleanup();return 0;
}

阻塞 IO 开发简单,但不适合多个客户端高并发


非阻塞

在非阻塞模式下,send函数将数据放入内核的发送缓冲区,并立即返回

但是,如果发送缓冲区没有足够的空间来容纳全部数据,send函数会返回已经成功拷贝到发送缓冲区的字节数,有多少空间就往里拷贝多少内容,而不是全部要发送的数据

当发送缓冲区空间不够大的时候,有多少空间就往里拷贝多少内容,剩下的数据应用程序自己处理

当调用recv函数并且套接字没有数据可读时,recv函数会立即返回一个错误(EWOULDBLOCK或EAGAIN),而不是阻塞当前线程。

/* Windows下UDP非阻塞socket*/
#include <iostream>
#include <WinSock2.h>
using namespace std;#pragma comment(lib,"Ws2_32.lib")int main()
{//1.加载库,相当于将dll拷贝到exe路径下WORD wVersion = MAKEWORD(2, 2);WSADATA wsadata;int errcode = 0;errcode = WSAStartup(wVersion, &wsadata);//1.1判断WSAStartup是否成功if (errcode != 0){cout << "WSAStartup error:" << errcode << endl;return -1;}//1.2判断版本是否正确if (LOBYTE(wsadata.wVersion)!= 2 || HIBYTE(wsadata.wVersion) != 2){cout << "WSAStartup wVersion error" << endl;WSACleanup();			return -1;	}else //库和版本都加载成功{cout << "WSAStartup success!" << endl;}//2.创建套接字SOCKET sock;sock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);//Windows下设置socket为非阻塞u_long val = 1;ioctlsocket(sock, FIONBIO, &val);//Linux下://int flags = fcntl(sock, F_GETFL, 0);//fcntl(sock, F_SETFL, flags | O_NONBLOCK);//2.1判断套接字是否创建成功if (sock == INVALID_SOCKET)		//INVALID无效的{cout << "set socket error:" << WSAGetLastError() << endl;WSACleanup();return -1;}else   //套接字创建成功{cout << "set socket success!" << endl;}//3.绑定ip和端口号struct sockaddr_in addrServer;addrServer.sin_family = AF_INET;addrServer.sin_addr.S_un.S_addr = INADDR_ANY;addrServer.sin_port = htons(8080);	//htons将端口号转换为大端模式errcode = bind(sock, (sockaddr*)&addrServer, sizeof(addrServer));//3.1判断绑定是否成功if (errcode == SOCKET_ERROR){cout << "bind error:" << WSAGetLastError() << endl;//关闭套接字closesocket(sock);//卸载库WSACleanup();return -1;}else{cout << "bind success!" << endl;}//4.循环接收发送消息int recvNum = 0;int sendNum = 0;char recvbuf[1024] = "";char sendbuf[1024] = "";struct sockaddr_in addrClient;int addrClientSize = sizeof(addrClient);while (true){//接收消息recvNum = recvfrom(sock, recvbuf, sizeof(recvbuf), 0, (sockaddr*)&addrClient, &addrClientSize);//判断接收成功if (recvNum > 0){//打印收到的数据//192.168.1.1——十进制四等分字符串类型ip地址:inet_addr//ulong类型ip地址:inet_ntoacout << "来自ip:" << inet_ntoa(addrClient.sin_addr) << " 的消息: " << recvbuf << endl;}else if (recvNum == 0)		//连接关闭{cout << "connection closed" << endl;break;}else if (WSAGetLastError() != 10035) {		//非阻塞{cout << "recvfrom error:" << WSAGetLastError() << endl;break;}}//5.关闭套接字closesocket(sock);//6.卸载库WSACleanup();return 0;
}

同步和异步

  • 同步

同步模型:一个操作必须等待前一个操作完成后才能进行。比如说,当客户端向服务器发送一个请求并期望得到响应时,它会处于阻塞状态,直到服务器返回响应

read(), write(), recv(), send()


  • 异步

异步模型则打破了这种顺序执行的限制。在异步 I/O 中,客户端发送请求后不会被阻塞,而是可以继续执行其他操作。当服务器的响应准备好时,通过回调函数、事件机制或者消息队列等方式通知客户端

异步模型的优点在于能够极大地提高系统的并发性和响应性。它能够充分利用系统资源,避免了因等待而造成的资源浪费

同步和异步的区分:是否是用户自己写代码实现的将数据从内核缓冲区拷贝到用户缓冲区,如果不是自己写代码实现的,通过回调函数、事件通知或其他机制告知用户 就是异步


在这里插入图片描述

大家的点赞、收藏、关注将是我更新的最大动力! 欢迎留言或私信建议或问题。
大家的支持和反馈对我来说意义重大,我会继续不断努力提供有价值的内容!如果本文哪里有错误的地方还请大家多多指出(●'◡'●)

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

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

相关文章

【ajax核心02】底层原理-Promise对象

目录 一&#xff1a;promise对象是什么 二&#xff1a;语法&#xff08;Promise使用步骤&#xff09; 三&#xff1a;Promise-三种状态 一&#xff1a;promise对象是什么 Promise 对象代表异步操作最终的完成&#xff08;或失败&#xff09;以及其结果值。 即Promise对象是…

uniapp接入微信小程序原生代码配置方案(优化版)

uniapp项目需要把微信小程序原生语法的功能代码嵌套过来&#xff0c;无需把原生代码转换为uniapp&#xff0c;可以配置拷贝的方式集成过来 1、拷贝代码包到src目录 2、vue.config.js中配置原生代码包直接拷贝到编译目录中 3、pages.json中配置分包目录&#xff0c;原生入口…

java基于ssm+jsp 医院远程诊断系统

1前台首页功能模块 医院远程诊断系统&#xff0c;在系统首页可以查看首页、医生信息、论坛信息、我的、跳转到后台、客服等内容&#xff0c;如图1所示。 图1前台首页功能界面图 用户登录&#xff0c;在用户登录页面可以填写用户名、密码、等信息进行用户登录&#xff0c;如图2…

安装Django Web框架

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 Django是基于Python的重量级开源Web框架。Django拥有高度定制的ORM和大量的API&#xff0c;简单灵活的视图编写&#xff0c;优雅的URL&#xff0c;适…

近2年时间,华为手机上的卫星通信功能发展成怎样了?

自从Mate 50 系列支持北斗卫星短报文功能以来&#xff0c;已经过去了近2年的时间&#xff0c;卫星相关的功能也从最开始的摸索、罕见&#xff0c;逐渐变得成熟、在各品牌旗舰机上常见起来。 那么&#xff0c;这近两年的发展&#xff0c;卫星相关的功能都有了怎样的变化呢&…

史上最全整合nacos单机模式整合哈哈哈哈哈

Nacos 是阿里巴巴推出的一个新开源项目&#xff0c;它主要是一个更易于构建云原生应用的动态服务发现、配置管理和服务管理平台。 Nacos提供了一组简单易用的特性集&#xff0c;帮助用户快速实现动态服务发现、服务配置、服务元数据及流量管理。 Nacos 的关键特性包括&#x…

简过网:公务员报班和不报班的有区别吗?

很多备考公务员的朋友可能都会有这种纠结&#xff0c;到底要不要报个培训班呢&#xff0c;报班了怕没什么用&#xff0c;不报班又怕自己考不上&#xff0c;如果你也有这个疑问&#xff0c;那么不妨来看看这篇文章&#xff01; ​ 先说一下&#xff0c;公务员报班和不报班的有区…

平方根的三种求法(袖珍计算器算法,二分查找,牛顿迭代)

一、袖珍计算器 袖珍计算器方法主要运用到了我们高数上所学的关于e底数转化的思想&#xff0c;即 一种用指数函数 exp⁡ 和对数函数 ln⁡ 代替平方根函数的方法 : 1、exp函数&#xff1a; exp是 C 标准库 <math.h> 中的一个函数&#xff0c;用于计算 e 的 x 次幂&…

微服务(服务治理)

服务远程调用时存在的问题 注册中心原理 服务治理中的三个角色分别是什么&#xff1f; 服务提供者&#xff1a;暴露服务接口&#xff0c;供其它服务调用服务消费者&#xff1a;调用其它服务提供的接口注册中心&#xff1a;记录并监控微服务各实例状态&#xff0c;推送服务变更信…

【乐吾乐2D可视化组态编辑器】图表动态显示

1. 添加数据 乐吾乐2D可视化组态编辑器地址&#xff1a;https://2d.le5le.com/ 图表动态展示是指一个图表图元的数据属性&#xff08;一般是dataY&#xff09;绑定多个变量&#xff0c;建立通信后数据动态变化展示。 官网默认Echarts图表拖拽到画布中是已经添加了图元的da…

【ChatBI】超轻量Python库Vanna快速上手,对接oneapi

oneapi 准备 首先确保你有oneapi &#xff0c;然后申请 kimi的api 需要去Moonshot AI - 开放平台 然后添加一个api key 然后打开oneapi的渠道界面&#xff0c;添加kimi。 然后点击 测试&#xff0c; 如果能生成响应时间&#xff0c;就是配置正确。 然后创建令牌 http:…

Ant Design Vue中的Table和Tag的基础应用

目录 一、Table表格 1.1、显示表格 1.2、列内容过长省略展示 1.3、完整分页 1.4、表头列颜色设置 二、Tag标签 2.1、根据条件显示不同颜色 2.2、控制关闭事件 一、Table表格 效果展示&#xff1a; 官网&#xff1a;Ant Design Vue 1.1、显示表格 <a-tableref&quo…

k8s集群搭建及对一些组件的简单理解

背景 k8s的学习环境&#xff08;用kubeadm方式搭建&#xff09;&#xff0c;我也搭过几次了&#xff0c;但都有点问题。 要么在云服务器上弄&#xff0c;这个的问题是就只有一台轻量服务器&#xff0c;只能搭个单节点的&#xff1b;后来买了一台便宜的&#xff0c;所以就有了两…

MIT6.s081 2021 Lab Utilities

Boot xv6 按照示例切换到 util 分支后&#xff0c;看到目录下包含 Makefile 文件&#xff0c;执行 make qemu 即可。 sleep 思路 借助系统调用 sleep 实现一个命令行程序&#xff0c;关键是要找到封装了系统调用的 C 函数的位置&#xff0c;根据提示&#xff1a; … user/u…

【LLM】GLM系列模型要点

note 文章目录 noteGLM一、数据层面1. 预训练数据 二、GLM4模型层面三、GLM-4 All Tools四、GLM的其他技术Reference GLM Paper&#xff1a;https://arxiv.org/abs/2406.12793 GitHub&#xff1a;https://github.com/THUDM HF&#xff1a;https://huggingface.co/THUDM 经过…

将独立的 Python 网络应用程序分发给非技术用户

1. 问题背景 我们需要编写一个 Python 网络应用程序&#xff0c;供教师和学生在课堂上使用。该应用程序将在托管的网站上运行&#xff0c;但我们也希望用户能够下载一个自包含的应用程序&#xff0c;以便他们可以在本地安装&#xff0c;以获得更好的性能或他们根本无法在教室中…

【Solr 学习笔记】Solr 源码启动教程

Solr 源码启动教程 本教程记录了如何通过 IDEA 启动并调试 Solr 源码&#xff0c;从 Solr9 开始 Solr 项目已由 ant 方式改成了 gradle 构建方式&#xff0c;本教程将以 Solr 9 为例进行演示&#xff0c;IDE 选择使用 IntelliJ IDEA。 Solr github 地址&#xff1a;https://gi…

对比A100和4090:两者的区别以及适用点

自2022年年末英伟达发布4090芯片以来&#xff0c;这款产品凭借着其优异的性能迅速在科技界占据了一席之地。现如今&#xff0c;不论是在游戏体验、内容创作能力方面还是模型精度提升方面&#xff0c;4090都是一个绕不过去的名字。而A100作为早些发布的产品&#xff0c;其优异的…

盘点5款最热门的AI绘画软件!总有一款是你的菜

在数字化艺术日益盛行的今天&#xff0c;AI绘画软件成为了创作者们的新宠。这些软件不仅能够帮助艺术家们快速生成独特的艺术作品&#xff0c;还能为普通用户带来全新的绘画体验。今天&#xff0c;我们就来盘点五款最热门的AI绘画软件&#xff0c;看看哪一款是你的菜&#xff0…

Kubernates容器化JVM调优笔记(内存篇)

Kubernates容器化JVM调优笔记&#xff08;内存篇&#xff09; 先说结论背景思路方案 先说结论 1、首先如果是JDK8&#xff0c;需要使用JDK8_191版本以上&#xff0c;才支持容器化环境和以下参数&#xff0c;否则就更新到JDK10以上&#xff0c;选择对应的镜像构建就行了 2、在容…