首页 技术 正文
技术 2022年11月19日
0 收藏 902 点赞 2,359 浏览 2057 个字
/* C语言零长度数组大小和取值问题 */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>struct str
{
int type;
char s[];//零长度的数组
};struct foo
{
int type;
char *s;
};void test()
{
printf("str size is [%d] \n", sizeof(struct str)); //打印 4
/*
使用GDB查看汇编代码
对于struct str 结构体中的 char s[0]来说,汇编代码用了lea指令,lea 0x04(%rax), %rdx
对于struct foo 结构体中的 char*s来说,汇编代码用了mov指令,mov 0x04(%rax), %rdx
lea全称load effective address,是把地址放进去,而mov则是把地址里的内容放进去。
访问成员数组名其实得到的是数组的相对地址,而访问成员指针其实是相对地址里的内容(这和访问其它非指针或数组的变量是一样的)
对于数组 char s[10]来说,数组名 s 和 &s 都是一样的。char s[0] 表示的是地址。char*s 表示的地址的内容
*/ printf("foo size is [%d] \n", sizeof(struct foo)); //32位机器上 打印8 //零长度的数组的打印
struct str s1;
printf("Arrays of Length Zero print [%p] \n", s1.s);
printf("Arrays of Length Zero print [%p] \n", &s1.s); //结果相同 打印的是char s[0] 的地址 struct foo f1;
//printf("Arrays of Length Zero print [%x] \n", f1.s); //程序core down 验证 char*s 访问成员指针其实是相对地址里的内容}//验证char s[]更多的类似于一个占位符
struct str1
{
int length;
int flags;
char s[];//零长度的数组(Flexible Array)
};void test1()
{
//零长度数组的占位符功能
//注意 char s[]更多的类似于一个占位符,并非结构体成员,所以计算结构体大小时,并没有char s[]
printf("===size==='[%d]====\n", sizeof(struct str1));
}int main()
{ test1();
printf("-----ok------\n");
getchar();
return ;
}
/* C语言零长度数组使用 */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>#define GTS_HDR(s) ((struct str *)((s)-(sizeof(struct str))))struct str
{
int length;
unsigned char flags;
char s[];//零长度的数组(Flexible Array)
};/*
零长度的数组优势
第一个优点是,方便内存释放。如果我们的代码是在一个给别人用的函数中,你在里面做了二次内存分配,并把整个结构体返回给用户。
用户调用free可以释放结构体,但是用户并不知道这个结构体内的成员也需要free,所以你不能指望用户来发现这个事。所以,
如果我们把结构体的内存以及其成员要的内存一次性分配好了,并返回给用户一个结构体指针,用户做一次free就可以把所有的内存也给释放掉。第二个优点是,这样有利于访问速度。连续的内存有益于提高访问速度,也有益于减少内存碎片。零长度数组的具体使用可以参考redis中sds结构
*/char * create(void)
{
int len = ;
struct str *s1 = NULL; s1 = calloc(, sizeof(struct str) + len);
//模仿redis中sds结构
s1->flags = ;
s1->length = len; //注意 char s[0]只是一个占位符,不占用实际内存空间,所以成员变量char s[0]的offset不是s1->s,而是s1+sizeof(struct str)
//因此也不应该对外暴露struct str 结构,防止用户操作struct str 的内存空间
strcpy((char *)s1 + sizeof(struct str), "hello world "); //错误示例 打印空
printf("====error show==[%s]=====\n", s1->s); return (char *)s1 + sizeof(struct str);
}void test()
{
char * s = create(); printf("--s is -[%s]---\n", s);}int main()
{ test();
printf("-----ok------\n");
getchar();
return ;
}

 零长度数组只有GUN/GCC支持 别的厂家可能不支持,此时需要用 char data[1]来代替

相关推荐
python开发_常用的python模块及安装方法
adodb:我们领导推荐的数据库连接组件bsddb3:BerkeleyDB的连接组件Cheetah-1.0:我比较喜欢这个版本的cheeta…
日期:2022-11-24 点赞:878 阅读:9,083
Educational Codeforces Round 11 C. Hard Process 二分
C. Hard Process题目连接:http://www.codeforces.com/contest/660/problem/CDes…
日期:2022-11-24 点赞:807 阅读:5,558
下载Ubuntn 17.04 内核源代码
zengkefu@server1:/usr/src$ uname -aLinux server1 4.10.0-19-generic #21…
日期:2022-11-24 点赞:569 阅读:6,407
可用Active Desktop Calendar V7.86 注册码序列号
可用Active Desktop Calendar V7.86 注册码序列号Name: www.greendown.cn Code: &nb…
日期:2022-11-24 点赞:733 阅读:6,180
Android调用系统相机、自定义相机、处理大图片
Android调用系统相机和自定义相机实例本博文主要是介绍了android上使用相机进行拍照并显示的两种方式,并且由于涉及到要把拍到的照片显…
日期:2022-11-24 点赞:512 阅读:7,817
Struts的使用
一、Struts2的获取  Struts的官方网站为:http://struts.apache.org/  下载完Struts2的jar包,…
日期:2022-11-24 点赞:671 阅读:4,900