精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

深度解析Innodb記錄格式源碼

數據庫 MySQL 數據庫運維
筆者很長時間又不寫東西了,之前已經看過了innodb格式,但現在想研究一下其它方面的東西,突然發現這個又忘了,索性再看看把它寫下來。

 可以通過一個最普遍的插入操作來跟蹤Innodb的記錄格式,因為在插入時,系統得到的是公共的mysql記錄格式record,現在它沒有涉及到任何的存儲引擎,那么這里不管當前這個表對應的存儲引擎是什么,記錄格式是一樣的,對于插入,mysql函數對應的是ha_write_row,具體到Innodb存儲引擎,實際調用的函數是ha_innobase::write_row函數,那么在這里,Innodb首先會將接收到的record記錄轉換為它自己的一個元組tuple,這其實是與record對應的innodb的表示方式,它是一個內存的記錄,邏輯的記錄,那么在系統將其真正的寫入到頁面之前,這條記錄的存在方式都是這個tuple,那么下面主要是從源碼的角度研究Innodb是如何將一個tuple轉換為它的物理的存儲記錄的,主要研究代碼的實現邏輯及記錄的格式。

這里只介紹格式為Compact類型的記錄格式。

實現在某一個頁面插入一個元組(一條記錄)操作的函數是page_cur_tuple_insert,它的參數就是一個dtuple_t*類型的tuple,在這里,它首先要分配一片空間來存儲將要轉換過來的物理記錄,所以這里需要先計算空間的大小,計算方法如下:

1. 首先每條記錄都要包括下面2個部分:REC_N_NEW_EXTRA_BYTES + UT_BITS_IN_BYTES(n_null),前面表示的是這種格式的固定長度的extra部分,這部分用來存儲什么內容后面會給出,后面表示的是所有字段中哪些字段的值是null,當然這里只存儲那些nullable屬于的字段,如果創建表的時候指定是not null的話,這里就不會被存儲,那么這里是用一個位來表示一個字段的null屬性。那么上面這部分被系統代碼命名為extra_size變量值。

2. 統計每一個列中數據的長度,在統計這個信息的時候,又有多種情況,主要分定長字段和變長字段,對于定長字段,它的長度直接就是數據類型的長度,比如int類型的那就是4個字節,rowid列就是6個字節等,沒有其它附加長度。對于變長字段而言,除了數據內容本身的長度外,還需要計算其數據長度的存儲空間,如果字段的字義長度大于255個字節,或者字段的數據類型為BLOB的,那么需要用2個字節來存儲這個字段的長度;如果定義長度小于128個字節,或者小于256個字節,但類型不是BLOB類型的,那么這個字段的數據長度用一個字節來存儲,除上面2種情況之外,都用2個字節來存儲。那么在這一部分中,用來存儲變長字段數據的長度的空間的長度也是被Innodb計算為extra_size的。

所以現在可以知道,一個innodb的記錄包括2個部分,一部分是extra_size,另一部分是數據內容,那么這2部分的總長度就是上面計算出來的結果,這里把它定義為record_size。

接下來,申請空間,進行元組到記錄的轉換工作。

轉換函數為rec_convert_dtuple_to_rec_new,參數有申請好的記錄空間buf,元組和索引的內存結構。

首先這里有一個操作是rec = buf + extra_size,變量rec表示的是數據內容的存儲開始位置。extra_size就是上面計算出來的2個數據部分。

那么真正執行轉換的是接下來調用的rec_convert_dtuple_to_rec_comp函數,下面是其原型:

  1. void 
  2.   
  3.  rec_convert_dtuple_to_rec_comp( 
  4.   
  5.  /*===========================*/ 
  6.   
  7.           rec_t*                          rec,   /*!< in: origin of record */ 
  8.   
  9.           ulint                    extra,        /*!< in: number of bytes to 
  10.   
  11.                                                 reserve between the record 
  12.   
  13.                                                 header and the data payload 
  14.   
  15.                                                 (normally REC_N_NEW_EXTRA_BYTES) */ 
  16.   
  17.           const dict_index_t*  index,        /*!< in: record descriptor */ 
  18.   
  19.           ulint                    status,       /*!< in: status bits of the record */ 
  20.   
  21.           const dfield_t*          fields,        /*!< in: array of data fields */ 
  22.   
  23.           ulint                    n_fields)/*!< in: number of data fields */ 

 

rec表示的是剛才上面計算出來的rec變量,extra表示的是固定長度的REC_N_NEW_EXTRA_BYTES。

 

  1. end = rec; 
  2.   
  3.           nulls = rec - (extra + 1); 
  4.   
  5.           n_null = index->n_nullable; 
  6.   
  7.           lens = nulls - UT_BITS_IN_BYTES(n_null); 
  8.   
  9.           /* clear the SQL-null flags */ 
  10.   
  11.           memset(lens + 1, 0, nulls - lens); 

在這里,這段代碼一下子很難看明白,那么首先這里畫一下記錄存儲格式:

 

 

         |---------------------extra_size-----------------------------------------|---------fields_data------------|

         |--columns_lens---|---null lens----|------fixed_extrasize(5)------|--col1---|---col2---|---col2----|

那么語句nulls = rec - (extra + 1);得到的結果是什么呢?想干什么?因為extra表示的是REC_N_NEW_EXTRA_BYTES,固定長度的fixed_extrasize,rec表示的是圖中col1的開始位置,那么現在可以知道這條語句的結果就是使得nulls指向了前面nulllens的后一個字節的開始位置。那現在我們知道nulls是一個或者多個字節,用來存儲每一個nullable字段的空標志的,那現在為什么要指向這個數組的后一個字節的開始位置呢?一下子很難想明白,不過從后面的代碼中可以知道,寫入nulls是從后面向前面寫的,所以這也理解了為什么指向了后面一個字節的位置了。

那接下來的一個語句lens = nulls - UT_BITS_IN_BYTES(n_null);道理也是一樣的,因為columns_lens正好是在nulllens的前面,那么如果向前跳過null標志的所有空間,則指向的位置lens就是columns_lens的后面一個字節的位置了。在寫入值的時候也是從后面向前面寫。

那最后一個語句memset(lens + 1, 0, nulls - lens);表示的意思就很明白了,因為lens指向的是columns_lens的最后一個字節的開始位置,那么加1就指向了nulls空間的開始位置,nulls – lens表示的是nulls空間的長度。這里是將nulls空間清零。

上面有兩個部分都是從后面向前面填寫數據,那是不是擔心在寫入的時候會不會向前面越界呢?其實是不會的,因為這些都是在前面計算好的,extrasize已經是固定的,包括了nulls和columns_lens的長度的。

上面算是初始化工作,下面就是根據每一個字段來填寫record記錄了,下面一段代碼是處理null信息的,對于每一個字段,都會做下面的處理:

 

  1. if (!(dtype_get_prtype(type) & DATA_NOT_NULL)) { 
  2.                 /* nullable field */ 
  3.                 ut_ad(n_null--); 
  4.                 if (UNIV_UNLIKELY(!(byte) null_mask)) { 
  5.                           nulls--; 
  6.                           null_mask = 1; 
  7.                 } 
  8.                 ut_ad(*nulls < null_mask); 
  9.                 if (dfield_is_null(field)) { 
  10.                        *nulls |= null_mask; 
  11.                        null_mask <<= 1; 
  12.                        continue
  13.               } 
  14.               null_mask <<= 1; 
  15.      } 

從第一行可以看出,要處理這個的條件首先必須是沒有定義not null屬性,所以nulls空間只存儲這些字段的信息。

 

第4行表示的是如果(byte) null_mask)為0時,nulls向前退一個字節,并且將null_mask恢復為1的初值,因為這個值初始值就是1的,可以猜到,如果這個條件滿足了,則說明已經寫入了8個nullable列了,那么需要移向前一個字節繼續寫null信息了,但發現null_mask是int類型的,而nulls是一個字節一個字節的填的,不匹配啊,不過仔細看,判斷條件是(byte) null_mask),所以只要寫入8個之后,這個值就為0了。因為對于每一個字段,都是執行null_mask向左移1個位的,所以移8次之后,低8位就都是0了。

第9行表示的是如果這個列的數據就是null值,那么需要將這個null反映到nulls數組中去,因為null_mask當前的值(其實是1的位置)其實表示的是當前nulls這個字節中正在處理的字段的對應關系,也就是說,如果當前的字段的值為null,那么像第10行所示的,將null_mask或到nulls字節上去,如果不為null,就不管,對應的位的值為0。

所以從這里可以看出,整個nulls空間中的位圖是以從后面向前面的順序來表示所有nullable列的null信息的。

 

  1. if (fixed_len) { 
  2.        } else if (dfield_is_ext(field)) { 
  3.                 *lens-- = (byte) (len >> 8) | 0xc0; 
  4.                 *lens-- = (byte) len; 
  5.        } else { 
  6.                 if (len < 128 || (dtype_get_len(type) < 256 && dtype_get_mtype(type) != DATA_BLOB)) { 
  7.                           *lens-- = (byte) len; 
  8.                 } else { 
  9.                           *lens-- = (byte) (len >> 8) | 0x80; 
  10.                        *lens-- = (byte) len; 
  11.               } 
  12.      } 
  13.      memcpy(end, dfield_get_data(field), len); 
  14.      end += len; 

從第一行可以看出,對于定長數據,只需要將其數據寫入到記錄里面即可,主要處理的是變長數據,第2行表示的是如果長度大于256個字節,或者數據類型為BLOB,則用兩個字節來存儲其長度,低字節存儲(len >> 8) | 0xc0,高字節存儲(byte) len(被截斷)。其它可以直接看出來。

 

到13行,是直接將數據拷到數據存儲空間,用end來表示,存儲完一個字段接著下一個字段,是按照索引定義的順序存儲的。

到這里,一條記錄的邏輯到物理的轉換就完成了,從中也知道了Innodb是如何實現其物理記錄的存儲的。

總結:看innodb的代碼,可以說它的代碼非常優美,非常精練的,所以有些地方很難一下子看懂,需要揣測,體會才能深入的理解。同時有很多地方是直接硬編碼的,這樣導致更加難理解,最好的方式是通過宏將其命名,有助于理解。

原文鏈接:http://www.cnblogs.com/bamboos/archive/2013/03/04/2943160.html

【編輯推薦】

  1. 適合初學者的MySQL學習筆記之庫操作示例
  2. 適合初學者的MySQL學習筆記之表操作示例
  3. 適合初學者的MySQL學習筆記之MySQL管理心得
  4. 適合初學者的MySQL學習筆記之MySQL查詢示例
  5. 適合初學者的MySQL學習筆記之管理員常用操作總結

責任編輯:彭凡 來源: 博客園
相關推薦

2016-12-15 09:44:31

框架Caffe源碼

2022-12-07 08:02:43

Spring流程IOC

2025-09-24 18:39:45

2025-09-26 02:00:55

JDKCPU內存

2024-09-11 09:25:03

Tomcat組件PREP

2022-08-17 07:52:31

Spring循環依賴單例池

2023-05-08 08:11:49

@Component使用場景時序圖

2021-05-26 11:30:24

Java線程池代碼

2024-01-11 12:14:31

Async線程池任務

2025-03-14 12:30:00

Redis RDBRedis數據庫

2025-06-04 02:10:35

`@scope`CSS源碼

2025-11-14 00:00:15

2021-07-03 08:51:30

源碼Netty選擇器

2022-09-27 18:56:28

ArrayList數組源代碼

2025-06-04 08:30:00

seata分布式事務開發

2023-10-10 11:02:00

LSM Tree數據庫

2023-03-27 08:12:40

源碼場景案例

2023-03-06 11:13:20

Spring注解加載

2013-12-09 10:34:12

2023-03-13 08:12:25

@DependsOn源碼場景
點贊
收藏

51CTO技術棧公眾號

久国产精品视频| 国产一区二区久久久| 超碰人人干人人| 精品网站999| 亚洲成人在线网站| 午夜精品福利一区二区| 国产av无码专区亚洲av| 一区二区黄色| 在线观看亚洲区| 欧美日韩一区二区区别是什么| 涩涩视频在线| 亚洲欧美偷拍卡通变态| 欧美久久久久久久| 国产成人三级在线播放| 校园激情久久| 欧美国产日本高清在线| 亚洲精品成人无码| 成人av综合网| 制服丝袜中文字幕亚洲| 成人在线免费播放视频| 午夜av在线播放| 国产午夜精品在线观看| 国产精品theporn88| 最近中文字幕在线观看| 激情综合久久| 久久天天躁狠狠躁老女人| 野花社区视频在线观看| 亚洲天堂av资源在线观看| 欧美视频一区二区三区在线观看| 岛国大片在线播放| 国产原创在线观看| 国产女同性恋一区二区| 久久99精品国产一区二区三区| 国产乱淫av免费| 日韩高清一级片| 88国产精品欧美一区二区三区| 国产在线一卡二卡| 成人亚洲一区二区| 亚洲美女喷白浆| 亚洲成av人片在线观看无| 视频一区中文字幕精品| 欧美精品一级二级三级| 欧美黑人又粗又大又爽免费| 美女露胸视频在线观看| 亚洲图片有声小说| 国产a级黄色大片| 国产鲁鲁视频在线观看特色| 国产精品免费视频观看| 日本一区免费看| 手机福利在线| 久久―日本道色综合久久| 国模精品一区二区三区| 人妻精品一区一区三区蜜桃91| 国产剧情在线观看一区二区| 亚洲aⅴ男人的天堂在线观看| 91国在线视频| 精品一区二区在线播放| 国产日韩欧美综合| 国产精品色综合| 国产真实乱偷精品视频免| 成人免费网站在线| 国产视频www| 国产成人av电影免费在线观看| 亚洲xxxxx电影| 亚洲福利在线观看视频| 成人午夜电影久久影院| 国内成+人亚洲| 婷婷丁香花五月天| 久久日一线二线三线suv| 欧洲av一区| 自拍视频在线| 亚洲精品国产精华液| 欧美中文字幕在线观看视频 | 国产精品免费大片| 亚洲人成在线观| 日本猛少妇色xxxxx免费网站| 99久久亚洲精品蜜臀| 按摩亚洲人久久| 麻豆91精品91久久久| 国产精品视频| 国产精品美女久久久久av超清| 夜夜嗨av禁果av粉嫩avhd| 国产乱国产乱300精品| 精品久久久久亚洲| 国产在线观看黄| 亚洲欧美韩国综合色| 奇米影视亚洲色图| 欧美成人h版| 欧美一区三区二区| 在线观看日韩精品视频| 99精品视频在线观看免费播放| 欧美丰满老妇厨房牲生活| 中日韩精品视频在线观看| 日本亚洲免费观看| 都市激情久久久久久久久久久| 三级视频在线| 亚洲欧美另类图片小说| 欧美精品99久久| 久久99久久久精品欧美| 精品国产1区二区| 久久久久久国产免费a片| 国产精品s色| 国产精品久久久久久av下载红粉 | 伊人久久大香线蕉综合75| 欧美videossex| 欧美日韩在线一区二区| 欧美激情一区二区三区p站| 国产一区国产二区国产三区| 久久91精品国产91久久跳| 依依成人综合网| 紧缚奴在线一区二区三区| 久久精彩视频| 手机电影在线观看| 欧美日韩在线三级| 日本少妇色视频| 欧美日韩蜜桃| 国产日韩亚洲欧美| 国产一区精品| 精品久久久一区二区| 日本美女久久久| 精品理论电影在线| 97精品伊人久久久大香线蕉| 国产婷婷一区二区三区久久| 日本一区二区视频在线| 亚洲午夜精品久久久久久人妖| 国产情侣一区在线| 国产一区二区三区高清在线观看| 国产在线观看免费视频今夜| 狠狠久久亚洲欧美| 亚洲成色最大综合在线| 欧美aa视频| 亚洲国产私拍精品国模在线观看| 超碰手机在线观看| 国内成人免费视频| 亚洲在线不卡| 福利一区二区| 这里只有精品在线观看| 一级一片免费看| 91美女福利视频| 人妻少妇精品无码专区二区| 亚洲一二av| 欧美黄色片视频| 精品黑人一区二区三区国语馆| 国产精品福利影院| 日日噜噜夜夜狠狠| 成人情趣视频网站| 国产精品欧美激情| av电影在线观看网址| 色8久久精品久久久久久蜜| 亚洲第一黄色网址| 亚洲一区二区三区四区五区午夜| 精品国产一区二区三区四区精华| 免费毛片b在线观看| 亚洲精品久久久久久久久久久久| 日本午夜精品理论片a级app发布| 成人av在线播放网站| 久久久久99精品成人片| 国产精品久av福利在线观看| 久久久久久九九九| 色婷婷在线视频| 偷拍与自拍一区| 亚洲av无码成人精品国产| 亚洲专区一区| 日韩性感在线| 在线欧美激情| 久久久久久久久久国产精品| 日韩有码第一页| 欧美特黄级在线| 天堂在线中文视频| 国内精品免费在线观看| 久久香蕉视频网站| 欧美激情网址| 国产精品看片资源| 超碰在线caoporen| 亚洲成人久久久| 天天爽夜夜爽人人爽| 国产精品麻豆视频| 中文字幕亚洲日本| 国产精品毛片在线看| 日日夜夜精品网站| 欧美成人精品午夜一区二区| 国内精品小视频在线观看| 亚洲欧洲成人在线| 欧美三级电影精品| 妺妺窝人体色www聚色窝仙踪| 91在线丨porny丨国产| 狠狠热免费视频| 中文字幕一区二区三区欧美日韩| 国产原创精品| 91九色综合| 久久久久国产精品www| 精品亚洲成a人片在线观看| 欧美日韩高清一区二区三区| 国产精久久久久久| 国产欧美日韩另类视频免费观看| 小日子的在线观看免费第8集| 91久久亚洲| 亚洲欧洲久久| 国产欧美一区二区三区米奇| 国产精品久久久久久久久免费看| 欧美寡妇性猛交xxx免费| 亚洲日韩欧美视频一区| 精品久久在线观看| 91国产成人在线| 国产网站在线看| 亚洲天堂免费看| 免费观看av网站| 福利一区二区在线| 91女神在线观看| 奶水喷射视频一区| 台湾无码一区二区| 91视频精品| 日本免费高清不卡| 国产精品午夜av| 川上优av一区二区线观看| 欧美舌奴丨vk视频| 97视频在线观看视频免费视频 | av在线网站观看| 国产成人在线色| 日韩欧美亚洲另类| 日韩精品亚洲一区二区三区免费| 91免费黄视频| 亚洲欧美综合| 2021狠狠干| 97精品一区| 亚洲成色www久久网站| 在线日韩网站| 久久国产精品精品国产色婷婷| 欧美电影在线观看一区| 国产精品日韩欧美综合| 欧美精品总汇| 日韩免费黄色av| 欧美电影免费观看高清完整| 66m—66摸成人免费视频| 欧美xxxx黑人又粗又长| 欧美精品日韩三级| 快射视频在线观看| 日韩在线观看免费高清完整版| 免费一级毛片在线观看| 亚洲男人天堂网站| 日韩电影网址| 亚洲人av在线影院| 邻家有女韩剧在线观看国语| 亚洲色在线视频| 精品亚洲综合| 国产一区二区三区在线看| 青青久在线视频| 亚洲色图综合网| 成年人在线免费观看| 亚洲美女在线视频| 黄色国产在线| 中文字幕亚洲二区| 91亚洲精选| 久久影院在线观看| 天堂av最新在线| 久久久久久久久网站| 91视频欧美| 2021久久精品国产99国产精品| 狠狠躁少妇一区二区三区| 欧美一级免费看| 欧美成人精品一区二区男人小说| 国产精品久久久久久久久| 国产精品99精品一区二区三区∴| 成人观看高清在线观看免费| 日本一区二区乱| 精品一区二区久久久久久久网站| 欧美天堂影院| 欧美在线3区| 三级电影一区| 国产在线无码精品| 亚洲激情不卡| 黄色一级一级片| 久久99久久久久| 激情av中文字幕| 久久香蕉国产线看观看99| www成人啪啪18软件| 亚洲精品第一国产综合野| 日韩免费观看一区二区| 一本久久精品一区二区| 一级特黄aaaaaa大片| 日韩免费高清视频| 青青操在线视频| 麻豆成人在线看| 蜜桃视频m3u8在线观看| 国产精品视频久久久| 91亚洲无吗| 日韩免费电影一区二区三区| 女主播福利一区| 日本wwww视频| 国产剧情一区二区| 国产亚洲无码精品| 亚洲日本一区二区| 欧美精品韩国精品| 91精品国产综合久久小美女| 天天干在线观看| 久久手机免费视频| 亚洲天堂资源| 97国产超碰| 日本女优一区| 99精品人妻少妇一区二区| 国产乱国产乱300精品| 变态另类ts人妖一区二区| 一区二区三区四区蜜桃| 亚洲av人无码激艳猛片服务器| 精品日韩欧美在线| 在线观看完整版免费| 91精品国产高清久久久久久久久| 在线观看亚洲精品福利片| 欧美久久综合性欧美| 亚洲大片在线| 一级 黄 色 片一| 中文字幕高清不卡| 麻豆成人免费视频| 精品久久久久av影院 | 国产精品美女一区二区| 日韩成人av毛片| 日韩小视频在线观看专区| av男人的天堂在线| 日韩女在线观看| 久久a爱视频| 妺妺窝人体色www看人体| 久久er99热精品一区二区| 欧美多人猛交狂配| 午夜精品视频一区| 成人毛片在线精品国产| 久久视频免费在线播放| 黄色精品视频网站| 色噜噜狠狠一区二区三区| 久久亚洲国产精品一区二区| 一级特级黄色片| 亚洲第一av色| 亚洲精品视频专区| 欧美老妇交乱视频| 在线免费成人| 黄色网络在线观看| 国产做a爰片久久毛片| 国精产品视频一二二区| 欧美在线免费播放| 精品无人乱码| 国产精品国产三级国产专播精品人 | 黄色网页网址在线免费| 国产男人精品视频| 成人激情免费视频| 精品999在线| 国产精品久久久久久久久久免费看| 五月天中文字幕| 中文字幕一区日韩电影| 香蕉久久一区| 久久久成人精品一区二区三区| 国模少妇一区二区三区| 永久久久久久久| 精品久久久三级丝袜| av在线播放资源| 免费一区二区三区| 视频在线观看国产精品| 天天舔天天操天天干| 欧美色国产精品| 免费看a在线观看| 97免费高清电视剧观看| 午夜精品久久| 中文字幕在线视频播放| 狠狠做深爱婷婷久久综合一区| 日本电影一区二区在线观看| 国产精品v片在线观看不卡| 欧美激情国产在线| 日本少妇xxx| 欧美视频在线免费看| av在线播放av| 99国产视频在线| 国产精品久久777777毛茸茸| 九一在线免费观看| 欧美一区二区三区免费在线看| 欧美人动性xxxxz0oz| 久久亚洲国产精品日日av夜夜| 天堂精品中文字幕在线| 欧美日韩生活片| 日韩精品中文字幕在线一区| 亚洲欧洲自拍| 天天爱天天做天天操| bt7086福利一区国产| 国产成人精品免费看在线播放| porn视频在线观看| 国产精品丝袜久久久久久高清| 亚洲草久电影| 国产又黄又粗又猛又爽的视频 | 99色精品视频| 国产精品免费视频一区| 成人免费观看在线视频| 人九九综合九九宗合| 在线中文一区| 免费成人深夜夜行p站| 7777精品伊人久久久大香线蕉经典版下载 | 91视频一区二区| 97成人在线观看| 91精品国产91久久久| 欧美高清视频在线观看mv| 亚洲天堂成人av| 在线播放日韩导航| 123区在线| 日韩视频在线观看视频|