<计算机网络自顶向下> TCPUDP套接字编程

news2024/11/17 0:04:07

应用实现:源端的应用进程交换报文实现应用协议,来实现各种各样的网络应用(dash,email,  etc)

而应用层通信不可以直接通信,需要借助下层的服务才可以进行,通过层间接口交给下层,通过下层的服务传输

传输层在TCP/IP提供的就是socket API服务,传输报文之前建立socket,借助于socket收发,使用完成了以后关掉(socket就像一个门,接和收都一样)

两种socket类型:

  • TCP:可靠的,字节流/管道(报文之间没有边界)服务
  • UDP:不可靠(数据UDP数据包)服务

TCP套接字编程 

  • 服务器首先运行,等待连接建立

1. 创建欢迎socket:目的是返回一个整数(这个整数就是welcome soket,但没什么具体含义)

2. 将这个创建的整数和本地端口捆绑

3. 在欢迎socket上阻塞式等待接受用户的连接 (调用accept函数,如果没连接就阻塞,反之接受连接)

总的来说:创建——捆绑——等待

  •  客户端主动和服务器建立连接

    1. 创建客户端本地套接字(隐式捆绑到本地端口,就是默认与当前没用的端口捆绑)

    2. 调用connect阻塞连接指定服务器进程的IP地址和端口号

  •  当与客户端连接请求到来时
    1. 服务器接受来自用户端的请求,解除阻塞式等待,返回一个新的sockket值(同时与服务端和客户端捆绑,这个是connection socket)
  •  连接API调用有效时,客户端与服务器建立了TCP链接

  数据结构 sockaddr_in

  • 作用:指定网络操作的目标地址(设置网络进程和端口号)
  • 不仅仅可以用于IP通信,还可以用于IPX通信。所以要给一个常量说明使用在哪个通信的
  • IP地址和port捆绑关系的数据结构(标示进程的端节点)
  • 这个结构体用于存储 IP 地址。它包含了一个地址族(通常是 AF_INET,表示 IPv4),一个端口号和一个 IP 地址。当你创建一个套接字并想要连接到一个特定的 IP 地址和端口号时,你需要设置一个 struct sockaddr_in 结构体,并将它传递给 connect 或 bind 函数
//IP地址和port捆绑关系的数据结构(标示进程的端节点)
struct sockaddr_in{
 short sin_family;    //AF_INET 
 u_short sin_port;   // port
 struct in_addrsin_addr; 
//  IP address, unsigned long
 char sin_zero[8]; // align
 }; 

数据结构 host_ent

  • 作用:获取主机信息(IP地址)
  • 更清晰的表述:struct hostent:这个结构体用于存储主机的信息,如主机名、别名、地址类型、地址长度和地址列表。当你调用 gethostbyname 函数时,它会返回一个指向 struct hostent 类型的指针。这个结构体包含了你所查询的主机的信息。例如,如果你查询的是一个域名,那么 gethostbyname 函数会返回这个域名对应的 IP 地址
  • 包括
    • *h_name:主机域名的指针
    • **h_aliases:主机的一系列别名,二级指针
    • h_length:地址长度
    • 第四个是IP地址的列表,二级指针
      //域名和IP地址的数据结构
      struct hostent
       { char *h_name; 
      char **h_aliases; 
      int h_addrtype; 
      int h_length; /*地址长度*/ 
      char **h_addr_list; 
      #define h_addr h_addr_list[0];
       } 
      //作为调用域名解析函数时的参数
      //返回后,将IP地址拷贝到sockaddr_in的IP地址部分

C/S socket交互:TCP

代码示例

  • 客户端

也就是说这个sockaddr实际存储的是即将连接的服务器信息以及联机方式,而gethostbyname实际上是提供映射函数,只要输出服务器端口号就可以得到一个hostent指针然后用箭头函数得出需要的信息比如IP地址等等然后copy到socketaddr。

注意!这里socket是隐式的绑定了本地可用端口,没有显示调用bind函数

/*client.c*/ 
void main(int argc, char *argv[])
 { 
struct sockaddr_in sad; /* structure to hold an IP address of server */ 
    //定义一个套接字,用于与服务器通信
int clientSocket; /* socket descriptor */ 
    //定义一个主机信息结构体指针,用于存储主机的信息
struct hostent* ptrh; /* pointer to a host table entry  */ 
    //定义一个字符数组,用于存储要发送给服务器的消息
char Sentence[128]; 
    //定义一个字符数组,用于存储从服务器接收的消息
char modifiedSentence[128]; 
    // argv[0]是执行的程序的名字; atoi转化为整型
host = argv[1]; port = atoi(argv[2]);
    /*创建socket,参数指明internet地址族,PF_INET:这是第一个参数,表示协议族(Protocol             
    Family)或地址族(Address Family)。PF_INET 表示使用 IPv4 网络协议。
    SOCK_STREAM:这是第二个参数,表示套接字类型(Socket Type)。SOCK_STREAM 表示使用面向连接的 
    TCP 协议。
    0:这是第三个参数,表示使用默认的协议。对于 SOCK_STREAM 类型的套接字,默认的协议就是 TCP。*/
clientSocket= socket(PF_INET, SOCK_STREAM, 0); 
//从下一行开始,所有缩进的代码都是为了创建连接socket然后和服务器连接
        //之前有一个sad变量,sad是个结构体变量,分配给内存,先清零
        //这是因为新声明的变量在内存中可能会有一些旧的、无用的数据,这可能会干扰我们后续的操作
    memset((char *)&sad,0,sizeof(sad)); /* clear sockaddr structure */ 
        //对sad赋值Internet地址族
    sad.sin_family= AF_INET; /* set family to Internet */ 
        //将port强制转化为无符号短整型
        /*htons 是一个在网络编程中常用的函数,全称是 “host to network short”。这个函数的作用是                        
        将一个16位的数从主机字节序转换为网络字节序。

        在网络通信中,为了保证数据的正确传输,我们需要统一数据的字节顺序。网络字节序通常是大端字节                
        序(Big-Endian),也就是最重要的字节(Most Significant Byte,MSB)存储在内存的最低地址        
        处。

        主机字节序则取决于你的机器,有的机器是大端字节序,有的机器是小端字节序(Least         
        Significant Byte,LSB,存储在内存的最低地址处)。因此,在发送数据之前,我们通常会使用         
        htons 函数将数据从主机字节序转换为网络字节序。

        例如,如果你的机器是小端字节序,那么一个16位的数 0x1234 在内存中的存储顺序是 34 12。使用 
        htons 函数后,这个数在内存中的存储顺序就会变为 12 34,符合网络字节序。

        总的来说,htons 函数的作用就是确保数据在网络上传输时,不同的机器能够正确地解析数据*/
    sad.sin_port= htons((u_short)port);
        /*host主机域名。ptrh为结构体指针,gethostbyname就是调用解析器 
        获取主机的信息,包括主机的IP地址
        在这段代码中,ptrh 是一个指向 struct hostent 类型的指针。struct hostent 是一个结构            
        体,它包含了主机的信息,如主机名、别名、地址类型、地址长度和地址列表*/
    ptrh= gethostbyname(host); 
        /* 将主机的IP地址复制到服务器地址结构体的 sin_addr 字段 */
    memcpy(&sad.sin_addr, ptrh->h_addr, ptrh->h_length); 
        //将IP地址拷贝到sad.sin_addr
    connect(clientSocket, (structsockaddr*)&sad, sizeof(sad));
  //从用户处得到输入流
gets(Sentence);
  //发送给服务器用户先前输入的东西
n=write(clientSocket, Sentence, strlen(Sentence)+1); 
  //读取服务器端发过来的转换完毕的字符
n=read(clientSocket, modifiedSentence, sizeof(modifiedSentence));
printf("FROM SERVER: %s\n",modifiedSentence); 
  //关闭连接
close(clientSocket); 
}
  • 服务器

/* server.c  */
// argc参数数量, argv里面的内容是服务器端守候的端口号,只有一个参数
 void main(int argc, char *argv[])
 { 
//sad是作为服务器本地的端节点结构体变量,cad放的是client的端节点的结构体变量
struct sockaddr_in sad; /* structure to hold an IP address  of server*/ 
struct sockaddr_in cad; /*client  */
//welcome socket仅仅有自己的端口号,和对方的无关
int welcomeSocket, connectionSocket; /* socket descriptor */ 
struct hostent *ptrh; /* pointer to a host table entry */ 
//存储客户端发来的信息
char clientSentence[128]; 
//处理客户端发过来的东西后的结果
char capitalizedSentence[128]; 
//端口号
port = atoi(argv[1]);
//建立并bind welcome_socket
welcomeSocket = socket(PF_INET, SOCK_STREAM, 0);   
    memset((char *)&sad,0,sizeof(sad)); /* clear sockaddr structure */ 
    sad.sin_family = AF_INET; /* set family to Internet */ 
        //INADDR_ANY代表本地任何一个IP地址
    sad.sin_addr.s_addr = INADDR_ANY; /* set the local IP address */ 
        //格式转化(同客户端)
    sad.sin_port = htons((u_short)port);/* set the port number */ 
bind(welcomeSocket, (struct sockaddr *)&sad, sizeof(sad));
/* Specify the maximum number of clients that can be queued */
 //在服务的时候又来了请求,然后这些请求放在这个长度为10的队列,如果超过队列就拒绝
 listen(welcomeSocket, 10)
  //服务器进入无限循环,不断接收和处理客户端的连接请求
 while(1) { 
     //等待用户连接建立请求
     connectionSocket=accept(welcomeSocket, (structsockaddr*)&cad, &alen);
     //读取并写入发过来的东西
     n=read(connectionSocket, clientSentence, sizeof(clientSentence));
     /* capitalize Sentence and store the result in capitalizedSentence*/
     n=write(connectionSocket, capitalizedSentence, strlen(capitalizedSentence)+1); 
     close(connectionSocket);  
    } 
}

UDP套接字编程  

  • 没有握手
  • UDP socket仅仅和本地的IP和端口相关,发送端需要在发送的时候显式的指定对方的IP地址和端口
  • 服务器必须从收到的分组中提取出发送端的IP地址和端口号
  • 传送的数据可能乱序,也可能丢失
  • 无连接的数据单元也叫数据报,所以没办法通过“数据报”这个名字判断TCP还是UDP
     

 C/S交互:UDP

  • recvfrom阻塞,知道对方发来消息

代码示例

  • 客户端

     /* client.c  */
     void main(int argc, char *argv[])
    { 
    struct sockaddr_in sad; /* structure to hold an IP address */ 
    int clientSocket; /* socket descriptor */ 
    struct hostent *ptrh; /* pointer to a host table entry */ 
    char Sentence[128]; 
    char modifiedSentence[128]; 
    host = argv[1]; port = atoi(argv[2]);
    clientSocket = socket(PF_INET, SOCK_DGRAM, 0); 
    /* determine the server's address */
    memset((char *)&sad,0,sizeof(sad)); /* clear sockaddr structure */ 
    sad.sin_family = AF_INET; /* set family to Internet */ 
    sad.sin_port = htons((u_short)port); 
    ptrh = gethostbyname(host);
     /* Convert host name to IP address */
    memcpy(&sad.sin_addr, ptrh->h_addr, ptrh->h_length);  
    gets(Sentence);
    addr_len =sizeof(struct sockaddr); 
    n=sendto(clientSocket, Sentence, strlen(Sentence)+1,
    (struct sockaddr *) &sad, addr_len); 
    n=recvfrom(clientSocket, modifiedSentence, sizeof(modifiedSentence),
    (struct sockaddr *) &sad, &addr_len);
    printf("FROM SERVER: %s\n",modifiedSentence); 
    close(clientSocket);  
    }
  • 服务器

     /* server.c  */
     void main(int argc, char *argv[])
     { 
    struct sockaddr_in sad; /* structure to hold an IP address */ 
    struct sockaddr_in cad;
    int serverSocket; /* socket descriptor */ 
    struct hostent *ptrh; /* pointer to a host table entry */ 
    char clientSentence[128]; 
    char capitalizedSentence[128]; 
    port = atoi(argv[1]);
    serverSocket = socket(PF_INET, SOCK_DGRAM, 0);   
    memset((char *)&sad,0,sizeof(sad)); /* clear sockaddr structure */ 
    sad.sin_family = AF_INET; /* set family to Internet */ 
    sad.sin_addr.s_addr = INADDR_ANY; /* set the local IP address */ 
    sad.sin_port = htons((u_short)port);/* set the port number */ 
    bind(serverSocket, (struct sockaddr *)&sad, sizeof(sad));
    while(1) { 
     n=recvfrom(serverSocket, clientSentence, sizeof(clientSentence), 0
     (struct sockaddr *) &cad, &addr_len );
     /* capitalize Sentence and store the result in capitalizedSentence*/
     n=sendto(serverSocket , capitalizedSentence, strlen(capitalizedSentence)+1,
     (struct sockaddr *) &cad, &addr_len); 
     } 
    }

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

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

相关文章

Linux(Ubuntu) 查看并删除使用【dpkg】安装的软件【mysql 8.3安装失败---原因调查(Depends: libc6 (>= 2.35) but 2.31-0ubuntu9.1)】

目录 ■前言 ■查看安装的软件 ■删除安装的软件 正常删除(dpkg -r xxxxName) 问题解决:use --purge to remove them too ■其他调查信息 命令 图片1 图片2 图片3 图片4 图片5(和镜像库有关) 图片6 ■前…

在MOS管栅极前加100Ω电阻,有啥妙用

我们经常会听到在MOSFET栅极前增加一个电阻。那么,为什么要增加这个电阻,进一步地来讲,为什么要增加一个100Ω电阻? 在MOSFET的栅极前增加一个电阻? MOS管是电压型控制器件,一般情况下MOS管的导通&#x…

【学习笔记十五】批次管理和容量管理

一、批次管理 1.配置 SAP EWM 特定参数 激活仓库的批次管理 2.ERP端物料需要启用批次管理 3.EWM物料需要启用批次管理 一般是ERP启用批次管理,相关的配置也会传输到EWM系统 4.建立批次主数据 5.创建采购订单并创建内向交货单,维护批次 6.维护产品主数…

Ubuntu 22.04 开机自动挂载webdav - 设置开机自启脚本 - 解决坚果云webdav无写入权限

效果图: 前言: 1)亲测/etc/fstab的办法没有成功自动挂载,换成传统的rc.local可以解决; 2)rc-local.service是系统自带的一个开机自启服务,但是在 ubuntu 20.04 上,该服务默认没有开…

NLP_知识图谱_图谱问答实战

文章目录 图谱问答NERac自动机实体链接实体消歧 多跳问答neo4j_graph执行流程结构图![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/1577c1d9c9e342b3acbf79824aae980f.png)company_data![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/20f567d877c743b…

深度学习图像处理基础工具——opencv 实战信用卡数字识别

任务 信用卡数字识别 穿插之前学的知识点 形态学操作 模板匹配 等 总体流程与方法 1.有一个模板 2 用轮廓检测把模板中数字拿出来 外接矩形(模板和输入图像的大小要一致 )3 一系列预处理操作 问题的解决思路 1.分析准备:准备模板&#…

libbpf-bootstrap库的代码结构介绍(用户层接口介绍),编译链接语句详细介绍,.skel.h文件介绍+示例,bpf程序的后续处理+文件关系总结

目录 libbpf-bootstrap 代码结构介绍 用户层函数 编译 查看 生成内核层的.o文件 第一模块 第二模块 第三模块 第四模块 第五模块 生成辅助文件(.skel.h) 介绍 示例 生成代码层的.o文件 第一模块 第二模块 第三模块 链接出可执行文件 后续总结 libbpf-bootst…

舒欣上门预约系统源码-按摩预约/家政预约全行业适用-小程序/h5/app

上门预约或者到店预约均可,家政,按摩,等等上门类行业均可适用。(后台的技师及前台技师这两个字是可以更改的,例如改成家政老师,保洁,等等) 视频教程是演示搭建的小程序端&#xff0c…

stm32报错问题集锦

PS:本文负责记录本人日常遇到的报错问题,以及问题描述、原因以及解决办法等,解决办法百分百亲测有效。本篇会不定期更新,更新频率就看遇到的问题多不多了 更换工程芯片型号 问题描述 例程最开始用的芯片型号是STM32F103VE&#…

Abstract Factory抽象工厂模式详解

模式定义 提供一个创建一系列相关或互相依赖对象的接口,而无需指定它们具体的类。 代码示例 public class AbstractFactoryTest {public static void main(String[] args) {IDatabaseUtils iDatabaseUtils new OracleDataBaseUtils();IConnection connection …

微信登录功能-保姆级教学

目录 一、使用组件 二、登录功能 2.1 步骤 2.2 首先找到网页权限 复制demo 代码 这里我们需要修改两个参数 三、前端代码 3.1 api 里weiXinApi.ts 3.2 api里的 index.ts 3.3 pinia.ts 3.4 My.vue 四、后端代码 4.1 WeiXinController 4.2 Access_Token.Java 4.3 We…

SHARE 203S PRO:倾斜摄影相机在地灾救援中的应用

在地质灾害的紧急关头,救援队伍面临的首要任务是迅速而准确地掌握灾区的地理信息。这时,倾斜摄影相机成为了救援测绘的利器。SHARE 203S PRO,这款由深圳赛尔智控科技有限公司研发的五镜头倾斜摄影相机,以其卓越的性能和功能&#…

OpenHarmony实战开发-FaultLoggerd组件。

简介 Faultloggerd部件是OpenHarmony中C/C运行时崩溃临时日志的生成及管理模块。面向基于 Rust 开发的部件,Faultloggerd 提供了Rust Panic故障日志生成能力。系统开发者可以在预设的路径下找到故障日志,定位相关问题。 架构 Native InnerKits 接口Sig…

新手教程 | 2024年最新Vmware17安装教程及许可证(详细图文)

目录 前言: 一、VMware Workstation 17 Pro 简介 二、下载安装(以Windows为例) 三、许可证 四、检查是否安装成功 前言: 重新装电脑后,安装虚拟机 一、VMware Workstation 17 Pro 简介 VMware Workstation 17 …

问题整理【2024-04-08】

一、关于MYSQL死锁问题 1.1 源由 一次上线过程中,遇到了MySQL死锁的问题…… 1.2 分析 1.2.1 前置知识 ​ 首先要知道,MySQL是一个多线程的数据库管理系统,查询是通常并发执行的,可以同时处理多个查询请求,并且My…

文献阅读:LESS: Selecting Influential Data for Targeted Instruction Tuning

文献阅读:LESS: Selecting Influential Data for Targeted Instruction Tuning 1. 文章简介2. 方法介绍 1. Overview2. 原理说明 1. SGD上的定义2. Adam上的定义 3. 具体实现 1. Overview1. LoRA使用2. 数据选择3. LESS-T 3. 实验考察 & 结论 1. 实验设计2. 主…

SPI 设备驱动编写流程:创建SPI节点以及SPI设备节点(在设备树文件中)

一. 简介 SPI 驱动框架和 I2C 很类似,都分为主机控制器驱动和设备驱动。 SPI主机控制器的驱动一般是芯片半导体厂商写好了,我们要编写的是SPI设备驱动代码。 本文开始来学习SPI设备驱动的编写流程(前提是支持设备树的情况)。 二…

Google最新论文: 复杂的 Prompt 如何更好的调试?

本文介绍了Sequence Salience,这是一个专为调试复杂的大模型提示而设计的系统。该系统利用广泛使用的显著性方法,支持文本分类和单标记预测,并将其扩展到可处理长文本的调试系统。现有的工具往往不足以处理长文本或复杂提示的调试需求。尽管存…

bugku-web-文件上传

提示他的名字是margin,给他一个图片文件,不要php文件 上传一句话木马的图片 抓包,后缀改为php 提示无效文件,即后台还会检测一次后缀 测试后台系统 为linux系统 开始绕过 截断绕过 上传成功,但是会变为jpg 开始分析…

黑马苍穹外卖--再来一单(stream流转换、赋值与收集映射)

1.首先明确一下业务规则: 业务规则: 再来一单就是将原订单中的商品重新加入到购物车中 2.产品页面原型和开发接口文档 3.业务层逻辑代码开发 3.1 查询方向 我们要明确的是: 再来一单就是将原订单中的商品重新加入到购物车中------直接把商品加入到购物车&#…