Linux系统编程(七)网络编程TCP、UDP

本文目录

  • 一、基础知识点
    • 1. IP地址
    • 2. 端口
    • 3. 域名
    • 4. 网络协议类型
    • 5. IP协议类型
    • 6. 字节序
    • 7. socket套接字
  • 二、常用API
    • 1. socket套接字描述符
    • 2. bind套接字绑定
    • 3. listen设置客户端连接个数
    • 4. accept接收客户端请求
    • 5. connect连接服务端
  • 三、编程流程
    • 1.TCP编程

  
在学习本章之前,可以查看 《计算机网络基础》这篇文章进行学习。

一、基础知识点

1. IP地址

   IP 地址的作用是标识计算机的网卡地址,每一台计算机都有一个 IP 地址。在程序中是通过IP 地址来访问一台计算机的。 本节将讲述 IP 地址的一些知识。 IP 地址是用来标识全球计算机地址的一种符号,就比如一个手机的号码,使用这个地址可以访问一个计算机。
   IP 地址具有统一的格式。 IP 地址是 32 位长度的二进制数值, 存储空间是 4 个字节。例如: 11000000 10101000 00000001 00000110 是一台计算机的 IP 地址,但二进制的数值是不便于记忆的,可以把每个字节用一个十进制的整数来表示,既 192.168.1.6
   在同一个网络中, IP 地址是唯一的。因为需要根据 IP 地址来访问一台计算机,所以在可以访问的范围以内,每一台计算机的 IP 地址是唯一的。在终端中输入命令 ifconfig 可以查看本机 IP 信息。

2. 端口

   在网络通信中,IP地址帮助我们定位到特定的计算机,而端口号则帮助我们在这台计算机上找到具体的应用程序或服务。每个应用程序监听不同的端口,以便接收和处理来自网络的请求。端口号通常以数字形式表示,并与IP地址一起使用,以标识网络通信的不同端点。
   通俗举例:在同一个主机上,IP地址是固定的,但是主机上有很多程序功能,这些功能有不同的端口号。我们为了访问主机的某个功能程序,就必须指定其功能对应的端口号。例如主机的IP为:192.168.12.4,QQ的端口号为10,微信的端口号为13。我们为了使用微信这个功能,我们就需要使用:192.168.12.4: 13来使用其功能。

3. 域名

   域名是互联网中用于标识一个网站或服务器的友好名称,它是IP地址的可读形式。域名系统(DNS)将域名转换为IP地址,使用户能够通过简单易记的名字访问网站,而不需要记住难记的IP地址。例如百度的域名为www.baidu.com,我们可以使用这个域名来代替百度的IP地址。小知识:可以使用 ping 命令来查看一个域名所对应的 IP 地址,例:ping www.baidu.com,我们就可以查看其真实的ip地址。

4. 网络协议类型

(1)TCP协议:提供可靠的、面向连接的数据传输,确保数据按顺序传递、不丢失、不重复,并进行错误检测和纠正。常用于需要可靠数据传输的应用,如Web、电子邮件和文件传输。
   TCP 是面向连接的协议。所谓连接,就是两个对等实体为进行数据通信而进行的一种结合。面向连接服务是在数据交换之前,必须先建立连接。当数据交换结束后,则应终止这个连接。面向连接服务具有:连接建立、数据传输和连接释放这三个阶段。在传送数据时是按序传送的。建立链接:三次握手(这个过程我们并不看得到)。
在这里插入图片描述

(2)UDP协议:提供不可靠的、面向无连接的数据传输,数据包可能会丢失、重复或乱序,不进行错误检测和纠正。传输速度比TCP快!常用于实时性要求高、数据量小、丢失少不影响的应用,如视频流、音频流和游戏通信。

5. IP协议类型

  IPv4和IPv6是两种不同版本的互联网协议(IP),用于标识网络设备的地址并进行数据包路由。它们的主要区别在于地址格式和容量。

(1)IPV4:是第四版互联网协议,目前仍是最广泛使用的IP协议。

●它使用32位地址空间,格式如下:
  地址格式:四个十进制数,每个数范围从0到255,由点分隔。例如:192.168.1.1。
  地址数量:IPv4的地址空间约有42亿个唯一地址(2^32)。
●IPv4的特点
  地址空间有限:由于地址空间较小,IPv4地址逐渐耗尽,尤其随着互联网设备数量的爆炸性增长。
  地址分配:IPv4地址的分配方式包括公共地址和私有地址,私有地址用于局域网内通信,不可在互联网中直接使用(如192.168.x.x)。
  NAT(网络地址转换):由于地址空间有限,NAT技术被广泛应用,使多个设备可以共享一个公共IP地址访问互联网。

(2)IPV6:是第六版互联网协议,设计为IPv4的继任者,提供更大的地址空间和其他改进。

● 它使用128位地址空间,格式如下:
   地址格式:八组十六进制数,每组四个十六进制数字,由冒号分隔。例如:2001:0db8:85a3:0000:0000:8a2e:0370:7334
  地址数量:IPv6的地址空间极其庞大,有约3.4×1038个地址(2128)。
●IPv6的特点
  几乎无限的地址空间:IPv6提供了极其庞大的地址空间,足以满足未来互联网设备的需求。
  简化的地址配置:支持无状态地址自动配置(SLAAC),设备可以自动生成IPv6地址,不需要手动配置或DHCP。
  内置安全性:IPv6支持IPSec协议,提供更好的安全性。
  无NAT需求:由于地址空间充足,不再需要NAT,大大简化了网络结构和提高了传输效率。
  改进的路由和网络配置:IPv6改进了路由选择和地址分配机制,简化了网络配置和管理。

6. 字节序

顾名思义就是字节的存放顺序,就是字节数大于1的数据在内存中的存放顺序,字节数为1的数据就没有顺序的问题。
(1)主机字节序
  数据在主机(计算机)内部的字节顺序。在计算机系统中,数据存储在内存中,字节的存储顺序取决于计算机的体系结构。
  ●有两种主要的字节序:大端序小端序。在大端序中,高位字节存储在低地址,低位字节存储在高地址。在小端序中,低位字节存储在低地址,高位字节存储在高地址。

(2)网络字节序
  是一种特定的字节序,用于在网络中传输数据。在网络通信中,数据在传输过程中需要保持一致的字节序,以确保不同主机之间能够正确解释和处理数据。网络字节序规定使用大端序作为标准字节序。

  为了确保在网络通信中使用统一的字节序,常见的网络编程库提供了一系列函数来进行字节序转换,以确保数据在传输过程中采用网络字节序。

htons() //将16位整数从主机字节序转换为网络字节序(Host to Network Short)。
htonl() //将32位整数从主机字节序转换为网络字节序(Host to Network Long)。
ntohs() //将16位整数从网络字节序转换为主机字节序(Network to Host Short)。
ntohl() //将32位整数从网络字节序转换为主机字节序(Network to Host Long)。

inet_ntoa() // 是一个用于将网络字节序的二进制 IPv4 地址转换为点分十进制字符串表示形式的函数。
inet_addr() //函数用于将点分十进制字符串形式的 IPv4 地址转换为网络字节序的二进制形式。

7. socket套接字

(1)定义
   Socket(套接字)是在网络编程中用于实现不同主机间通信的一种机制,它允许应用程序通过网络发送和接收数据。
   Socket的工作原理基于客户端-服务器模型,其中一个程序充当客户端,另一个充当服务器。通常,服务器在一个主机上运行并侦听特定的端口,客户端则连接到服务器的IP地址和端口号。

(2)类型
   ①流式 socket(SOCK_STREAM):提供可靠的、面向连接的通信流;它使用 TCP 协议,从而保证了数据传输的正确性和顺序性。
   ②数据报 socket(SOCK_DGRAM):定义了一种无连接的服务,数据通过相互独立的报文进行传输,是无序的,并且不保证是可靠、无差错的。它使用数据报协议 UDP
   ③原始 socket:原始套接字允许对底层协议如 IP 或 ICMP 进行直接访问,它功能强大但使用较为不便,不常用。

二、常用API

●头文件

#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <strings.h>
#include <arpa/inet.h>

1. socket套接字描述符

返回值:套接字描述符

int socket(int domain, int type, int protocol)
//int domain ::代表一个协议族。AF_INET 决定了要用 ipv4 地址;
//int type:指定IP协议类型,常见的是TCP: SOCK_STREAM。 UDP:SOCK_DGRAM等。
//int protocol:当为 0 时,会自动选择 type 类型对应的默认协议。

2. bind套接字绑定

   将套接字绑定到指定的本地地址上,这样就可以在该地址上进行数据收发操作。通常,在服务器程序中,使用bind()函数将服务器套接字绑定到服务器的IP地址和端口号上,以便客户端可以连接到该地址并与服务器通信。

int bind(int sockfd, struct sockaddr* my_addr, int addrlen);
//int sockfd:socket套接字描述符
//struct sockaddr* my_addr :本地ip地址结构体,包含ip地址、端口等。
// int addrlen: struct sockaddr 结构的大小。

使用举例:

//下述结构体已经在#include <sys/socket.h>头文件中定义,不需要程序员自己定义。
/*
	struct sockaddr_in {
	 sa_family_t sin_family;     // address family: IP协议类型:AF_INET(IPV4)、AF_INET6(IPV6)。
	 in_port_t sin_port;         // port in network byte order 端口号 
	 struct in_addr sin_addr;    //internet address IP 地址 
	};
	
	struct in_addr {
		in_addr_t s_addr;  //ip地址
*/
//填写绑定的结构体内容	
sockaddr_in server_info;    //初始化结构体
server_info.sin_family =AF_INET;    //IPV4类型
server_info.sin_port=htons(50000);  //htons()将一个主机字节序端口号转为网络字节序下的端口号。
server_info.sin_addr.s_addr=htonl(INADDR_ANY); //所有人都可以连接。

bind(sockfd, (struct sockaddr*)&server_info, sizeof(server_info));  //绑定ip以及端口号

3. listen设置客户端连接个数

   设置客户端连接个数(最大排队长度):即多个客户端连接服务端时,当服务端没有接收这个请求,则客户端会进入排队等待中,那么这个排队长度就是最大可以容忍排队客户端的数量。

int listen(int sockfd, int backlog);
//int sockfd:socket套接字描述符。
// int backlog:设置请求排队的最大长度,当有多个客户端程序和服务端相连时, 使用这个表示可以介绍的排队长度。

4. accept接收客户端请求

监听是否有客户端链接,调用 accpet 之后会一直阻塞,直到有用户连接。
返回值:客户端的文件描述符。

int accept(int sockfd, void *addr, int *addrlen);
//int sockfd:socket套接字描述符。
//void *addr :用于存放客户端的信息
//int *addrlen :接收客户端的信息长度。

举例:

sockaddr_in client_info;    //用于存储连接的客户端的信息
int client_socket;

client_socket=accept(sockfd, (struct sockaddr*)&client_info, &(sizeof(client_info)));
printf("客户端的socket:%d, ip:%s, 端口:%d\n",client_socket, client_info.sin_addr, client_info.sin_port);

5. connect连接服务端

用于在客户端与服务器进行端连接。0 :成功。1 :失败。

int connect(int sockfd, struct sockaddr * serv_addr,int addrlen);
//int sockfd:socket()返回的文件描述符。
//struct sockaddr * serv_addr:服务端的地址,通常传递 struct sockaddr_in 结构体类型。
//int addrlen) :serv_addr长度。

三、编程流程

1.TCP编程

在这里插入图片描述

使用例程:

●TCP服务端:
   服务端是固定的,客户端是改变的。只能客户端去连接服务端,不能服务端连接客户端。正常先断开客服端,再断开服务端。

#include <sys/types.h>        
#include <sys/socket.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <arpa/inet.h>

//-----------------------------------------------服务端---------------------------
int main(int argc, char **argv)
{	  
	int ret;
	int client_sock;  
	int  server_sock;
	char receive_data[100]={0};
	struct sockaddr_in client_info;
	struct sockaddr_in server_info;
	socklen_t length;
	
	//1.创建socket,ip协议类型,网络协议类型,具体协议
	//SOCK_STREAM(流式): 使用TCP协议,保证数据传输的正确性和顺序性;
	//SOCK_DGRAM(数据报):使用UDP协议,不保证数据传输的正确性和顺序性。(速度快)	 
	server_sock=socket(AF_INET, SOCK_STREAM, 0);
	if(server_sock < 0){
		perror("socket error");
		return -1;
	}
	
	//2.设置服务端 
	server_info.sin_family = AF_INET;                 //ip协议类型为 ipv4
	server_info.sin_port   = htons(8888);            //端口号为8888。将小端转为大端。
	server_info.sin_addr.s_addr=htonl(INADDR_ANY);   //接收所有人的连接,程序在哪作为服务端运行,就填该地的ip地址。
	
	//3.绑定端口:绑定到指定的socket,绑定的信息,信息的长度
	ret= bind(server_sock, (const struct sockaddr *)&server_info,sizeof(server_info));
	if(ret < 0){
		perror("bind error");
		return -1;
	}
	
	//4.开启监听:监听的socket,最大排队数(当1个客户端申请连接,但服务端还没有同意连接请求时,客户端会等待。最多有10个可以等待排队的客户端)
	ret=listen(server_sock, 10);
	if(ret < 0){
		perror("listen error");
		return -1;
	}
	//5. 接收客户端连接。
	length=sizeof(client_info);
	client_sock= accept(server_sock, (struct sockaddr *)&client_info,  &length); //将连接的客户端的信息存到client_info中。	
	printf("客户端的socket:%d, ip:%s, 端口:%d\n",client_sock, inet_ntoa(client_info.sin_addr), client_info.sin_port);
	
	//6. 服务端向指定的客户端发送数据
	write(client_sock, "hello world",strlen("hello world"));
	//7. 接收指定的客户端发送的数据。读不到会阻塞。
	read(client_sock, receive_data,100);
	printf("Client:%d,Send_data:%s\n",client_sock, receive_data);
	//7. 关闭描述符
	close(client_sock);                //先关闭客户端描述符
	close(server_sock);                //再关闭服务端描述符
}

在调试时:服务端在哪运行就填哪的IP地址。
在这里插入图片描述

●TCP客户端:

#include <sys/types.h>        
#include <sys/socket.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <arpa/inet.h>
#include <pthread.h>

int main(int argc, char **argv)
{
	int client_sock;
	int ret;
	struct sockaddr_in server_info;
	char receive_data[100]={0};
	
	// 1.创建socket,ip协议类型,网络协议类型,具体协议
	//SOCK_STREAM(流式): 使用TCP协议,保证数据传输的正确性和顺序性;
	//SOCK_DGRAM(数据报):使用UDP协议,不保证数据传输的正确性和顺序性。(速度快)
	client_sock=socket(AF_INET, SOCK_STREAM, 0);
	if(client_sock< 0){
		perror("socket error");
		return -1;
	}	  
	
	// 2.设置服务端的信息	  
	server_info.sin_family = AF_INET;                           //ip协议类型为 ipv4
	server_info.sin_port   = htons(8888);                      //服务端端口号为40000。将小端转为大端
	server_info.sin_addr.s_addr= inet_addr("192.168.195.15");     //写入服务端的ip地址
	
	// 3.连接服务端	  
	ret=connect(client_sock,(const struct sockaddr *)&server_info, sizeof(server_info));
	if(ret < 0){
		perror("connect error");
		return -1;
	}
	printf("connect ok!\n");
	
	//4. 向服务器发送数据
	write(client_sock, "i am client_one", strlen("i am client_one"));
	//5. 接收服务器发送的数据,读不到会阻塞。
	read(client_sock, receive_data, 100);  
	printf("receive_data:%s\n",receive_data);
	//6. 创建接收服务端数据的线程,实现客户端与服务端互相收发
	close(client_sock);                //关闭服务端通信句柄
	return 0;
}

我们使用两个命令窗口,先执行服务端的代码,再执行客户端的代码。
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

sqoop操作

介绍 sqoop是隶属于Apache旗下的, 最早是属于cloudera公司的,是一个用户进行数据的导入导出的工具, 主要是将关系型的数据库(MySQL, oracle...)导入到hadoop生态圈(HDFS,HIVE,Hbase...) , 以及将hadoop生态圈数据导出到关系型数据库中 操作 将数据从mysql中导入到HDFS中 1.全量…

[AI Google] Google I/O 2024: 为新一代设计的 I/O

编辑注&#xff1a;以下是 Sundar Pichai 在 I/O 2024 上讲话的编辑版&#xff0c;并包含了更多在舞台上宣布的内容。查看我们收藏中的所有公告。 Google 完全进入了我们的 Gemini 时代。 在开始之前&#xff0c;我想反思一下我们所处的这一刻。我们已经在 AI 上投资了十多年…

【LeetCode 101】对称二叉树

1. 题目 2. 分析 这道题比较经典。我又一次做错了&#xff0c;这次是花了20min都没有做出来。 最开始我的思想就是&#xff0c;递归比较左根节点的左子树和右根节点的右子树是否对称即可&#xff0c;然后觉得能解决问题了&#xff0c;便动手coding。哪知道&#xff0c;又碰到了…

23.Labview中的数值类型讨论 ---- 位(bit)、字节(byte)、I8、U8、单双精度、复数

hello&#xff0c;大家好&#xff0c;本篇向大家介绍一个最常用但最容易让人忽略和最容易犯错的知识&#xff1a;数值。 “数值” 这个概念在Labview中被涉及的还是很多的&#xff0c;几乎任何一个程序都无可避免的会用到&#xff0c;但我相信大家绝大多数人对数值这个概念应用…

CentOS8安装opensips 3.5

环境&#xff1a;阿里云 操作系统CentOS8.5 依赖包安装&#xff1a; libmicrohttpd cd /usr/local/src wget https://ftp.gnu.org/gnu/libmicrohttpd/libmicrohttpd-latest.tar.gz tar vzxf libmicrohttpd-latest.tar.gz cd libmicrohttpd-1.0.1/./configure make make …

【CVPR_2024】:逐元素乘积为什么会产生如此令人满意的结果?

写在前面&#xff1a;本博客仅作记录学习之用&#xff0c;部分图片来自网络&#xff0c;如需引用请注明出处&#xff0c;同时如有侵犯您的权益&#xff0c;请联系删除&#xff01; 文章目录 前言论文重写星形运算一层网络推广多层网络特殊情况 W 1 W_1 W1​和/或 W 2 W_2 W2​…

Python-3.12.0文档解读-内置函数sorted()详细说明+记忆策略+常用场景+巧妙用法+综合技巧

一个认为一切根源都是“自己不够强”的INTJ 个人主页&#xff1a;用哲学编程-CSDN博客专栏&#xff1a;每日一题——举一反三Python编程学习Python内置函数 Python-3.12.0文档解读 目录 Python-3.12.0文档解读详细说明 功能描述 参数说明 用法示例 备注 进阶用法 参考…

集合操作进阶:关于移除列表元素的那点事

介绍 日常开发中&#xff0c;难免会对集合中的元素进行移除操作&#xff0c;如果对这方面不熟悉的话&#xff0c;就可能遇到 ConcurrentModificationException&#xff0c;那么&#xff0c;如何优雅地进行元素删除&#xff1f;以及其它方式为什么不行&#xff1f; 数据初始化…

力扣--双指针15.三数之和

详细思路 排序数组&#xff1a;首先对数组 nums 进行排序&#xff0c;目的是为了方便后续使用双指针查找和避免重复结果。遍历数组&#xff1a;使用一个 for 循环从头遍历到倒数第三个元素。i 表示当前固定的元素。 跳过重复元素&#xff1a;如果当前元素 nums[i] 与前一个元素…

使用matplotlib绘制折线条形复合图

使用matplotlib绘制折线条形复合图 介绍效果代码 介绍 在数据可视化中&#xff0c;复合图形是一种非常有用的工具&#xff0c;可以同时显示多种数据类型的关系。在本篇博客中&#xff0c;我们将探讨如何使用 matplotlib 库来绘制包含折线图和条形图的复合图。 效果 代码 imp…

登录安全分析报告:小米官网注册

前言 由于网站注册入口容易被黑客攻击&#xff0c;存在如下安全问题&#xff1a; 暴力破解密码&#xff0c;造成用户信息泄露短信盗刷的安全问题&#xff0c;影响业务及导致用户投诉带来经济损失&#xff0c;尤其是后付费客户&#xff0c;风险巨大&#xff0c;造成亏损无底洞 …

【算法】模拟算法——数青蛙(medium)

题解&#xff1a;模拟算法——数青蛙(medium) 目录 1.题目2.题解3.参考代码4.总结 1.题目 题目链接&#xff1a;LINK 2.题解 用循环进行遍历&#xff0c; 如果该字符为o\o\a\k 找一下前驱字符是否存在 如果存在&#xff0c;前驱字符–&#xff0c;该字符如果不存在&#x…

STM32_IIC

1、IIC简介 I2C&#xff0c;即Inter IC Bus。是由Philips公司开发的一种串行通用数据总线&#xff0c;主要用于近距离、低速的芯片之间的通信&#xff1b;有两根通信线&#xff1a;SCL&#xff08;Serial Clock&#xff09;用于通信双方时钟的同步、SDA&#xff08;Serial Data…

echarts渐变色与css渐变色互转(两个坐标点转角度)

前言 用于 echarts 的小伙伴都知道&#xff0c;他使用的渐变色写法和 css 的写法不一样。css 中直接使用角度定义渐变的方向&#xff0c;而 echarts 使用的是两个坐标点来进行标识方向&#xff08;线性渐变&#xff09;。 本文主要针对线性渐变的转换 那怎么在 css 中使用 e…

BrainGPT1,一个帮你b站点歌放视频的多模态多轮对话模型

BrainGPT1&#xff0c;一个帮你b站点歌放视频的多模态多轮对话模型 返回论文目录 项目地址 模型地址 作者&#xff1a;华东师范大学&#xff0c;计算机科学与技术学院&#xff0c;智能教育研究院的小怪兽会微笑。 介绍 BrainGPT1是一个工具调用多轮对话模型&#xff0c;与G…

[机器学习]GPT LoRA 大模型微调,生成猫耳娘

往期热门专栏回顾 专栏描述Java项目实战介绍Java组件安装、使用&#xff1b;手写框架等Aws服务器实战Aws Linux服务器上操作nginx、git、JDK、VueJava微服务实战Java 微服务实战&#xff0c;Spring Cloud Netflix套件、Spring Cloud Alibaba套件、Seata、gateway、shadingjdbc…

BU01板卡引脚

概述 BU01 是一款高速采集卡&#xff0c;主要用于高带宽数据采集及传输&#xff0c;应用领域多为数据中 心及数据采集领域。 端口提供60Gbps 传输带宽&#xff0c;可兼容2 个SFP万兆网口&#xff0c;和1 个40GE QSFP 光 口。和主机通信采用的是PCIE 2.0 x8 模式&#xff0c;最…

C++哈希的应用:位图 布隆过滤器 哈希切割

目录 位图 bitset 构造空间 将某个位变为0 将某个位变为1 检查是否存在 完整代码 拓展问题一 ​编辑 拓展问题二 布隆过滤器 判断是否存在 使用场景 哈希切割 拓展问题一 拓展问题二 位图 问题&#xff1a;有四十个亿未排序的不重复的无符号整数&#xff0c;此…

算法导论 总结索引 | 第三部分 第十四章:数据结构的扩张

1、通过存储 额外信息的方法来扩张一 种标准的数据结构&#xff0c;然后对这种数据结构&#xff0c;编写新的操作来支持所需的应用。因为添加的信息 必须要能被该数据结构上的常规操作更新和维护 2、通过扩张红黑树构造出的两种数据结构&#xff1a;14.1介绍 一种支持一般动态…

对boot项目拆分成cloud项目的笔记

引言&#xff1a;这里我用的是新版本的技术栈 spring-boot-starter-parent >3.2.5 mybatis-spring-boot-starter >3.0.3 mybatis-plus-boot-starter >3.5.5 spring-cloud-dependencies …