当前位置: 首页 > news >正文

Linux网络编程- 网络字节顺序

基本概念

网络字节顺序是一种规定的数据表示格式,被用于TCP/IP协议栈,特别是在网络传输数据时。它确保不同的计算机和架构之间可以无缝地通信。网络字节顺序是大端字节序(big-endian)。

字节序的背景

计算机存储多字节数据(例如32位整数、64位整数)时有两种主要方式:大端(big-endian)和小端(little-endian)。

  • 大端字节序 (Big-Endian):

    • 高位字节存储在内存的低地址。
    • 例如,数字 0x12345678 在内存中表示为 12 34 56 78
  • 小端字节序 (Little-Endian):

    • 低位字节存储在内存的低地址。
    • 例如,数字 0x12345678 在内存中表示为 78 56 34 12

不同的计算机架构可能使用不同的字节顺序,例如,x86架构使用小端字节序,而早期的Motorola芯片则使用大端字节序。

网络字节顺序的必要性

考虑到不同的计算机可能有不同的字节顺序,TCP/IP协议规定了一个统一的字节顺序来发送和接收数据,称为网络字节顺序。这样,不同的机器在交换数据时就知道如何解释收到的字节。

网络字节顺序是大端字节序。所以,不论系统使用哪种字节顺序,当数据被发送到网络时,它都被转换为大端字节序。接收数据时,数据被转换回主机的字节顺序。

工具函数

为了在网络字节顺序和主机字节顺序之间进行转换,提供了一系列的函数:

  • htonl(): 主机到网络长整数转换
  • htons(): 主机到网络短整数转换
  • ntohl(): 网络到主机长整数转换
  • ntohs(): 网络到主机短整数转换

在大端系统上,这些函数通常不做任何事情(因为主机字节顺序已经是大端),而在小端系统上,它们会反转字节顺序。

示例

如果在小端机器上有一个16位的整数 0x1234,并且想发送它到网络上,可以使用htons()函数。这会将数字转换为0x3412。另一台机器从网络上接收到这个数字后,可以使用ntohs()函数将它转换回其本地表示形式。

htonl、htons、ntohl、ntohs

这四个函数是用于在主机字节顺序和网络字节顺序之间进行转换的。这样不同的计算机架构可以在TCP/IP网络上无缝地通信。让我们详细地看一下每一个函数:

  1. htonl(): 主机到网络长整数转换

    • 描述: 这个函数将一个无符号长整数(通常是32位)从主机字节顺序转换为网络字节顺序。
    • 参数: 一个无符号长整数(如uint32_t)。
    • 返回值: 参数值转换为网络字节顺序的整数。
  2. htons(): 主机到网络短整数转换

    • 描述: 这个函数将一个无符号短整数(通常是16位)从主机字节顺序转换为网络字节顺序。
    • 参数: 一个无符号短整数(如uint16_t)。
    • 返回值: 参数值转换为网络字节顺序的整数。
  3. ntohl(): 网络到主机长整数转换

    • 描述: 这个函数将一个无符号长整数(通常是32位)从网络字节顺序转换为主机字节顺序。
    • 参数: 一个无符号长整数(如uint32_t)。
    • 返回值: 参数值转换为主机字节顺序的整数。
  4. ntohs(): 网络到主机短整数转换

    • 描述: 这个函数将一个无符号短整数(通常是16位)从网络字节顺序转换为主机字节顺序。
    • 参数: 一个无符号短整数(如uint16_t)。
    • 返回值: 参数值转换为主机字节顺序的整数。

使用场景

当您在网络上发送或接收数据时,使用这些函数非常重要,尤其是当您处理多字节字段(如IP地址、TCP/UDP端口)时。

例如,当您设置一个sockaddr_in结构的sin_port字段时,您需要使用htons()来确保端口号以网络字节顺序存储。类似地,当您从网络上接收数据并想查看其中的端口号或IP地址时,您需要使用相应的ntohs()ntohl()函数。

注意

  1. "长"和"短"的术语是相对的。在这里,"短"通常指的是16位整数,而"长"通常指的是32位整数。但请注意,这可能会根据平台和编译器实现而有所不同。
  2. 在大端系统上,这些函数可能不执行任何实际操作,因为这些系统的主机字节顺序已经是网络字节顺序。但为了代码的可移植性和清晰性,仍然建议使用它们。

示例:主机和网络字节顺序的转换

通过下面的例子,我们来演示如何使用htons()ntohs()函数。

#include <stdio.h>
#include <arpa/inet.h>  // for htons, ntohsint main() {unsigned short port_host_order = 8080;  // 0x1F90 in hexadecimalunsigned short port_network_order;// Convert port number from host byte order to network byte orderport_network_order = htons(port_host_order);printf("Port in host byte order: 0x%X\n", port_host_order);printf("Port in network byte order: 0x%X\n", port_network_order);// Now, let's pretend we received this port number from the network// and we want to convert it back to host byte orderunsigned short received_port_host_order = ntohs(port_network_order);printf("Received port converted back to host byte order: 0x%X\n", received_port_host_order);return 0;
}

这个程序首先定义了一个端口号8080(它在十六进制中是0x1F90)。它使用htons()函数将端口号从主机字节顺序转换为网络字节顺序,并打印出结果。

然后,该程序模拟从网络上接收到这个端口号的情况,并使用ntohs()函数将它从网络字节顺序转回主机字节顺序。最后,它打印出转换回来的值。

如果在一个小端系统(如大多数x86和x86-64系统)上运行此程序,你可能会看到以下输出:

Port in host byte order: 0x1F90
Port in network byte order: 0x901F
Received port converted back to host byte order: 0x1F90

注意htonsntohs函数如何交换字节,从而在主机和网络字节顺序之间进行转换。


通过下面的例子,我们来演示如何使用htonl()ntohl()函数。

#include <stdio.h>
#include <arpa/inet.h>  // for htonl, ntohlint main() {unsigned int ip_address_host_order = 0xC0A80001;  // Represents 192.168.0.1 in hexadecimalunsigned int ip_address_network_order;// Convert IP address from host byte order to network byte orderip_address_network_order = htonl(ip_address_host_order);printf("IP Address in host byte order: 0x%X\n", ip_address_host_order);printf("IP Address in network byte order: 0x%X\n", ip_address_network_order);// Now, let's pretend we received this IP address from the network// and we want to convert it back to host byte orderunsigned int received_ip_host_order = ntohl(ip_address_network_order);printf("Received IP Address converted back to host byte order: 0x%X\n", received_ip_host_order);return 0;
}

这个程序首先定义了一个IP地址0xC0A80001(在十进制中表示为192.168.0.1)。它使用htonl()函数将IP地址从主机字节顺序转换为网络字节顺序,并打印出结果。

然后,该程序模拟从网络上接收到这个IP地址的情况,并使用ntohl()函数将它从网络字节顺序转回主机字节顺序。最后,它打印出转换回来的值。

如果在一个小端系统(如大多数x86和x86-64系统)上运行此程序,你可能会看到以下输出:

IP Address in host byte order: 0xC0A80001
IP Address in network byte order: 0x100A8C0
Received IP Address converted back to host byte order: 0xC0A80001

这显示了如何使用htonlntohl函数在主机和网络字节顺序之间转换32位整数值。

相关文章:

Linux网络编程- 网络字节顺序

基本概念 网络字节顺序是一种规定的数据表示格式&#xff0c;被用于TCP/IP协议栈&#xff0c;特别是在网络传输数据时。它确保不同的计算机和架构之间可以无缝地通信。网络字节顺序是大端字节序&#xff08;big-endian&#xff09;。 字节序的背景 计算机存储多字节数据&…...

如何永久关闭WPS任务窗口?

1、按住任务窗口上的浮动按钮&#xff0c;将其拖出来成悬浮窗口。 第二步&#xff0c;使用火绒弹窗拦截&#xff0c;选中弹出的窗口&#xff0c;进行拦截。注意&#xff1a;拦截次数为2次。即进行2次操作。 操作两次后&#xff0c;弹窗被拦截&#xff0c;此时Word文档改为双页显…...

Cesium 问题:加载 geojson 数据量大浏览器会崩,使用primitive方式加载

文章目录 问题分析 问题 之前加载geojson数据都是使用dataSource和entity的方式&#xff0c;但是当数据量大时&#xff0c;浏览器就会崩掉&#xff1a;提示浏览器内存不足&#xff0c;已暂停渲染 分析 使用primitive方式加载数据&#xff0c;可以提高加载渲染效率。实现方法…...

C++ Primer----1.5类简介 章节练习

头文件 Sales_item.h #ifndef SALESITEM_H #define SALESITEM_H #include <iostream> #include <string>class Sales_item{ public:Sales_item(const std::string &book):isbn(book),units_sold(0),revenue(0.0){}Sales_item(std::istream &is){ is >&…...

爬楼梯Java(斐波那契数列)

题目:有n阶楼梯,一次只能爬一层或者两层,请问有多少种方法? 这类题目其实都可以用斐波那契数列来解决,比如: 一阶楼梯只有一种方法 二阶楼梯有(11,2)两种方法 三阶楼梯有(111,12,21)三种方法 四阶楼梯有(1111,121,112,22,211)五种方式 五阶楼梯有(11111,1112,122,1211,1…...

Maven项目package为jar包后在window运行报A JNI error has occurred

原因&#xff1a;本地java版本与项目结构中使用的java版本不一致&#xff08;之前因为别的需求把idea的java版本改为了18&#xff09; 解决方法 打开项目结构&#xff0c;将idea的java版本改为与本地一致 再修改项目中的pom.xml 重新编译&#xff0c;package即可...

iview 的table表格组件使单元格可编辑和输入

表格的列定义中&#xff0c;在需要编辑的字段下使用render函数 template表格组件 <Table border :data"data" :columns"tableColumns" :loading"loading"></Table>data中定义table对象 table: {tableColumns: [{title: 商品序号,k…...

统计的基本概念及抽样分布

文章目录 &#x1f34b;引言&#x1f34b;总体&#xff08;Population&#xff09;&#x1f34b;总体参数 &#x1f34b;样本&#xff08;Sample&#xff09;&#x1f34b;随机样本&#x1f34b;样本统计量 &#x1f34b;统计量&#xff08;Statistic&#xff09;&#x1f34b;…...

【C++】class的设计与使用(四)this指针

this指针 this作用域是在类内部&#xff0c;只能在成员函数中使用&#xff1b;this在成员函数的开始前构造&#xff0c;在成员函数的结束后清除&#xff1b;编译器在编译的时候也会自动加上this&#xff0c;它作为非静态成员函数的隐含形参&#xff0c;对各成员的访问均通过th…...

mysql 导入sql文件

mysql 导入sql文件 sudo mysql -uroot -p123456 -h127.0.0.1 sudo mysql -uroot -p123456 -h127.0.0.1然后 show databases;然后 use 数据库名称; 然后 source 20230920031001.sql;如果不加 -h127.0.0.1 可能会出现错误 /var/lib/mysql.sock error 通过 navicat导入的话&am…...

springcloud:三、ribbon负载均衡原理+调整策略+饥饿加载

Ribbon负载均衡原理 调整Ribbon负载均衡策略 第一种会对order-service里所有的服务消费者都采用该新规则 第二种会针对order-service里某个具体的服务消费者采用该新规则 饥饿加载...

【Unity编辑器扩展】Tranform组件自定义扩展,复制位置旋转缩放数据

目录 一、Tranform组件数据的扩展 二、 RectTransform组件数据的扩展 很多时候我们在做Tranform属性配置的时候需要反复的获取对象的位置信息,旋转信息。这个时候有个方便的工具会使得我们的效率大大提升。 一、Tranform组件数据的扩展 1.可以复制世界坐标,世界旋转 V…...

自动驾驶领域中的CMS系统应用探讨

由佐思汽研主办的“ ICVS汽车智能网联大会”正式启幕&#xff0c;邀请Tier1、软件供应商、操作系统商、智能驾驶及智能驾舱相关企业&#xff0c;共同探讨行业现状、创新技术应用交流、以及未来行业发展趋势等。 赛格导航视频产品线总监刘玉龙应邀参加本次大会&#xff0c;并发…...

十分钟理解OSPF路由协议

十分钟理解OSPF路由协议 1.RIP的缺陷以跳数为度量值最大跳数为15更新路由表采用全更新收敛速度慢 2.RIP与OSPF比较OSPF概述运行OSPF协议之前运行OSPF协议之后 3.OSPF协议工作过程1.发现邻居2.建立邻接关系3.传递链路状态信息4.计算路由 4.OSPF分区域管理 有RIP协议&#xff0c;…...

Python 编程基础 | 第一章-预备知识 | 1.4、包管理工具

一、包管理工具 1、pip简介 pip是Python自带的包管理器&#xff0c;它可以帮助我们安装、升级和卸载Python包。Python包是一组Python模块&#xff0c;它们可以提供各种功能&#xff0c;例如数据分析、Web开发、机器学习等等。pip可以让我们轻松地安装这些包&#xff0c;以便我…...

delphi中使用CADVCL 10.0 Enterprise控件解析DXF文件生成图片保存到本地

使用案例demos中GettingStarted案例 修改OnClick方法 更换代码 varvDrawing: TsgCADDXFImage;I: Integer;vEnt: TsgDXFEntity;vDXFImage:TsgCADImage;Bitmap: TBitmap;jpg: TJpegImage; beginvDrawing : TsgCADDXFImage.Create;vDrawing.LoadFromFile(d:\Entities.dxf);for …...

Hazelcast系列(三):hazelcast管理中心

系列文章 Hazelcast系列(一)&#xff1a;初识hazelcast Hazelcast系列(二)&#xff1a;hazelcast集成 Hazelcast系列(三)&#xff1a;hazelcast管理中心 目录 前言 平台搭建 测试 其他 参考 总结 前言 前面&#xff0c;我们通过几种配置方式&#xff08;Hazelcast系…...

QT 绘画功能的时钟

.h #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QPaintEvent> #include <QDebug> //信息调试类 #include <QPainter> #include <QPixmap> //图像引擎类 #include <QTime> #include <QTimer> …...

设计模式之道-模板方法模式

文章目录 模板方法模式简介作用模板方法模式的缺点模板方法模式的应用场景业务场景开源框架中的应用 对比回调和Hook模式关于组合优先于继承 关于设计模式乱用的现象 模板方法模式 简介 模板方法模式是一种行为型设计模式&#xff0c;该设计模式的核心在于通过抽象出一套相对…...

头哥的实践平台的Linux文件/目录管理

一 Linux 文件/目录管理 1.本关的编程任务是补全右侧代码片段中Begin至End中间的代码&#xff0c;具体要求如下&#xff1a; 新创建两个文件空文件file1和file2。 删除系统已存在的两个文件oldFile1和oldFile2。 #!/bin/bash#在以下部分写出完成任务的命令 #***********begi…...

装饰模式(Decorator Pattern)重构java邮件发奖系统实战

前言 现在我们有个如下的需求&#xff0c;设计一个邮件发奖的小系统&#xff0c; 需求 1.数据验证 → 2. 敏感信息加密 → 3. 日志记录 → 4. 实际发送邮件 装饰器模式&#xff08;Decorator Pattern&#xff09;允许向一个现有的对象添加新的功能&#xff0c;同时又不改变其…...

ES6从入门到精通:前言

ES6简介 ES6&#xff08;ECMAScript 2015&#xff09;是JavaScript语言的重大更新&#xff0c;引入了许多新特性&#xff0c;包括语法糖、新数据类型、模块化支持等&#xff0c;显著提升了开发效率和代码可维护性。 核心知识点概览 变量声明 let 和 const 取代 var&#xf…...

label-studio的使用教程(导入本地路径)

文章目录 1. 准备环境2. 脚本启动2.1 Windows2.2 Linux 3. 安装label-studio机器学习后端3.1 pip安装(推荐)3.2 GitHub仓库安装 4. 后端配置4.1 yolo环境4.2 引入后端模型4.3 修改脚本4.4 启动后端 5. 标注工程5.1 创建工程5.2 配置图片路径5.3 配置工程类型标签5.4 配置模型5.…...

黑马Mybatis

Mybatis 表现层&#xff1a;页面展示 业务层&#xff1a;逻辑处理 持久层&#xff1a;持久数据化保存 在这里插入图片描述 Mybatis快速入门 ![在这里插入图片描述](https://i-blog.csdnimg.cn/direct/6501c2109c4442118ceb6014725e48e4.png //logback.xml <?xml ver…...

【Java学习笔记】Arrays类

Arrays 类 1. 导入包&#xff1a;import java.util.Arrays 2. 常用方法一览表 方法描述Arrays.toString()返回数组的字符串形式Arrays.sort()排序&#xff08;自然排序和定制排序&#xff09;Arrays.binarySearch()通过二分搜索法进行查找&#xff08;前提&#xff1a;数组是…...

ffmpeg(四):滤镜命令

FFmpeg 的滤镜命令是用于音视频处理中的强大工具&#xff0c;可以完成剪裁、缩放、加水印、调色、合成、旋转、模糊、叠加字幕等复杂的操作。其核心语法格式一般如下&#xff1a; ffmpeg -i input.mp4 -vf "滤镜参数" output.mp4或者带音频滤镜&#xff1a; ffmpeg…...

c#开发AI模型对话

AI模型 前面已经介绍了一般AI模型本地部署&#xff0c;直接调用现成的模型数据。这里主要讲述讲接口集成到我们自己的程序中使用方式。 微软提供了ML.NET来开发和使用AI模型&#xff0c;但是目前国内可能使用不多&#xff0c;至少实践例子很少看见。开发训练模型就不介绍了&am…...

Xen Server服务器释放磁盘空间

disk.sh #!/bin/bashcd /run/sr-mount/e54f0646-ae11-0457-b64f-eba4673b824c # 全部虚拟机物理磁盘文件存储 a$(ls -l | awk {print $NF} | cut -d. -f1) # 使用中的虚拟机物理磁盘文件 b$(xe vm-disk-list --multiple | grep uuid | awk {print $NF})printf "%s\n"…...

人机融合智能 | “人智交互”跨学科新领域

本文系统地提出基于“以人为中心AI(HCAI)”理念的人-人工智能交互(人智交互)这一跨学科新领域及框架,定义人智交互领域的理念、基本理论和关键问题、方法、开发流程和参与团队等,阐述提出人智交互新领域的意义。然后,提出人智交互研究的三种新范式取向以及它们的意义。最后,总结…...

日常一水C

多态 言简意赅&#xff1a;就是一个对象面对同一事件时做出的不同反应 而之前的继承中说过&#xff0c;当子类和父类的函数名相同时&#xff0c;会隐藏父类的同名函数转而调用子类的同名函数&#xff0c;如果要调用父类的同名函数&#xff0c;那么就需要对父类进行引用&#…...