您的位置:首页 > 娱乐 > 明星 > C语言:大小端模式、判断大小端、大小端转换

C语言:大小端模式、判断大小端、大小端转换

2024/12/24 21:59:43 来源:https://blog.csdn.net/d704791892/article/details/141787687  浏览:    关键词:C语言:大小端模式、判断大小端、大小端转换

目录

1. 什么是大端和小端

2.为什么会存在大小端的问题

3. 判断主机字节序 (主机大小端)

3.1 使用联合体 (union)

3.2 使用指针

3.3 强制转为 char 类型法

4. 大小端转换


1. 什么是大端和小端

对于一个存储空间大于 1 个字节的数据,在内存中有两种存储模式,

大端模式 (big-endian):数据的高字节在内存的低地址存放,数据的低字节在内存的高地址存放

小端模式 (little-endian):数据的高字节在内存的高地址存放,数据的低字节在内存的低地址存放

简单记忆:

大端模式:大高低

小端模式:小低低

比如,比如数字:0x12345678,大小端存储模式如下:

2.为什么会存在大小端的问题

        在 C 语言中,有占 1 个字节内存空间的 char 类型,占 2 个字节内存空间的 short int 类型,以及占 4 个字节内存空间的 int,double 类型和占 8 个字节内存空间的 double 类型等等,那么对于大于 1 个字节空间的数据类型的数据在内存中存储时,就必然产生一个数据在内存中应该将其低字节数据安排在内存的低地址还是高地址呢?,对这个问题做的不同选择,就形成了大小端问题。

3. 判断主机字节序 (主机大小端)

3.1 使用联合体 (union)

        联合体的每一个成员共用一个内存地址,修改其中一个成员的数据,可能会影响其它成员的数据的值。比如以下联合体,成本变量 i 和 c 的内存地址是相同的,它的内存分布示意图为

	union {int i;char c;}un;	// 匿名联合体

#include <stdio.h>int is_little_endian() {union {int i;char c;}un;	// 匿名联合体un.i = 1;	return un.c;	// 小端:返回 1,说明数据的低字节在内存的低地址存放// 大端:返回 0,说明数据的低字节在内存的高地址存放
}int main() {if (is_little_endian())printf("little-endian\n");elseprintf("big-endian\n");return 0;
}

3.2 使用指针

#include <stdio.h>int is_little_endian() {int i = 1;// 等同于 char* p = (char*)&i; return *p;return *(char*)&i;	// 小端:返回 1,说明数据的低字节在内存的低地址存放// 大端:返回 0,说明数据的低字节在内存的高地址存放
}int main() {if (is_little_endian())printf("little-endian\n");elseprintf("big-endian\n");return 0;
}

3.3 强制转为 char 类型法

#include <stdio.h>int is_little_endian() {int i = 1;// (char)i : 强转为 char 类型,实际就是取 i 在内存中第一个字节的数据return (char)i;	// 小端:返回 1,说明数据的低字节在内存的低地址存放// 大端:返回 0,说明数据的低字节在内存的高地址存放
}int main() {if (is_little_endian())printf("little-endian\n");elseprintf("big-endian\n");return 0;
}

4. 大小端转换

        1. 编写大小端转化函数,并打印本机字节序下的数字:0x11223344,分别在小端模式和大端模式下是什么数据。

#include <stdio.h>int is_little_endian();
int to_opposite_endian(int data);
int to_big_endian(int data);
int to_little_endian(int data);int main() {int i = 0x11223344;printf("little_endian:	%x\n", to_little_endian(i));printf("big_endian:	%x\n", to_big_endian(i));return 0;
}int is_little_endian() {int i = 1;return (char)i;	// 小端:返回 1,大端:返回 0
}// 大小端转换
int to_opposite_endian(int data) {int size = sizeof(data);int des = 0;		// 目标数据int mask = 0xff;	// 掩码int temp;for (int i = 0; i < size; i++) {des = des << 8;			// 左移8位temp = data & mask;		// 取值temp = temp >> i * 8;	// 移到低8位des |= temp;mask = mask << 8;		// 掩码左移8位,为下一次取数据用}return des;
}int to_big_endian(int data) {if (!is_little_endian())	// 大端return data;return to_opposite_endian(data);
}int to_little_endian(int data) {if (is_little_endian())	// 小端return data;return to_opposite_endian(data);
}

2. 在网络编程中,可以使用 htonl, htons, ntohl, ntohs 等函数

  • htonl(uint32_t hostlong) // host to network long,32位无符号整型的主机字节序转成网络字节序
  • htons(uint16_t hostshort) // host to network short,16位无符号短整型的主机字节序转成网络字节序
  • ntohl(uint32_t netlong) // network to host long,32位无符号整型的网络字节序转成主机字节顺序的
  • ntohs(uint16_t netshort) //16位无符号短整型的网络字节序转成主机字节序

网络字节序:大端模式(big-endian),即数据的高字节在内存的高地址存放,数据的低字节在内存的低地址存放。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com