栈区内存地址
#include <stdio.h>
#include<string.h>
#include<stdlib.h>
int main() {
char a[20] = { "helloworld" };
printf("%d\n",&a[1]);
printf("%d\n",&a[2]);
printf("%d\n",&a[3]);
int b = 1;
int c = 2;
printf("%d\n", &b);
printf("%d\n", &c);
//printf(a[3]);
return 0;
}
根据图片可以看出,代买在定义的过程中,从上至下,由左到右,变量依次压栈(局部变量),栈底变量地址大,栈顶变量地址小,但定义数组时,虽然数组也是压栈定义,但是数组的第一个元素位置到最后一个元素位置是依次增大的,所以数组的每个元素不是压栈形式的,是反向的。
看个图
大小端存储方式
内容:
大端存储(Big-Endian)
数据的高字节存储在低地址中,数据的低字节存储在高地址中 如:
int a = 1;
// int在内存中占 4Byte;
// a在内存中的分布为 0x1 0x0 0x0 0x0;
// 从左到右内存地址降低,也就是高字节地址存放的是a的低字节数据
应用:C51是大端存储;socket编程中网络字节序一般是大端存储
小端存储(Little-Endian)
数据的高字节存储在高地址中,数据的低字节存储在低地址中 如:
int a = 1;
// int在内存中占 4Byte;
// a在内存中的分布为 0x0 0x0 0x0 0x1;
// 从左到右内存地址降低,也就是低字节地址存放的是a的低字节数据
应用:x86结构、ARM和DSP都是小端存储
测试
进行大小端存储测试主要有两种方法,一种是通过强转,另一种是联合 强转测试
int a = 134480385; // 二进制0000 1000 0000 0100 0000 0010 0000 0001
char *p = (char *)&a; // 一次操作一个字节
printf("%c\n", p[0] + ‘0’);
printf("%c\n", p[1] + ‘0’);
printf("%c\n", p[2] + ‘0’);
printf("%c\n", p[3] + ‘0’);
可以查看p[0], p[1], p[2], p[3]的数据判断大小端存储
联合测试
union un {
int i;
char c[4];
}un1 = { 134480385 };
int main() {
printf("%c\n", un1.c[0] + ‘0’);
printf("%c\n", un1.c[1] + ‘0’);
printf("%c\n", un1.c[2] + ‘0’);
printf("%c\n", un1.c[3] + ‘0’);
system(“pause”);
return 0;
}
可以输出c[0],c[1],c[2],c[3]可根据输出判断是大端还是小端存储
以上测试均是在win10 vs2017 64bit编译器下进行的
再看个图
针对数据十六进制数0x0A0B0C0D
针对大小端存储方式的理解与应用(与联合一起)
因为联合的共用存储空间的性质,所以char c一定在起始位置进行放数据,读取的时候也就读取固定的位置,主要是靠int a来判断存储的方式,char c用来读取数据判断的。
#include <stdio.h>
int checkSystem() {
union System {
int a;
char c;
}u;
u.a = 1;
return u.c == 1;
}
int main() {
printf("%d", checkSystem());
return 0;
}