Linux系统编程--网络编程

news2025/1/15 16:37:06

一、OSI网络七层模型

OSI模型将整个网络通信过程分解为七个层次,每个层次都为网络通信提供了特定的功能。以下是OSI模型的七个层次,从上到下依次是:

  1. 应用层(Application Layer):为应用软件提供网络服务,如HTTP、FTP、SMTP等协议,处理网络应用程序的具体功能。

  2. 表示层(Presentation Layer):确保一个系统的应用层所发送的信息可以被另一个系统的应用层读取,负责数据格式转换、数据加密解密等。

  3. 会话层(Session Layer):建立、管理和终止应用程序之间的会话,提供数据交换定界和同步功能,包括建立检查点和恢复方案。

  4. 传输层(Transport Layer):负责主机中两个进程的对话控制,如TCP(传输控制协议)和UDP(用户数据报协议),处理数据包的顺序、错误校验、流量控制等。

  5. 网络层(Network Layer):负责数据包从源到宿的传递和路由选择,如IP(互联网协议)。

  6. 数据链路层(Data Link Layer):传输有地址的帧以及错误检测功能,如以太网协议。

  7. 物理层(Physical Layer):处理通过物理设备传输数据的技术细节,如网线、光纤、无线电波等。

  8. OSI模型的每一层都为它的上层提供服务,同时依赖于它的下层。这种分层的设计使得网络通信更加模块化,便于理解和设计复杂的网络系统。在实际应用中,TCP/IP模型是更为广泛使用的网络模型,它简化了层次结构,但基本原理与OSI模型相似。

后来人们对 OSI 进行了简化,合并了一些层,最终只保留了 4 层,从下到上分别是接口层、网络层、传输 层和应用层,这就是大名鼎鼎的 TCP/IP 模型。

本次讲的是spcket编程,是站在传输层的基础上,可以使用TCP/UDP协议。

二、网络数据传输过程

平常使用的软件都是通过应用层来访问网络,程序产生的数据会一层层地往下传输,直到最后的网络接口层,就通过网线发送到互联网上去了。

数据向下:原始数据每通过一层,就会多一层协议的包装,通过四层,原始数据变成包装数据。

数据向上:包装数据每通过一层,就会少一层协议的包装,通过四层,包装数据被完全拆解成原始数据。

1.TCP通信协议

TCP 是面向连接的传输协议,可靠性传输,建立连接时要经过三次握手,断开连接时要经过四次挥手,中间 传输数据时也要回复 ACK 包确认,多种机制保证了数据能够正确到达,不会丢失或出错。

1、TCP 的 3 次握手过程 1、客户端发送 TCP 连接请求 客户端会随机一个初始序列号 seq=x(client_isn),设置 SYN=1,表示这是 SYN 握手报文。然后 就 可以把这个 SYN 报文发送给服务端了,表示向服务端发起连接,之后客户端处于同步已发送状态。

2、服务端发送针对 TCP 连接请求的确认,服务端收到客户端的 SYN 报文后,也随机一个初始序列号 (server_isn)(seq=y),设置 ack=x+1, 表示收到了客户端的 x 之前的数据,希望客户端下次发送的数据从 x+1 开始。设置 SYN=1 和 ACK=1。表示这是一个 SYN 握手和 ACK 确认应答报文。 最后把该报文发给客户端,该报文也不包含应用层数据,之后服务端处于同步已接收状态。

3、客户端发送确认的确认 客户端收到服务端报文后,还要向服务端回应最后一个应答报文,将 ACK 置为 1 ,表示这是一个 应答报文 ack=y+1 ,表示收到了服务器的 y 之前的数据,希望服务器下次发送的数据从 y+1 开始。最 后把报文发送给服务端,这次报文可以携带数据,之后客户端处于连接已建立 状态。服务器收到客户 端的应答报文后,也进入连接已建立状态 通过这样的三次握手过程,TCP 能够确保双方能够收到对方的请求和回应,并且双方都知道彼此的初始 序列号和确认号。这样建立起来的连接可以提供可靠的数据传输和顺序控制。 ACK:确认序号有效。 SYN:发起一个新连接。

CLOSED:不在连接状态(这是为方便描述假想的状态,实际不存在)

LISTEN:等待从任何远端 TCP 和端口的连接请求。

SYN_SENT:发送完一个连接请求后等待一个匹配的连接请求。

syn_sent SYN_RCVD: 这个状态表示接受到了 SYN 报文,在正常情况下,这个状态是服务器端的 SOCKET 在建立 TCP 连接时的三次握手会话过程中的一个中间状态,很短暂,基本上用 netstat 你是很难看到这种状态的,除非你特 意写了一个客户端测试程序,故意将三次 TCP 握手过程中最后一个 ACK 报文不予发送。因此这种状态时,当收到客户端的 ACK 报文后,它会进入到 ESTABLISHED 状态 ESTABLISHED:表示一个打开的连接,接收到的数据可以被投递给用户。连接的数据传输阶段的正常状态

为什么是三次握手,为什么不是两次或者四次?

主要原因:防止已经失效的连接请求报文突然又传送到了服务器,从而产生错误 如果采用两次握手会出现以下情况: 客户端向服务器端发送的请求报文由于网络等原因滞留,未能发送到服务器端,此时连接请求报文失效, 客户端会再次向服务器端发送请求报文,之后与服务器端建立连接,当连接释放后,由于网络通畅了,第一次客 户端发送的请求报文又突然到达了服务器端,这条请求报文本该失效了,但此时服务器端误认为客户端又发送了 一次连接请求,两次握手建立好连接,此时客户端忽略服务器端发来的确认,也不发送数据,造成不必要的错误 和网络资源的浪费。 如果采用三次握手的话,就算那条失效的报文发送到服务器端,服务器端确认并向客户端发送报文,但此时 客户端不会发出确认,由于客户端没有确认,由于服务器端没有接收到确认,就会知道客户端没有请求连接。

为什么不是四次?如果三次就能够确定正常连接,就没有必要在进行确认,来浪费资源了

2.UDP通信协议

UDP 是非面向连接的传输协议,没有建立连接和断开连接的过程,它只是简单地把数据丢到网络中,也不 需要 ACK 包确认。在数据传输过程中延迟小、数据传输效率高。 当强调传输性能而不是传输的完整性时,如:音频和多媒体应用,UDP 是最好的选择。

总结:

短链接,不连接通信 

相对来说没有 TCP 那么稳定

有可能丢失相应数据  

它的发送速度相对TCP来说比较快

3.网络编程的 IP 地址

windows 下:ipconfig

Linux 下:ifconfig

1、ipv4地址是32(bit)地址数据、现在发展到ipv6

2、ip由网络号和主机号组成

3、子网掩码:255.255.255.0

4、端口

端口(Port)是一个逻辑概念,用于区分不同的服务或进程。端口号与IP地址一起工作,帮助网络协议(如TCP/IP)确定数据应该被送往网络中哪一台主机的哪个具体服务。端口号是16位的数字,范围从0到65535。

端口号分为几个范围:

  1. 知名端口(Well-Known Ports):范围从0到1023,通常被系统或者常见服务和应用程序使用。例如,HTTP服务通常使用端口80,HTTPS服务使用端口443,而FTP服务使用端口21。

  2. 注册端口(Registered Ports):范围从1024到49151,用于用户或应用程序。这些端口号用于运行用户开发的服务或应用程序,但它们不像知名端口那样广为人知。

  3. 动态或私有端口(Dynamic or Private Ports):范围从49152到65535,通常不由任何服务固定使用,而是由各种应用程序在需要时临时使用。

端口的作用包括:

  • 区分服务:不同的网络服务监听不同的端口,例如,Web服务器监听端口80或443,而电子邮件服务器可能监听端口25(SMTP)或110(POP3)。

  • 进程间通信:在一台主机上,不同的进程可以通过不同的端口号进行网络通信。

  • 网络配置:端口号也用于配置网络服务,如防火墙和路由器规则,允许或阻止对特定端口的访问。

  • 安全性:通过限制对某些端口的访问,可以提高网络安全性。例如,关闭不常用或不必要的端口可以减少潜在的安全风险。

在网络编程中,端口的使用通常涉及以下步骤:

  • 绑定(Binding):一个网络应用程序在启动时会绑定到一个特定的端口号,这样它就可以监听进入该端口的网络请求。

  • 监听(Listening):绑定到端口的应用程序会进入监听状态,等待网络请求的到来。

  • 连接(Connecting):当客户端程序需要与服务端通信时,它会指定服务器的IP地址和端口号来建立连接。

  • 通信:一旦连接建立,客户端和服务器就可以通过这个端口进行数据传输。

端口的使用对于网络通信至关重要,它们使得复杂的网络环境能够有序地运行。

5.TCP编程协议框架

6.TCP 服务端编写流程

1、创建套接字:sockeet

int socket(int domain, int type, int protocol);

参数: domain: 网域

        AF_INET :IPv4

        AF_INET6 :IPv6

        AF_UNIX :本地通讯

type:选择传输协议 tcp /udp

        SOCK_STREAM ;tcp

        SOCK_DGRAM : udp

protocol: 基本废弃,一般赋 0

返回值:

        成功返回描述网络套接字 sockfd,失败返回-1

举例:

        创建描述网络的套接字:

         int sfd = socket(AF_INET,SOCK_STREAM,0)

2、绑定:bind

绑定一个端口号和 IP 地址,使套接口与指定的端口号和 IP 地址相关

#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
int bind(int sockfd, const struct sockaddr *my_addr, socklen_t addrlen)

参数:sockfd 为前面 socket 的返回值,描述网络的套接字

        my_addr:封装 ip 地址和端口号 struct sockaddr //一般很少用

struct sockaddr //一般很少用
{
    unsigned short int sa_family; // AF_INET。
    char sa_data[14]; //IP 和端口
};
#include<netinet/in.h>
struct sockaddr_in //常用的结构体
{
    unsigned short int sin_family; //AF_INET
    uint16_t sin_port; //使用的 port 编号
    struct in_addr sin_addr; // IP 地址
    unsigned char sin_zero[8]; //未使用
};

端口:10000-65535
12345
htons h-host to n-net s:short
转化:
    uint32_t htonl(uint32_t hostlong);//本函数将一个 32 位数从主机字节顺序转换成            
     无符号长整型网络字节
顺序:
uint16_t htons(uint16_t hostshort);//将一个无符号短整型的主机数值转换为网络字节顺序
uint32_t ntohl(uint32_t netlong);//将一个无符号长整形数从网络字节顺序转换为主机字节顺序。
uint16_t ntohs(uint16_t netshort);//将一个 16 位数由网络字节顺序转换为主机字节顺序。
IP 的填写方式:
struct in_addr
{
    uint32_t s_addr; //=inet_addr("192.168.1.22");
};
addrlen:第二个参数大小
返回值:成功则返回 0,失败返回-1
举例:
#define PORT 33333
#define IP "192.168.110.123" struct sockaddr_in ser_addr;
ser_addr.sin_family = AF_INET;
ser_addr.sin_port = htons(PORT);
ser_addr.sin_addr.s_addr = inet_addr(IP);
inet_aton(“192.168.1.22”,&ser_addr.sin_addr);
bind(sfd,(struct sockaddr)&ser_addr,sizeof(ser_addr));

3、监听:listen

设置允许的最大连接数(瞬时处理的阈值),listen 函数

使服务器的这个端口和 IP 处于监听状态,等待网络中某一客户机的连接请求。如果客户端有连接请求, 端口就会接受这个连接

int listen(int sockfd, int backlog);
参数
    sockfd 为前面 socket 的返回值,sfd
    backlog 指定同时能处理的最大连接要求,通常为 10 或者 5。 最大值可设至 128(不是最多可以连接 128 个客户端,是一个瞬时处理的阈值)

返回值:成功则返回 0,失败返回-1
例子:listen(sfd,10);

4、等待客户端连接:accept

int accept(int sockfd, struct sockaddr *addr,socklen_t *addrlen);

参数 :

        sockfd 为前面 socket 的返回值,即 sfd

        addr:提供空间,用于接受客户端的 ip 地址和端口号

        addrlen:第二个参数大小

返回值: 返回新的套接字描述符,专门用于与建立的客户端通信 失败-1;

举例:

struct sockaddr_in cli_addr = {0};

socklen_t len = sizeof(cli_addr);

int cfd = accept(sfd,(struct sockaddr *)&cli_addr,&len)

5、write 和 read 进行通信

ssize_t send(int s, const void *buf, size_t len, int flags);

参数: s:通信套接字

        buf:要发送的数据缓冲区

        len: 数据长度

        flags:一般赋 0 .阻塞

返回值:成功返回真正发送的数据长度,失败-1

ssize_t recv(int s, void* buf,size_t len,int flags);

参数 s:通信的套接字

        buf:存放接收数据的缓冲区

        len: 数据长度

        flags:一般赋 0 .阻塞

返回值:成功返回真正接收的数据长度,失败-1

7.TCP 客户端编写流程

1、创建套接字:socket

#include <sys/socket.h>
int socket(int domain, int type, int protocol);

参数:

        domain: 网域

                AF_INET :IPv4

                AF_INET6 :IPv6

        type:选择传输协议 tcp /udp

                SOCK_STREAM ;tcp

                SOCK_DGRAM : udp

        protocol: 基本废弃,一般赋 0

返回值:成功返回套接字 fd,失败返回-1

2、connect : 连接
int connect(int sockfd, const struct sockaddr*serv_addr, socklen_t addrlen)

参数:sockfd 为前面 socket 的返回值,即 fd

        serv_addr 为结构体指针变量,存储着远程服务器的 IP 与端口号信息。

        addrlen 表示结构体变量的长度

返回值:成功则返回 0,失败返回-1

举例:
int fd = socket(AF_INET,SOCK_STREAM,0);
#define PORT 33333
#define IP "192.168.110.123" struct sockaddr_in ser_addr;
ser_addr.sin_family = AF_INET;
ser_addr.sin_port = htons(PORT);
ser_addr.sin_addr.s_addr = inet_addr(IP);
connect(fd,(struct sockaddr *)&ser_addr,sizeof(ser_addr))
3、write 和 read 进行通信 或者 send 和 recv 进行通信

TCP 问题: 当服务器结束之后,再次运行会出现 bind 错误(地址被占用)

解决办法:

        int sfd = socket(AF_INET,SOCK_STREAM,0); 创建套接字之后,

用:

        int val = 1;

        setsockopt(sfd,SOL_SOCKET,SO_REUSEADDR,&val,sizeof(val));         setsockopt(sfd,SOL_SOCKET,SO_REUSEPORT,&val,sizeof(val));

7、单播、广播、多播

单播 -- 一对一 广播 -- 一对多 组播 -- 多

UDP 不像 TCP,无需在连接状态下交换数据,因此基于 UDP 的接收方和发送方也无需经过连接过程。也就 是说,不必调用 listen()和 accept()函数。UDP 中只有创建套接字的过程和数据交换的过程。不管是服务器端还是 客户端都只需要 1 个套接字

udp 具有广播功能,即一个发送方,多个接收方; 广播:处于局部网中的所有设备都可以接收到消息 比如:校园广播 广播的地址:网络号不变,主机号为 25

对一组特定的主机发送消息 比如:直播,多播(D 类),IP 地址分为:224.0.0.0~239.255.255.255

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

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

相关文章

MySQL部署系列-centos离线安装MySQL

MySQL部署系列-centos离线安装MySQL 文章目录 MySQL部署系列-centos离线安装MySQL1. 查看是否已经安装 Mysql3. 下载官方 Mysql 包3. 下载之后上传到服务器4. 创建用户组5. 创建数据目录并赋予权限6. 修改配置文件 vim /etc/my.cnf7. 初始化数据库(数据库安装)8. 加入到系统服务…

多个开源的js补环境框架测试

原文链接&#xff1a;https://mp.weixin.qq.com/s/uEMFGpE5bqmTvzSgX2twvA 前言 在做js逆向时肯定会遇到补环境的情况&#xff0c;看到github开源了好几个补环境用的框架&#xff0c;这篇文章做个测试&#xff0c;看看哪个比较好用。 https://github.com/pysunday/sdenvhttp…

word格式技巧

文章目录 论文格式技巧论文交叉引用怎么弄论文的页码怎么弄 论文格式技巧 论文交叉引用怎么弄 1.取消文献原有的编号 2.定义新编号 3.具体编号设置 4.在引用的地方插入&#xff0c;具体引用选项卡–>交叉引用–>选择后插入 2. 4. 论文的页码怎么弄 假设我们有这样一…

探索DeepSeek平台:新一代MoE模型的深度体验

简介 DeepSeek是一个创新的人工智能平台&#xff0c;它最近推出了其最新版本的模型——DeepSeek-V2 MoE&#xff08;Mixture of Experts&#xff09;。这个平台不仅提供了一个交互式的聊天界面&#xff0c;还提供了API接口&#xff0c;让用户可以更深入地体验和利用这一先进的…

scala速通(精简版)

1.变量和常量 var name [:VariableType] value // variable val name [:ConstantType] value // constant1.声明变量时&#xff0c;类型可以省略 2.类型定义后就不能修改言 3.变量声明必须有初始值 4.变量&#xff0c;常量分别用var&#xff0c;val声明修饰 2.标识符命名…

构建自己的docker镜像node.js

学习资源&#xff1a; 构建自己的 Docker 镜像_哔哩哔哩_bilibili 针对其中的一些比较困难的点写篇文章。 以下是对app.js的注释&#xff1a; // 使用 Koa 框架搭建 Node.js 应用的示例代码// 这两行代码引入了 koa 模块&#xff0c;并创建了一个新的 Koa 应用实例&#xf…

vue2项目升级到vue3经历分享4

后端重构&#xff0c;如果接口做好抽象封装&#xff0c;只需要考虑jar之间的兼容性问题&#xff0c;jdk版本不变&#xff0c;基本不用做太大的调整&#xff0c;但是前端就不一样&#xff0c;除了vue框架本身&#xff0c;css的调整&#xff0c;改起来更是让人头疼。前面写了vue2…

如何让vim支持python3

首先删除旧的vim。 sudo apt-get remove vim //输入re按下tab直接显示remove sudo apt-get remove vim-runtime sudo apt-get remove vim -tiny sudo apt-get remove vim-common 然后下载vim8源码&#xff1a; git clone https://github.com/vim/vim.git 进行编译安装…

一键剪辑1000条视频的矩阵系统小魔推到底有多牛?

小魔推是一款短视频营销工具&#xff0c;主要针对想做短视频营销的实体商家与企业。通过BGC、PGC、UGC流量的打造&#xff0c;帮助更多实体行业实现流量裂变与转化。通过小魔推不需要做额外的拍摄剪辑创作动作&#xff0c;只需要通过小魔推宣传码&#xff0c;就能一键发布带有门…

20240508请问GTX2080TI的300和300A核心的差异?

20240508请问GTX2080TI的300和300A核心的差异&#xff1f; 在拼多多/淘宝上&#xff0c;GTX2080TI的300A核心的会比300核心的贵100&#xffe5;左右。 但是怎么区分呢&#xff1f; 300a核心和300请问怎么区分呢&#xff1f;[嘻嘻] devicr ID diviceid 1e07是300a 1e04是300 Gp…

2024 GESP6级 编程第一题 游戏

题目描述 你有四个正整数 &#xff0c;并准备用它们玩一个简单的小游戏。 在一轮游戏操作中&#xff0c;你可以选择将 减去 &#xff0c;或是将 减去 。游戏将会进行多轮操作&#xff0c;直到当 时游戏结束。 你想知道游戏结束时有多少种不同的游戏操作序列。两种游戏操作…

docker-compose部署gitlab

需要提前安装docker和docker-compose环境 参考&#xff1a;部署docker-ce_安装部署docker-ce-CSDN博客 参考&#xff1a;docker-compose部署_docker compose部署本地tar-CSDN博客 创建gitlab的数据存放目录 mkdir /opt/gitlab && cd mkdir /opt/gitlab mkdir {conf…

Python深度学习基于Tensorflow(6)神经网络基础

文章目录 使用Tensorflow解决XOR问题激活函数正向传播和反向传播解决过拟合权重正则化Dropout正则化批量正则化 BatchNormal权重初始化残差连接 选择优化算法传统梯度更新算法动量算法NAG算法AdaGrad算法RMSProp算法Adam算法如何选择优化算法 使用tf.keras构建神经网络使用Sequ…

【C++】二叉搜索树(手撕插入、删除、寻找)

一、什么是二叉搜索树 二叉搜索树又称二叉排序树&#xff0c;它或者是一棵空树&#xff0c;或者是具有以下性质的二叉树: 若它的左子树不为空&#xff0c;则左子树上所有节点的值都小于根节点的值若它的右子树不为空&#xff0c;则右子树上所有节点的值都大于根节点的值它的左…

【Linux】25. 网络基础(一)

网络基础(一) 计算机网络背景 网络发展 独立模式: 计算机之间相互独立; 网络互联: 多台计算机连接在一起, 完成数据共享; 其实本质上一台计算机内部也是一个小型网络结构(如果我们将计算机内部某个硬件不存放在电脑中&#xff0c;而是拉根长长的线进行连接。这其实也就是网…

【Arduino IDE 2】Windows平台安装ESP8266 NodeMCU LittleFS Uploader(文件上传插件)

在Arduino IDE 2&#xff08;2.2.1或更高版本&#xff09;上&#xff0c;如何安装基于ESP8266 NodeMCU的LittleFS文件系统上传插件&#xff0c;以及如何将文件上传到ESP8266 NodeMCU板文件系统。 一、LittleFS简介 LittleFS是一个为微控制器创建的轻量级文件系统&#xff0c;可…

五种算法(BWO、RUN、SO、HO、GWO)求解复杂城市地形下无人机路径规划,可以修改障碍物及起始点(MATLAB)

一、算法介绍 &#xff08;1&#xff09;白鲸优化算法BWO 参考文献&#xff1a;Zhong C, Li G, Meng Z. Beluga whale optimization: A novel nature-inspired metaheuristic algorithm[J]. Knowledge-Based Systems, 2022, 109215. &#xff08;2&#xff09;龙格-库塔优化…

基于Springboot的校园招聘系统(有报告)。Javaee项目,springboot项目。

演示视频&#xff1a; 基于Springboot的校园招聘系统&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&…

6层板学习笔记1

说明:笔记基于6层全志H3消费电子0.65MM间距BGA 目的:掌握各类接口的布局思路和布线,掌握DDR高速存储设计 1、网表的导入是原理图的元件电气连接关系,位号,封装,名称等参数信息的总和 2、原理图文件包含(历史版本记录,功能总框图,电源树,GPIO分配,DDR功能,CPU,US…

渐进淡出背景个人导航页源码(火影版)

渐进淡出背景个人导航页源码&#xff08;火影版&#xff09; 效果图部分源码领取源码下期更新预报 效果图 部分源码 <!DOCTYPE html> <html> <head> <!--小K网 www.xkwo.com --><meta charset"UTF-8"><title>火影版个人主页<…