柔性数组(Redis源码学习)
柔性数组(Redis源码学习)
1. 问题背景
在阅读Redis源码中的字符串有如下结构,在sizeof(struct sdshdr)得到结果为8,在后续内存申请和计算中也用到。其实在工作中有遇到过这种 struct结构 + 应用数据的情况,但没有意识到自己使用的是柔性数组,在学习阅读Redis代码中,遇到该方法,就特总结记录之。
/* * 类型别名,用于指向 sdshdr 的 buf 属性 */
typedef char * sds;
/* * 保存字符串对象的结构 */
struct sdshdr {
// buf 中已占用空间的长度
int len;
// buf 中剩余可用空间的长度
int free;
// 数据空间
char buf[];
};
2. 柔性数组
柔性数组(flexible array member)也叫伸缩性数组成员,这种结构产生与对动态结构体的去求。在日常编程中,有时需要在结构体中存放一个长度是动态的字符串(也可能是其他数据类型)。
一般的做法,是在结构体中定义一个指针成员,这个指针成员指向该字符串所在的动态内存空间。在通常情况下,如果想要高效的利用内存,那么在结构体内部定义静态的数组是非常浪费的行为。其实柔性数组的想法和动态数组的想法是一样的。
柔性数组用来在结构体中存放一个长度动态的字符串。
本文基于redis 的sds.c源码,进行简单编码验证测试,其实这种柔性数组,在工作中用到过,但是没有意识到这是柔性数组。
上述struct sdshdr结构中,要注意:最后一个变量 buf 数组中,没有长度,这和自己遇到的正常的使用方式不一样,新的知识点
这种用法是C语言中的柔性数组,上面 的sizeof(sdshdr )结果是8,即后面的buf不占空间,只是一个符号,测试上面sdshdr结果如下:
int main(int argc,char **argv){
struct sdshdr t;
printf("int len:%d\n",sizeof(int));
printf("sdshdr len:%d\n",sizeof(struct sdshdr));
printf("Address:\n");
printf("t\t %p\n", &t);
printf("t.len\t %p\n", &(t.len));
printf("t.free\t %p\n", &(t.free));
printf("t.buf\t %p\n", &(t.buf));
return 0;
}
RHEL6.9上执行上面代码块得到结果如下:
$ ./sdshdr
int len:4
sdshdr len:8
Address:
t 0x7fff9572fa50
t.len 0x7fff9572fa50
t.free 0x7fff9572fa54
t.buf 0x7fff9572fa58
可以看到 t.buf 是该结构的最后的地址,是最后一个点,简单图示如下:

如果后续再malloc相关的内存,则就会在t.buf后面连续,简单编写代码进行验证。要加入对应的sds.h文件,或者直接将结构定义在main函数之前。
int main(int argc,char **argv){
struct sdshdr t;
printf("int len:%d\n",sizeof(int));
printf("sdshdr len:%d\n",sizeof(struct sdshdr));
printf("Address:\n");
printf("t\t %p\n", &t);
printf("t.len\t %p\n", &(t.len));
printf("t.free\t %p\n", &(t.free));
printf("t.buf\t %p\n", &(t.buf));
printf("sizeof(char):\t %d\n", sizeof(char));
struct sdshdr *p=(struct sdshdr*)malloc(sizeof(struct sdshdr) + sizeof(char)*8);
printf("After malloc the struct's size is %d\n",sizeof(struct sdshdr));
printf("Address:\n");
printf("p\t %p\n", p);
printf("p->len\t %p\n", &(p->len));
printf("p->free\t %p\n", &(p->free));
printf("p->buf\t %p,sizeof(p):%d\n", &(p->buf),sizeof(p));
memset(p,0,sizeof(struct sdshdr) + sizeof(char)*8);
char *str="Hello";
memcpy(p->buf,str,strlen(str));
printf("p->buf:%s\n",p->buf);
char *str1="HelloWorldttttttt";
memcpy(p->buf,str1,sizeof(char)*8-1);
printf("p->buf:%s\n",p->buf);
printf("strlen(p->buf):%d\n",strlen(p->buf));
return 0;
}
上述代码进行编译,获得可执行文件,执行结果如下:
$ ./sdshdr
int len:4
sdshdr len:8
Address:
t 0x7ffea0a8c420
t.len 0x7ffea0a8c420
t.free 0x7ffea0a8c424
t.buf 0x7ffea0a8c428
sizeof(char): 1
After malloc the struct's size is 8
Address:
p 0x1bc3010
p->len 0x1bc3010
p->free 0x1bc3014
p->buf 0x1bc3018,sizeof(p):8
p->buf:Hello
p->buf:HelloWo
strlen(p->buf):7
$
## 3. 使用方法
从C99开始便支持了不完整类型实现柔性数组成员。为什么使用不完整类型呢?
```C language
int a[] = {10};
看到这个声明语句,我们发现a[]其实就是个数组记号,不完整类型,由于赋值语句,所以在编译时便确定了数组的大小,是一个完整的数组类型。
在结构体中便利用不完整类型在运行对动态的数组进行指明。
C99标准的定义如下:
struct Test{
int a;
char p[]; // 不只是char类型,其他类型同样也是可以
}
由于声明内存连续性的关系,柔性数组成员必须定义在结构体的最后一个,并且不能是唯一的成员。
我们再来看一看整个结构体(包含数组内存的分布情况),进行简单编码验证。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
typedef struct Test
{
int a;
char p[];
} Test;
int main()
{
Test *t=(Test*)malloc(sizeof(Test)+sizeof(char)*(10+1));
printf("sizeof(int):%d,sizeof(Test):%d\n",sizeof(int),sizeof(Test));
strcpy(t->p,"hello");
printf("t->p:%s\n", (t->p));
printf("Address:\n");
printf("t\t %p\n", t);
printf("t.a\t %p\n", &(t->a));
printf("t.p\t %p\n", (t->p));
free(t); //只需要释放一次内存
return 0;
}
在linux上的执行结果如下:
$ ./sdshdr
sizeof(int):4,sizeof(Test):4
t->p:hello
Address:
t 0x7e0010
t.a 0x7e0010
t.p 0x7e0014
4. 小结
- 在结构体中存放一个长度是动态数据类型时,可以考虑到柔性数组。
- 一般做法,是在结构体中定义一个指针成员,这个指针成员指向所在的动态内存空间。
- 该指针成员,不占结构体空间,只是一个符号。
- 柔性数组成员必须定义在结构体的最后一个,并且不能是唯一的成员。
5. 参考文献
https://www.cnblogs.com/davygeek/p/5748852.html
https://blog.csdn.net/qq_40477151/article/details/78905567
https://www.cnblogs.com/pluviophile/p/7571410.html
本人才疏学浅,参考网络文章及代码验证,如有错误不当之处,请批评指正,如有侵权,请立即联系我进行删除。
如果能为您带来一点点帮助,那将是我的荣幸,多谢您关注和转发推荐,谢谢!

柔性数组(Redis源码学习)的更多相关文章
- Redis源码学习:字符串
Redis源码学习:字符串 1.初识SDS 1.1 SDS定义 Redis定义了一个叫做sdshdr(SDS or simple dynamic string)的数据结构.SDS不仅用于 保存字符串, ...
- Redis源码学习:Lua脚本
Redis源码学习:Lua脚本 1.Sublime Text配置 我是在Win7下,用Sublime Text + Cygwin开发的,配置方法请参考<Sublime Text 3下C/C++开 ...
- redis源码学习之slowlog
目录 背景 环境说明 redis执行命令流程 记录slowlog源码分析 制造一条slowlog slowlog分析 1.slowlog如何开启 2.slowlog数量限制 3.slowlog中的耗时 ...
- __sync_fetch_and_add函数(Redis源码学习)
__sync_fetch_and_add函数(Redis源码学习) 在学习redis-3.0源码中的sds文件时,看到里面有如下的C代码,之前从未接触过,所以为了全面学习redis源码,追根溯源,学习 ...
- redis源码学习之lua执行原理
聊聊redis执行lua原理 从一次面试场景说起 "看你简历上写的精通redis" "额,还可以啦" "那你说说redis执行lua脚本的原理&q ...
- redis源码学习之工作流程初探
目录 背景 环境准备 下载redis源码 下载Visual Studio Visual Studio打开redis源码 启动过程分析 调用关系图 事件循环分析 工作模型 代码分析 动画演示 网络模块 ...
- redis源码学习-skiplist
1.初步认识跳跃表 图中所示,跳跃表与普通链表的区别在于,每一个节点可以有多个后置节点,图中是一个4层的跳跃表 第0层: head->3->6->7->9->12-> ...
- Redis源码学习1-sds.c
https://github.com/huangz1990/redis-3.0-annotated/blob/unstable/src/sds.c#L120 /* SDSLib, A C dynami ...
- redis源码学习_字典
redis中字典有以下要点: (1)它就是一个键值对,对于hash冲突的处理采用了头插法的链式存储来解决. (2)对rehash,扩展就是取第一个大于等于used * 2的2 ^ n的数作为新的has ...
随机推荐
- window 共享打印机
https://www.zhihu.com/question/20653708 https://h30471.www3.hp.com/t5/da-yin-ji-yu-sao-miao-yi-de-an ...
- Linux中级之ansible概念及hoc命令行调用模式
一.Ansible简介 ansible是新出现的开源的自动化运维工具,基于Python开发,集合了众多运维工具(puppet.cfengine.chef.func.fabric)的优点,实现了批量系统 ...
- INFJ名言
财富是由什么构成的? 按世俗的观点,就是占有金钱和财宝. 但如果我们用除金钱之外的其他方式来衡量财富, 那么许多在物质上匮乏的人在精神上却是富有的, 许多在物质上富有的人在精神上却是匮乏的. The ...
- RIP OSPF 等路由协议属于计算机网络分层中的哪一层
RIP基于UDP,BGP基于TCP,OSPF EGP基于IP 在TCP/IP协议栈中定义的路由协议用于发现和维护前往目的地的最短路径.可以认为它们不属于网络层协议(注意,是用based on,而不是实 ...
- JavaScript正则表达式(深度)(Day_14)
忘不掉的是回忆,继续的是生活,错过的,就当是路过. 简介 正则表达式是用于匹配字符串中字符组合的模式.在 JavaScript中,正则表达式也是对象. 这些模式被用于 RegExp 的 exec 和 ...
- rocketmq常见问题及使用 新手篇
一 部署阶段 1.启动命令 nameServer启动:nohup sh bin/mqnamesrv -n ip地址:9876 & broker启动:nohup sh bin/mqbroker ...
- openresty 学习笔记番外篇:python的一些扩展库
openresty 学习笔记番外篇:python的一些扩展库 要写一个可以使用的python程序还需要比如日志输出,读取配置文件,作为守护进程运行等 读取配置文件 使用自带的ConfigParser模 ...
- 激光雷达Lidar Architecture and Lidar Design(上)
激光雷达Lidar Architecture and Lidar Design(上) 介绍 激光雷达结构: 基本条件 构型和基本布置 激光雷达设计: 基本思想和基本原则 总结 介绍 激光雷达结构是激光 ...
- FFmpeg扩展开发
FFmpeg扩展开发 对FFmpeg RTMP/FLV部分做了扩展,用于支持H.265. 针对<video_file_format_spec_v10_1> VIDEODATA部分扩展如下: ...
- 编译器架构Compiler Architecture(下)
编译器架构Compiler Architecture(下) Combining Scanning and Parsing 实际上没有必要将扫描(词法分析/标记化)与解析(语法分析/树生成)分开.基于P ...