linux程序设计——主机字节序和网络字节序(第十五章)

15.2.10    主机字节序和网络字节序

当在基于intel处理器的linux机器上运行新版本的服务器和客户程序时,可以用netstat命令查看网络连接状况.它显示了客户/服务器连接正在等待关闭.连接将在一段超时间之后关闭,如下所示:

可以看到这条连接对应的服务器和客户的端口号.local address一栏显示的是服务器,而foreign address一栏显示的是远程客户(即使是在同一台机器上,它仍然是通过网络连接的).为了确保所有套接字都是不同的,这些客户端口一般都与服务器监听套接字不同,并且在这台计算机上是唯一的.

可是,显示的本地地址(服务器套接字)端口号是1574,而选择的是9734.为什么不一样呢?原因是,通过套接字接口传递的端口号和地址都是二进制数字,不同的计算机使用不同的字节序来表示整数.例如,Intel处理器将32位的整数分为4个连续的字节,并以字节序1-2-3-4存储在内存中,这里的1表示最高位的字节.而IBM PowerPC处理器是以字节4-3-2-1的方式存储整数.如果保存整数的内存只是以逐个字节的方式来复制,两个不同的计算机得到的整数值就会不一致.

为了使不同类型的计算机可以通过网络传输的多字节整数的值达成一致,需要定义一个网络字节序,客户和服务器程序必须在传递之前,将它们的内部整数表示方式转换为网络字节序.它们通过定义在头文件netinet/in.h中的函数来完成这一工作.这些函数如下所示:

#include <netinet/in.h>
unsigned long int htonl(unsigned long int hostlong);
unsigned short int htons(unsigned short int hostshort);
unsigned long int ntohl(unsigned long int netlong);
unsigned short int ntohs(unsigned short int netshort);

这些函数将16位整数和32位整数在主机字节序和标准的网络字节序之间进行转换.函数名是与之对应的转换操作的简写形式.例如"host to network, long"(hton,长整数从主机字节序到网络字节序的转换)和"host to network, short"(htons,短整数从主机字节序到网络字节序的转换).如果计算机本身的主机字节序和网络字节序相同,这些函数的内容实际上就是空操作.

为了保证16位的端口号有正确的字节序,服务器和客户需要这些函数转换端口地址.新服务器程序server3.c中的改动是:

server_address.sin_addr.s_addr = htol(INADDR_ANY);
server_address.sin_port = htons(9734);

不需要对函数调用inet_addr("127.0.0.1")进行转换,因为inet_addr已被定义为产生一个网络字节序的结果.新程序client3.c中的改动是:

address.sin_port = htons(9734);

服务器也做了改动,通过INADDR_ANY来允许到达服务器任一网络接口的连接.

运行server3和client3时,看到如下所示:

使用netstat -A inet将看到本地连接使用的是正确的端口:

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-08-24 19:20:00

linux程序设计——主机字节序和网络字节序(第十五章)的相关文章

linux程序设计——网络信息(第十五章)

15.3    网络信息 当目前为止,客户和服务器程序一直是吧地址和端口号编译到它们自己的内部.对于一个更通用的服务器和客户程序来说,可以通过网络信息函数来决定应该使用的地址和端口. 如果有足够的权限,可以将自己的服务添加到/etc/services文件中的已知服务列表中,并在这个文件中为端口号分配一个名字,使用户可以使用符号化的服务名而不是端口号的数字. 类似地,如果给定一个计算机的名字,可以通过调用解析地址的主机数据库函数来确定它的IP地址.这些函数通过查询网络配置文件来完成这一工作,如/e

第五篇:主机字节序与网络字节序的转换

前言 我们知道,数据在主机内的存放有两种模式,也就是说,主机字节序有两种:大端和小端( 这里假定读者已经清楚这个问题 ).但在网络通信中,要求通信数据( 通信数据这里指IP号和端口号 )的使用必须用网络字节序.什么又是网络字节序? 网络字节序可以理解为主机字节序的大端模式.如果你的主机字节序原本就是大端模式,那么你可以考虑不用将通信数据转换为网络字节序:但如果你的主机字节序是小端,那么通信数据必须被下面提及到的其中某个函数进行处理,转换成网络字节序后方可使用( 即存放进套接字地址结构变量 ).

Python网络编程——主机字节序和网络字节序之间的相互转换

If you ever need to write a low-level network application, it may be necessary to handle the low-level data transmission over the wire between two machines. This operation requires some sort of conversion of data from the native host operating system

主机字节序和网络字节序转换

为什么要转换? 主机字节序:整数在内存中保存的顺序,不同的处理器对应不容的模式 Little endian 将低序字节存储在起始地址 Big endian    将高序字节存储在起始地址 网络字节序:整数在网络中的发送顺序 网络字节顺序是TCP/IP中规定好的一种数据表示格式,它与具体的CPU类型.操作系统等无关,从而可以保证数据在不同主机之间传输时能够被正确解释. 网络字节顺序采用big endian排序方式 htons 本地的无符号short型主机字节序转换为网络字节序 htonl    

TCP/IP网络编程之字节序和网络字节序

一.概要 本篇文章主要讲解基于.net中tcp/ip网络通信编程中的网络字节序.在自我进步的过程中记录这些内容,方便自己记忆的同时也希望可以帮助到大家.技术的进步源自于分享和不断的自我突破. 技术交流QQ群:580749909  欢迎交流有问必答,文章尾有个人的微信公众号有兴趣的小伙伴多多关注. 二.简介 在此之前我们需要了解清楚几个概念. CPU架构:人们常说的x86 x64是一种架构,但是他有32位的和64位的.32位的叫x86 ,后来出现基于它的64位版,就叫x64. 操作系统位数 :64

主机字节序 与 网络字节序

一.字节顺序 是指占用内存多于一个字节类型的数据在内存中的存放顺序. java中一个int型数据占用4个字节,假如有一个16进制的int数,int value = 0x01020304 小端字节序(little endian):低字节数据存放在内存低地址 大端字节序(bigendian): 低字节数据存放在高地址处 主机字节序跟CPU有关的,IA架构(Intel.AMD)的CPU中是Little-Endian,而PowerPC .SPARC和Motorola处理器是Big-Endian. 网络字

主机字节序 和 网络字节序

不同的CPU有不同的字节序类型,这些字节序是指 整数 在内存中保存的顺序,这个叫做 主机序. 最常见的有两种: 1.Little endian:将低序字节存储在起始地址 2.Big endian:将高序字节存储在起始地址 LE little-endian(小端) 最符合人的思维的字节序: 地址低位存储值的低位: 地址高位存储值的高位: 怎么讲是最符合人的思维的字节序,是因为从人的第一观感来说: 低位值小,就应该放在内存地址小的地方,也即内存地址低位: 反之,高位值就应该放在内存地址大的地方,也即

大端与小端,大尾与小尾,高尾端与低尾端,主机字节序与网络字节序

概念剖析 一时记忆与理解大端.小端的概念很容易,但时间一长,对于相似的概念人类的记忆向来是模糊的,甚至是换位的.所以除非你的记忆非常牢靠,否则借助大端和小端这样的名字,你很难将概念与内容联系紧密. 也有文章提到用大尾与小尾的概念,个人觉得这个概念还是没有解决存储概念中的基本问题,大与小还是没有脱离以前的概念,但是引入了尾的概念,已经比大端与小端要更清晰一点. 目前对于记忆大小端,我觉得这篇文章中提到的高尾端/低尾端名词记忆方法比较科学.形象.实话说,当时计算机科学著作翻译Big Endian与S

python通过ntohl和htonl等函数实现主机字节序和网络字节序相互转换

Python的socket库提供了将数据在网络字节序和主机字节序之间相互转换的函数.有什么作用呢? 在编写低层网络应用时,或许需要处理通过电缆在两台设备之间传送的低层数据.在这种操作中,需要把主机操作系统发出的数据转换成网络格式,或者做逆向转换,因为这两种数据的表示方式不一样. 技术点解析: 1.定义convert_integer()函数,注意函数格式(以后都会提醒这个,要养成习惯) 2.socket库中的类函数 ntohl() 把网络字节序转换成了长整形主机字节序 htonl() 把长整形主机