找回密码
 注册
搜索
热搜: 回贴
微赢网络技术论坛 门户 站长资讯 业界 查看内容

innodb动态数组内部实现

2009-12-25 00:25| 发布者: admin| 查看: 78| 评论: 0|原作者: 夙玉

1. 动态数组概述

动态数组涉及的文件是innodb存储引擎的三个文件:dyn0dyn.h、dyn0dyn.ic以及dyn0dyn.c。

这是一个基本的组件功能,是作为一个动态的虚拟线性数组。数组的基本元素是byte。动态数组dyn主要用来存放mtr的锁定信息以及log。Dyn在实现上,如果block需要分裂节点,则会使用一个内存堆。每个blok块存储数据的数据字段的长度是固定的(默认值是512),但是不一定会完全用完。假设需要存储的数据项的尺寸大于数据块时,该数据项被分拆,这种情况主要用于log的缓冲。



2. 数据结构

typedef struct dyn_block_struct dyn_block_t;

typedef dyn_block_t dyn_array_t;

#define DYN_ARRAY_DATA_SIZE 512



struct dyn_block_struct{

mem_heap_t* heap;

ulint used;

byte data[DYN_ARRAY_DATA_SIZE];

UT_LIST_BASE_NODE_T(dyn_block_t) base;

UT_LIST_NODE_T(dyn_block_t) list;

#ifdef UNIV_DEBUG

ulint buf_end;

ulint magic_n;

#endif

};



//下面两个是公共宏定义,见ut0lst.h

#define UT_LIST_BASE_NODE_T(TYPE)\

struct {\

ulint count; /* count of nodes in list */\

TYPE * start; /* pointer to list start, NULL if empty */\

TYPE * end; /* pointer to list end, NULL if empty */\

}\



#define UT_LIST_NODE_T(TYPE)\

struct {\

TYPE * prev; /* pointer to the previous node,\

NULL if start of list */\

TYPE * next; /* pointer to next node, NULL if end of list */\

}\



字段解释:

1) heap

从字面上理解,heap是内存堆的意思。从上面的结构体中,我们可以看出,dyn_array_t就是dyn_block_struct。这个结构体里面的字段date用来存放数据,used显示已经使用的字节数量,假设这时候还要存储一个长度为x的数据,这时候data里面已经存储不了,所以就需要再分配一个新的block节点。那么分配结构所需要的内存从哪里来了,就从第一个节点里面的heap里面分配。

这里面,我们还需要注意一点。虽然数据结构用的是同一个dyn_block_struct,但是我们称第一个节点为arr,表明这个是动态数组的头节点。其它的节点,我们称为block节点。

我们这里面先来看看,创立dyn的函数是怎么实现的:

UNIV_INLINE

dyn_array_t*

dyn_array_create(

/*=============*/

/* out: initialized dyn array */

dyn_array_t* arr) /* in: pointer to a memory buffer of

size sizeof(dyn_array_t) */

{

ut_ad(arr);

ut_ad(DYN_ARRAY_DATA_SIZE < DYN_BLOCK_FULL_FLAG);



arr->heap = NULL;

arr->used = 0;



#ifdef UNIV_DEBUG

arr->buf_end = 0;

arr->magic_n = DYN_BLOCK_MAGIC_N;

#endif

return(arr);

}

其中的ud_ad是属于宏判断,相当于assert。这个我们暂时不表。这个函数是在mtr创建的时候来调用的,在调用的时候已经分配了arr指针。

在dyn_array_create函数中,系统将heap字段设置为NULL,Used设置为0。Buf_end和magic_n是调试的时候使用的,每个结构体定义都会唯一的对应一个magic_n值。这样在调试的时候,就可以快速进行问题的跟踪。

下面一个问题是,既然heap在起初的设置为NULL,那么什么时候分配给它值?是第一次分配block成员的时候?还是每次分配block成员的时候?

我们这里暂不分析如何插入,我们先看看,当dyn数组已经数据不够用的时候,我们怎么给这个arr增加一个新的block。代码如下:

dyn_block_t*

dyn_array_add_block(

/*================*/

/* out: created block */

dyn_array_t* arr) /* in: dyn array */

{

mem_heap_t* heap;

dyn_block_t* block;



ut_ad(arr);

ut_ad(arr->magic_n == DYN_BLOCK_MAGIC_N);



if (arr->heap == NULL) {

UT_LIST_INIT(arr->base);

UT_LIST_ADD_FIRST(list, arr->base, arr);



arr->heap = mem_heap_create(sizeof(dyn_block_t));

}



block = dyn_array_get_last_block(arr);

block->used = block->used | DYN_BLOCK_FULL_FLAG;



heap = arr->heap;



block = mem_heap_alloc(heap, sizeof(dyn_block_t));



block->used = 0;



UT_LIST_ADD_LAST(list, arr->base, block);



return(block);

}

代码中,我们可以看出,第一次增加block的时候,因为“arr->heap == NULL”条件为真,就会创建一个heap堆,在以后的再增加block时,就会使用的该堆直接分配。所以,我们可以得知,对于dyn动态数组,只有首节点的heap才是有效的。

好,那我们通过图形来看下,增加新节点之后的图形变化。



heap

used

data

base

list

……

图1.只有一个节点

NULL

指示data字段被使用的字节数字。

这是个线性链表的头节点。此时该值为NULL。只在首节点中有效。存储指针。

指向前一个、下一个节点指针。此时为NULL。

最新评论

QQ|小黑屋|最新主题|手机版|微赢网络技术论坛 ( 苏ICP备08020429号 )

GMT+8, 2024-9-29 23:32 , Processed in 0.218978 second(s), 12 queries , Gzip On, MemCache On.

Powered by Discuz! X3.5

© 2001-2023 Discuz! Team.

返回顶部