什么是大小端?如何确定大小端?

一、什么是大小端?

对于一个由2个字节组成的16位整数,在内存中存储这两个字节有两种方法:一种是将低序字节存储在起始地址,这称为小端(little-endian)字节序;另一种方法是将高序字节存储在起始地址,这称为大端(big-endian)字节序。

假如现有一32位int型数0x12345678,那么其MSB(Most Significant Byte,最高有效字节)为0x12,其LSB (Least Significant Byte,最低有效字节)为0x78,在CPU内存中有两种存放方式:(假设从地址0x4000开始存放)

总结:

大端是高字节存放到内存的低地址

小端是高字节存放到内存的高地址

二、如何确定大小端

有些CPU公司用大端(譬如C51单片机);有些CPU用小端(譬如ARM)。(大部分是用小端模式,大端模式的不算多)。于是乎我们写代码时,当不知道当前环境是用大端模式还是小端模式时就需要用代码来检测当前系统的大小端。下面给出用程序判断大小端的两种方法:

#include <stdio.h>

// 共用体中很重要的一点:a和b都是从u1的低地址开始存放的。
// 假设u1所在的4字节地址分别是:0、1、2、3的话,那么a自然就是0、1、2、3;
// b所在的地址是0而不是3.

union myunion
{
	int a;
	char b;
};

// 如果是小端模式则返回1,小端模式则返回0
int is_little_endian(void)
{
	union myunion u1;
	u1.a = 0x12345678;				// 地址0的那个字节内是0x78(小端)或者0x12(大端)
    if(0x78 == u1.b)
        return 1;
    else if(0x12 == u1.b)
	    return 0;
}

int is_little_endian2(void)
{
	int a = 0x12345678;
	char b = *((char *)(&a));		// 指针方式其实就是共用体的本质
	if(0x78 == b)
        return 1;
    else if(0x12 == b)
	    return 0;
}


int main(void)
{
	int i = is_little_endian2();
	//int i = is_little_endian();
	if (i == 1)
	{
		printf("小端模式\n");
	}
	else
	{
		printf("大端模式\n");
	}
	
	return 0;
}

下面给出三种确定大小端错误的方案

	// 强制类型转换
int a;
char b;
a = 1;
b = (char)a;
printf("b = %d.\n", b);
	// 移位
int a, b;
a = 1;
b = a >> 1;
printf("b = %d.\n", b);
	// 位与
int a = 1;
int b = a & 0xff;		// 也可以写成:char b = a & 0x01;
printf("b = %d.\n", b);

位与、移位、强制类型转换等运算是编译器提供的运算,这个运算是高于内存层次的(或者说这些运算在二进制层次具有可移植性,也就是说&的时候一定是高字节&高字节,低字节&低字节,和二进制存储无关)。

 

REF:

朱有鹏课堂笔记

  • 149
    点赞
  • 627
    收藏
    觉得还不错? 一键收藏
  • 14
    评论
您提到的确实是一个问题,实际上在使用 Snap7 库进行数据读写时,需要注意数据的大小端问题。 Snap7 库使用的是 Siemens S7 协议,而该协议规定数据传输时采用的是大端模式(Big Endian)。因此,在进行数据转换时,需要将小端数据转换为大端数据。 下面是一份修改后的代码,可以支持大小端转换: ```c++ #include <iostream> #include "snap7/snap7.h" #include <algorithm> using namespace std; int main() { // 创建 Snap7 客户端 TS7Client* client = new TS7Client(); // 连接到PLC int status = client->ConnectTo("192.168.1.100", 0, 1); if (status != 0) { cout << "连接PLC失败!" << endl; return -1; } // 定义写入数据的地址和数值 int db = 301; int start = 2; // DB301.DBW2 地址的起始位置 float value = 2.3456; // 将 float 类型的数据转换为大端 byte 数组 unsigned char data[4]; std::reverse_copy(reinterpret_cast<const char*>(&value), reinterpret_cast<const char*>(&value) + sizeof(value), data); // 写入数据到 PLC status = client->WriteArea(S7AreaDB, db, start, sizeof(data), S7WLReal, data); if (status != 0) { cout << "写入数据失败!" << endl; return -1; } // 断开与 PLC 的连接 client->Disconnect(); cout << "写入数据成功!" << endl; return 0; } ``` 在这个代码中,我们使用 std::reverse_copy 函数将小端的 float 类型数据转换为大端的 byte 数组,然后再将该数组写入到 PLC 中。这样就可以避免大小端问题的出现。
评论 14
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值