日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

网络数据的大小端问题

發(fā)布時間:2025/3/12 编程问答 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 网络数据的大小端问题 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

不同 CPU 中,4 字節(jié)整數(shù) 1 在內(nèi)存空間的存儲方式是不同的。4 字節(jié)整數(shù) 1 可用 2 進制表示如下:

00000000 00000000 00000000 00000001

有些 CPU 以上面的順序存儲到內(nèi)存,另外一些 CPU 則以倒序存儲,如下所示:

00000001 00000000 00000000 00000000

若不考慮這些就收發(fā)數(shù)據(jù)會發(fā)生問題,因為保存順序的不同意味著對接收數(shù)據(jù)的解析順序也不同。

大端序和小端序

CPU 向內(nèi)存保存數(shù)據(jù)的方式有兩種:

  • 大端序(Big Endian):高位字節(jié)存放到低位地址(高位字節(jié)在前)。
  • 小端序(Little Endian):高位字節(jié)存放到高位地址(低位字節(jié)在前)。


僅憑描述很難解釋清楚,不妨來看一個實例。假設在 0x20 號開始的地址中保存 4 字節(jié) int 型數(shù)據(jù) 0x12345678,大端序 CPU 保存方式如下圖所示:


圖1:整數(shù) 0x12345678 的大端序字節(jié)表示


對于大端序,最高位字節(jié) 0x12 存放到低位地址,最低位字節(jié) 0x78 存放到高位地址。小端序的保存方式如下圖所示:


圖2:整數(shù) 0x12345678 的小端序字節(jié)表示


不同 CPU 保存和解析數(shù)據(jù)的方式不同(主流的 Intel 系列 CPU 為小端序),小端序系統(tǒng)和大端序系統(tǒng)通信時會發(fā)生數(shù)據(jù)解析錯誤。因此在發(fā)送數(shù)據(jù)前,要將數(shù)據(jù)轉換為統(tǒng)一的格式——網(wǎng)絡字節(jié)序(Network Byte Order)。網(wǎng)絡字節(jié)序統(tǒng)一為大端序。

主機 A 先把數(shù)據(jù)轉換成大端序再進行網(wǎng)絡傳輸,主機 B 收到數(shù)據(jù)后先轉換為自己的格式再解析。

網(wǎng)絡字節(jié)序轉換函數(shù)

在《bind()和connect()函數(shù):綁定套接字并建立連接》一節(jié)中講解了 sockaddr_in 結構體,其中就用到了網(wǎng)絡字節(jié)序轉換函數(shù),如下所示:

  • //創(chuàng)建sockaddr_in結構體變量
  • struct sockaddr_in serv_addr;
  • memset(&serv_addr, 0, sizeof(serv_addr)); //每個字節(jié)都用0填充
  • serv_addr.sin_family = AF_INET; //使用IPv4地址
  • serv_addr.sin_addr.s_addr = inet_addr("127.0.0.1"); //具體的IP地址
  • serv_addr.sin_port = htons(1234); //端口號
  • htons() 用來將當前主機字節(jié)序轉換為網(wǎng)絡字節(jié)序,其中h代表主機(host)字節(jié)序,n代表網(wǎng)絡(network)字節(jié)序,s代表short,htons 是 h、to、n、s 的組合,可以理解為”將 short 型數(shù)據(jù)從當前主機字節(jié)序轉換為網(wǎng)絡字節(jié)序“。

    常見的網(wǎng)絡字節(jié)轉換函數(shù)有:

    • htons():host to network short,將 short 類型數(shù)據(jù)從主機字節(jié)序轉換為網(wǎng)絡字節(jié)序。
    • ntohs():network to host short,將 short 類型數(shù)據(jù)從網(wǎng)絡字節(jié)序轉換為主機字節(jié)序。
    • htonl():host to network long,將 long 類型數(shù)據(jù)從主機字節(jié)序轉換為網(wǎng)絡字節(jié)序。
    • ntohl():network to host long,將 long 類型數(shù)據(jù)從網(wǎng)絡字節(jié)序轉換為主機字節(jié)序。


    通常,以s為后綴的函數(shù)中,s代表 2 個字節(jié) short,因此用于端口號轉換;以l為后綴的函數(shù)中,l代表 4 個字節(jié)的 long,因此用于 IP 地址轉換。

    舉例說明上述函數(shù)的調(diào)用過程:

  • #include <stdio.h>
  • #include <stdlib.h>
  • #include <WinSock2.h>
  • #pragma comment(lib, "ws2_32.lib")
  • ?
  • int main(){
  • unsigned short host_port = 0x1234, net_port;
  • unsigned long host_addr = 0x12345678, net_addr;
  • ?
  • net_port = htons(host_port);
  • net_addr = htonl(host_addr);
  • ?
  • printf("Host ordered port: %#x\n", host_port);
  • printf("Network ordered port: %#x\n", net_port);
  • printf("Host ordered address: %#lx\n", host_addr);
  • printf("Network ordered address: %#lx\n", net_addr);
  • ?
  • system("pause");
  • return 0;
  • }
  • 運行結果:
    Host ordered port: 0x1234
    Network ordered port: 0x3412
    Host ordered address: 0x12345678
    Network ordered address: 0x78563412

    另外需要說明的是,sockaddr_in 中保存 IP 地址的成員為 32 位整數(shù),而我們熟悉的是點分十進制表示法,例如 127.0.0.1,它是一個字符串,因此為了分配 IP 地址,需要將字符串轉換為 4 字節(jié)整數(shù)。

    inet_addr() 函數(shù)可以完成這種轉換。inet_addr() 除了將字符串轉換為 32 位整數(shù),同時還進行網(wǎng)絡字節(jié)序轉換。請看下面的代碼:

  • #include <stdio.h>
  • #include <stdlib.h>
  • #include <WinSock2.h>
  • #pragma comment(lib, "ws2_32.lib")
  • ?
  • int main(){
  • char *addr1 = "1.2.3.4";
  • char *addr2 = "1.2.3.256";
  • ?
  • unsigned long conv_addr = inet_addr(addr1);
  • if(conv_addr == INADDR_NONE){
  • puts("Error occured!");
  • }else{
  • printf("Network ordered integer addr: %#lx\n", conv_addr);
  • }
  • ?
  • conv_addr = inet_addr(addr2);
  • if(conv_addr == INADDR_NONE){
  • puts("Error occured!");
  • }else{
  • printf("Network ordered integer addr: %#lx\n", conv_addr);
  • }
  • ?
  • system("pause");
  • return 0;
  • }
  • 運行結果:
    Network ordered integer addr: 0x4030201
    Error occured!

    從運行結果可以看出,inet_addr() 不僅可以把 IP 地址轉換為 32 位整數(shù),還可以檢測無效 IP 地址。

    注意:為 sockaddr_in 成員賦值時需要顯式地將主機字節(jié)序轉換為網(wǎng)絡字節(jié)序,而通過 write()/send() 發(fā)送數(shù)據(jù)時 TCP 協(xié)議會自動轉換為網(wǎng)絡字節(jié)序,不需要再調(diào)用相應的函數(shù)。

    總結

    以上是生活随笔為你收集整理的网络数据的大小端问题的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。