日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > 数据库 >内容正文

数据库

Redis数据持久化机制AOF原理分析一---转

發布時間:2025/4/5 数据库 27 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Redis数据持久化机制AOF原理分析一---转 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

http://blog.csdn.net/acceptedxukai/article/details/18136903

http://blog.csdn.net/acceptedxukai/article/details/18181563

本文所引用的源碼全部來自Redis2.8.2版本。

Redis AOF數據持久化機制的實現相關代碼是redis.c, redis.h, aof.c, bio.c, rio.c, config.c

在閱讀本文之前請先閱讀Redis數據持久化機制AOF原理分析之配置詳解文章,了解AOF相關參數的解析,文章鏈接

http://blog.csdn.net/acceptedxukai/article/details/18135219

轉載請注明,文章出自http://blog.csdn.net/acceptedxukai/article/details/18136903

下面將介紹AOF數據持久化機制的實現

?

Server啟動加載AOF文件數據

?

Server啟動加載AOF文件數據的執行步驟為:main() -> initServerConfig() -> loadServerConfig() -> initServer() -> loadDataFromDisk()。initServerConfig()主要為初始化默認的AOF參數配置;loadServerConfig()加載配置文件redis.conf中AOF的參數配置,覆蓋Server的默認AOF參數配置,如果配置appendonly on,那么AOF數據持久化功能將被激活,server.aof_state參數被設置為REDIS_AOF_ON;loadDataFromDisk()判斷server.aof_state == REDIS_AOF_ON,結果為True就調用loadAppendOnlyFile函數加載AOF文件中的數據,加載的方法就是讀取AOF文件中數據,由于AOF文件中存儲的數據與客戶端發送的請求格式相同完全符合Redis的通信協議,因此Server創建偽客戶端fakeClient,將解析后的AOF文件數據像客戶端請求一樣調用各種指令,cmd->proc(fakeClient),將AOF文件中的數據重現到Redis Server數據庫中。

?

[cpp]?view plaincopyprint?
  • /*?Function?called?at?startup?to?load?RDB?or?AOF?file?in?memory.?*/??
  • void?loadDataFromDisk(void)?{??
  • ????long?long?start?=?ustime();??
  • ????if?(server.aof_state?==?REDIS_AOF_ON)?{??
  • ????????if?(loadAppendOnlyFile(server.aof_filename)?==?REDIS_OK)??
  • ????????????redisLog(REDIS_NOTICE,"DB?loaded?from?append?only?file:?%.3f?seconds",(float)(ustime()-start)/1000000);??
  • ????}?else?{??
  • ????????if?(rdbLoad(server.rdb_filename)?==?REDIS_OK)?{??
  • ????????????redisLog(REDIS_NOTICE,"DB?loaded?from?disk:?%.3f?seconds",??
  • ????????????????(float)(ustime()-start)/1000000);??
  • ????????}?else?if?(errno?!=?ENOENT)?{??
  • ????????????redisLog(REDIS_WARNING,"Fatal?error?loading?the?DB:?%s.?Exiting.",strerror(errno));??
  • ????????????exit(1);??
  • ????????}??
  • ????}??
  • }??
  • Server首先判斷加載AOF文件是因為AOF文件中的數據要比RDB文件中的數據要新。

    ?

    ?

    [cpp]?view plaincopyprint?
  • int?loadAppendOnlyFile(char?*filename)?{??
  • ????struct?redisClient?*fakeClient;??
  • ????FILE?*fp?=?fopen(filename,"r");??
  • ????struct?redis_stat?sb;??
  • ????int?old_aof_state?=?server.aof_state;??
  • ????long?loops?=?0;??
  • ??
  • ????//redis_fstat就是fstat64函數,通過fileno(fp)得到文件描述符,獲取文件的狀態存儲于sb中,??
  • ????//具體可以參考stat函數,st_size就是文件的字節數??
  • ????if?(fp?&&?redis_fstat(fileno(fp),&sb)?!=?-1?&&?sb.st_size?==?0)?{??
  • ????????server.aof_current_size?=?0;??
  • ????????fclose(fp);??
  • ????????return?REDIS_ERR;??
  • ????}??
  • ??
  • ????if?(fp?==?NULL)?{//打開文件失敗??
  • ????????redisLog(REDIS_WARNING,"Fatal?error:?can't?open?the?append?log?file?for?reading:?%s",strerror(errno));??
  • ????????exit(1);??
  • ????}??
  • ??
  • ????/*?Temporarily?disable?AOF,?to?prevent?EXEC?from?feeding?a?MULTI?
  • ?????*?to?the?same?file?we're?about?to?read.?*/??
  • ????server.aof_state?=?REDIS_AOF_OFF;??
  • ??
  • ????fakeClient?=?createFakeClient();?//建立偽終端??
  • ????startLoading(fp);?//?定義于?rdb.c?,更新服務器的載入狀態??
  • ??
  • ????while(1)?{??
  • ????????int?argc,?j;??
  • ????????unsigned?long?len;??
  • ????????robj?**argv;??
  • ????????char?buf[128];??
  • ????????sds?argsds;??
  • ????????struct?redisCommand?*cmd;??
  • ??
  • ????????/*?Serve?the?clients?from?time?to?time?*/??
  • ????????//?有間隔地處理外部請求,ftello()函數得到文件的當前位置,返回值為long??
  • ????????if?(!(loops++?%?1000))?{??
  • ????????????loadingProgress(ftello(fp));//保存aof文件讀取的位置,ftellno(fp)獲取文件當前位置??
  • ????????????aeProcessEvents(server.el,?AE_FILE_EVENTS|AE_DONT_WAIT);//處理事件??
  • ????????}??
  • ????????//按行讀取AOF數據??
  • ????????if?(fgets(buf,sizeof(buf),fp)?==?NULL)?{??
  • ????????????if?(feof(fp))//達到文件尾EOF??
  • ????????????????break;??
  • ????????????else??
  • ????????????????goto?readerr;??
  • ????????}??
  • ????????//讀取AOF文件中的命令,依照Redis的協議處理??
  • ????????if?(buf[0]?!=?'*')?goto?fmterr;??
  • ????????argc?=?atoi(buf+1);//參數個數??
  • ????????if?(argc?<?1)?goto?fmterr;??
  • ??
  • ????????argv?=?zmalloc(sizeof(robj*)*argc);//參數值??
  • ????????for?(j?=?0;?j?<?argc;?j++)?{??
  • ????????????if?(fgets(buf,sizeof(buf),fp)?==?NULL)?goto?readerr;??
  • ????????????if?(buf[0]?!=?'$')?goto?fmterr;??
  • ????????????len?=?strtol(buf+1,NULL,10);//每個bulk的長度??
  • ????????????argsds?=?sdsnewlen(NULL,len);//新建一個空sds??
  • ????????????//按照bulk的長度讀取??
  • ????????????if?(len?&&?fread(argsds,len,1,fp)?==?0)?goto?fmterr;??
  • ????????????argv[j]?=?createObject(REDIS_STRING,argsds);??
  • ????????????if?(fread(buf,2,1,fp)?==?0)?goto?fmterr;?/*?discard?CRLF?跳過\r\n*/??
  • ????????}??
  • ??
  • ????????/*?Command?lookup?*/??
  • ????????cmd?=?lookupCommand(argv[0]->ptr);??
  • ????????if?(!cmd)?{??
  • ????????????redisLog(REDIS_WARNING,"Unknown?command?'%s'?reading?the?append?only?file",?(char*)argv[0]->ptr);??
  • ????????????exit(1);??
  • ????????}??
  • ????????/*?Run?the?command?in?the?context?of?a?fake?client?*/??
  • ????????fakeClient->argc?=?argc;??
  • ????????fakeClient->argv?=?argv;??
  • ????????cmd->proc(fakeClient);//執行命令??
  • ??
  • ????????/*?The?fake?client?should?not?have?a?reply?*/??
  • ????????redisAssert(fakeClient->bufpos?==?0?&&?listLength(fakeClient->reply)?==?0);??
  • ????????/*?The?fake?client?should?never?get?blocked?*/??
  • ????????redisAssert((fakeClient->flags?&?REDIS_BLOCKED)?==?0);??
  • ??
  • ????????/*?Clean?up.?Command?code?may?have?changed?argv/argc?so?we?use?the?
  • ?????????*?argv/argc?of?the?client?instead?of?the?local?variables.?*/??
  • ????????for?(j?=?0;?j?<?fakeClient->argc;?j++)??
  • ????????????decrRefCount(fakeClient->argv[j]);??
  • ????????zfree(fakeClient->argv);??
  • ????}??
  • ??
  • ????/*?This?point?can?only?be?reached?when?EOF?is?reached?without?errors.?
  • ?????*?If?the?client?is?in?the?middle?of?a?MULTI/EXEC,?log?error?and?quit.?*/??
  • ????if?(fakeClient->flags?&?REDIS_MULTI)?goto?readerr;??
  • ??
  • ????fclose(fp);??
  • ????freeFakeClient(fakeClient);??
  • ????server.aof_state?=?old_aof_state;??
  • ????stopLoading();??
  • ????aofUpdateCurrentSize();?//更新server.aof_current_size,AOF文件大小??
  • ????server.aof_rewrite_base_size?=?server.aof_current_size;??
  • ????return?REDIS_OK;??
  • ????…………??
  • }??
  • 在前面一篇關于AOF參數配置的博客遺留了一個問題,server.aof_current_size參數的初始化,下面解決這個疑問。

    ?

    ?

    [cpp]?view plaincopyprint?
  • void?aofUpdateCurrentSize(void)?{??
  • ????struct?redis_stat?sb;??
  • ??
  • ????if?(redis_fstat(server.aof_fd,&sb)?==?-1)?{??
  • ????????redisLog(REDIS_WARNING,"Unable?to?obtain?the?AOF?file?length.?stat:?%s",??
  • ????????????strerror(errno));??
  • ????}?else?{??
  • ????????server.aof_current_size?=?sb.st_size;??
  • ????}??
  • }??
  • redis_fstat是作者對Linux中fstat64函數的重命名,該還是就是獲取文件相關的參數信息,具體可以Google之,sb.st_size就是當前AOF文件的大小。這里需要知道server.aof_fd即AOF文件描述符,該參數的初始化在initServer()函數中

    ?

    ?

    [cpp]?view plaincopyprint?
  • /*?Open?the?AOF?file?if?needed.?*/??
  • ????if?(server.aof_state?==?REDIS_AOF_ON)?{??
  • ????????server.aof_fd?=?open(server.aof_filename,O_WRONLY|O_APPEND|O_CREAT,0644);??
  • ????????if?(server.aof_fd?==?-1)?{??
  • ????????????redisLog(REDIS_WARNING,?"Can't?open?the?append-only?file:?%s",strerror(errno));??
  • ????????????exit(1);??
  • ????????}??
  • ????}??
  • ?

    ?

    至此,Redis Server啟動加載硬盤中AOF文件數據的操作就成功結束了。

    ?

    ?

    Server數據庫產生新數據如何持久化到硬盤


    當客戶端執行Set等修改數據庫中字段的指令時就會造成Server數據庫中數據被修改,這些修改的數據應該被實時更新到AOF文件中,并且也要按照一定的fsync機制刷新到硬盤中,保證數據不會丟失。

    ?

    在上一篇博客中,提到了三種fsync方式:appendfsync always,?appendfsync everysec,?appendfsync no. 具體體現在server.aof_fsync參數中。

    首先看當客戶端請求的指令造成數據被修改,Redis是如何將修改數據的指令添加到server.aof_buf中的。

    call() ->?propagate() ->?feedAppendOnlyFile(),call()函數判斷執行指令后是否造成數據被修改。

    feedAppendOnlyFile函數首先會判斷Server是否開啟了AOF,如果開啟AOF,那么根據Redis通訊協議將修改數據的指令重現成請求的字符串,注意在超時設置的處理方式,接著將字符串append到server.aof_buf中即可。該函數最后兩行代碼需要注意,這才是重點,如果server.aof_child_pid != -1那么表明此時Server正在重寫rewrite AOF文件,需要將被修改的數據追加到server.aof_rewrite_buf_blocks鏈表中,等待rewrite結束后,追加到AOF文件中。具體見下面代碼的注釋。

    ?

    [cpp]?view plaincopyprint?
  • /*?Propagate?the?specified?command?(in?the?context?of?the?specified?database?id)?
  • ?*?to?AOF?and?Slaves.?
  • ?*?
  • ?*?flags?are?an?xor?between:?
  • ?*?+?REDIS_PROPAGATE_NONE?(no?propagation?of?command?at?all)?
  • ?*?+?REDIS_PROPAGATE_AOF?(propagate?into?the?AOF?file?if?is?enabled)?
  • ?*?+?REDIS_PROPAGATE_REPL?(propagate?into?the?replication?link)?
  • ?*/??
  • void?propagate(struct?redisCommand?*cmd,?int?dbid,?robj?**argv,?int?argc,??
  • ???????????????int?flags)??
  • {??
  • ????//將cmd指令變動的數據追加到AOF文件中??
  • ????if?(server.aof_state?!=?REDIS_AOF_OFF?&&?flags?&?REDIS_PROPAGATE_AOF)??
  • ????????feedAppendOnlyFile(cmd,dbid,argv,argc);??
  • ????if?(flags?&?REDIS_PROPAGATE_REPL)??
  • ????????replicationFeedSlaves(server.slaves,dbid,argv,argc);??
  • }??
  • [cpp]?view plaincopyprint?
  • //cmd指令修改了數據,先將更新的數據寫到server.aof_buf中??
  • void?feedAppendOnlyFile(struct?redisCommand?*cmd,?int?dictid,?robj?**argv,?int?argc)?{??
  • ????sds?buf?=?sdsempty();??
  • ????robj?*tmpargv[3];??
  • ??
  • ????/*?The?DB?this?command?was?targeting?is?not?the?same?as?the?last?command?
  • ?????*?we?appendend.?To?issue?a?SELECT?command?is?needed.?*/??
  • ????//?當前?db?不是指定的?aof?db,通過創建?SELECT?命令來切換數據庫??
  • ????if?(dictid?!=?server.aof_selected_db)?{??
  • ????????char?seldb[64];??
  • ??
  • ????????snprintf(seldb,sizeof(seldb),"%d",dictid);??
  • ????????buf?=?sdscatprintf(buf,"*2\r\n$6\r\nSELECT\r\n$%lu\r\n%s\r\n",??
  • ????????????(unsigned?long)strlen(seldb),seldb);??
  • ????????server.aof_selected_db?=?dictid;??
  • ????}??
  • ??
  • ????//?將?EXPIRE?/?PEXPIRE?/?EXPIREAT?命令翻譯為?PEXPIREAT?命令??
  • ????if?(cmd->proc?==?expireCommand?||?cmd->proc?==?pexpireCommand?||??
  • ????????cmd->proc?==?expireatCommand)?{??
  • ????????/*?Translate?EXPIRE/PEXPIRE/EXPIREAT?into?PEXPIREAT?*/??
  • ????????buf?=?catAppendOnlyExpireAtCommand(buf,cmd,argv[1],argv[2]);??
  • ????}//?將?SETEX?/?PSETEX?命令翻譯為?SET?和?PEXPIREAT?組合命令??
  • ????else?if?(cmd->proc?==?setexCommand?||?cmd->proc?==?psetexCommand)?{??
  • ????????/*?Translate?SETEX/PSETEX?to?SET?and?PEXPIREAT?*/??
  • ????????tmpargv[0]?=?createStringObject("SET",3);??
  • ????????tmpargv[1]?=?argv[1];??
  • ????????tmpargv[2]?=?argv[3];??
  • ????????buf?=?catAppendOnlyGenericCommand(buf,3,tmpargv);??
  • ????????decrRefCount(tmpargv[0]);??
  • ????????buf?=?catAppendOnlyExpireAtCommand(buf,cmd,argv[1],argv[2]);??
  • ????}?else?{//其他的指令直接追加??
  • ????????/*?All?the?other?commands?don't?need?translation?or?need?the?
  • ?????????*?same?translation?already?operated?in?the?command?vector?
  • ?????????*?for?the?replication?itself.?*/??
  • ????????buf?=?catAppendOnlyGenericCommand(buf,argc,argv);??
  • ????}??
  • ??
  • ????/*?Append?to?the?AOF?buffer.?This?will?be?flushed?on?disk?just?before?
  • ?????*?of?re-entering?the?event?loop,?so?before?the?client?will?get?a?
  • ?????*?positive?reply?about?the?operation?performed.?*/??
  • ????//?將?buf?追加到服務器的?aof_buf?末尾,在beforeSleep中寫到AOF文件中,并且根據情況fsync刷新到硬盤??
  • ????if?(server.aof_state?==?REDIS_AOF_ON)??
  • ????????server.aof_buf?=?sdscatlen(server.aof_buf,buf,sdslen(buf));??
  • ??
  • ????/*?If?a?background?append?only?file?rewriting?is?in?progress?we?want?to?
  • ?????*?accumulate?the?differences?between?the?child?DB?and?the?current?one?
  • ?????*?in?a?buffer,?so?that?when?the?child?process?will?do?its?work?we?
  • ?????*?can?append?the?differences?to?the?new?append?only?file.?*/??
  • ????//如果server.aof_child_pid不為1,那就說明有快照進程正在寫數據到臨時文件(已經開始rewrite),??
  • ????//那么必須先將這段時間接收到的指令更新的數據先暫時存儲起來,等到快照進程完成任務后,??
  • ????//將這部分數據寫入到AOF文件末尾,保證數據不丟失??
  • ????//解釋為什么需要aof_rewrite_buf_blocks,當server在進行rewrite時即讀取所有數據庫中的數據,??
  • ????//有些數據已經寫到新的AOF文件,但是此時客戶端執行指令又將該值修改了,因此造成了差異??
  • ????if?(server.aof_child_pid?!=?-1)??
  • ????????aofRewriteBufferAppend((unsigned?char*)buf,sdslen(buf));??
  • ????/*這里說一下server.aof_buf和server.aof_rewrite_buf_blocks的區別?
  • ??????aof_buf是正常情況下aof文件打開的時候,會不斷將這份數據寫入到AOF文件中。?
  • ??????aof_rewrite_buf_blocks?是如果用戶主動觸發了寫AOF文件的命令時,比如?config?set?appendonly?yes命令?
  • ??????那么redis會fork創建一個后臺進程,也就是當時的數據快照,然后將數據寫入到一個臨時文件中去。?
  • ??????在此期間發送的命令,我們需要把它們記錄起來,等后臺進程完成AOF臨時文件寫后,serverCron定時任務?
  • ??????感知到這個退出動作,然后就會調用backgroundRewriteDoneHandler進而調用aofRewriteBufferWrite函數,?
  • ??????將aof_rewrite_buf_blocks上面的數據,也就是diff數據寫入到臨時AOF文件中,然后再unlink替換正常的AOF文件。?
  • ??????因此可以知道,aof_buf一般情況下比aof_rewrite_buf_blocks要少,?
  • ??????但開始的時候可能aof_buf包含一些后者不包含的前面部分數據。*/??
  • ??
  • ????sdsfree(buf);??
  • }??
  • ?

    ?

    Server在每次事件循環之前會調用一次beforeSleep函數,下面看看這個函數做了什么工作?

    ?

    [cpp]?view plaincopyprint?
  • /*?This?function?gets?called?every?time?Redis?is?entering?the?
  • ?*?main?loop?of?the?event?driven?library,?that?is,?before?to?sleep?
  • ?*?for?ready?file?descriptors.?*/??
  • void?beforeSleep(struct?aeEventLoop?*eventLoop)?{??
  • ????REDIS_NOTUSED(eventLoop);??
  • ????listNode?*ln;??
  • ????redisClient?*c;??
  • ??
  • ????/*?Run?a?fast?expire?cycle?(the?called?function?will?return?
  • ?????*?ASAP?if?a?fast?cycle?is?not?needed).?*/??
  • ????if?(server.active_expire_enabled?&&?server.masterhost?==?NULL)??
  • ????????activeExpireCycle(ACTIVE_EXPIRE_CYCLE_FAST);??
  • ??
  • ????/*?Try?to?process?pending?commands?for?clients?that?were?just?unblocked.?*/??
  • ????while?(listLength(server.unblocked_clients))?{??
  • ????????ln?=?listFirst(server.unblocked_clients);??
  • ????????redisAssert(ln?!=?NULL);??
  • ????????c?=?ln->value;??
  • ????????listDelNode(server.unblocked_clients,ln);??
  • ????????c->flags?&=?~REDIS_UNBLOCKED;??
  • ??
  • ????????/*?Process?remaining?data?in?the?input?buffer.?*/??
  • ????????//處理客戶端在阻塞期間接收到的客戶端發送的請求??
  • ????????if?(c->querybuf?&&?sdslen(c->querybuf)?>?0)?{??
  • ????????????server.current_client?=?c;??
  • ????????????processInputBuffer(c);??
  • ????????????server.current_client?=?NULL;??
  • ????????}??
  • ????}??
  • ??
  • ????/*?Write?the?AOF?buffer?on?disk?*/??
  • ????//將server.aof_buf中的數據追加到AOF文件中并fsync到硬盤上??
  • ????flushAppendOnlyFile(0);??
  • }??
  • 通過上面的代碼及注釋可以發現,beforeSleep函數做了三件事:1、處理過期鍵,2、處理阻塞期間的客戶端請求,3、將server.aof_buf中的數據追加到AOF文件中并fsync刷新到硬盤上,flushAppendOnlyFile函數給定了一個參數force,表示是否強制寫入AOF文件,0表示非強制即支持延遲寫,1表示強制寫入。

    ?

    ?

    [cpp]?view plaincopyprint?
  • void?flushAppendOnlyFile(int?force)?{??
  • ????ssize_t?nwritten;??
  • ????int?sync_in_progress?=?0;??
  • ????if?(sdslen(server.aof_buf)?==?0)?return;??
  • ????//?返回后臺正在等待執行的?fsync?數量??
  • ????if?(server.aof_fsync?==?AOF_FSYNC_EVERYSEC)??
  • ????????sync_in_progress?=?bioPendingJobsOfType(REDIS_BIO_AOF_FSYNC)?!=?0;??
  • ??
  • ????//?AOF?模式為每秒?fsync?,并且?force?不為?1?如果可以的話,推延沖洗??
  • ????if?(server.aof_fsync?==?AOF_FSYNC_EVERYSEC?&&?!force)?{??
  • ????????/*?With?this?append?fsync?policy?we?do?background?fsyncing.?
  • ?????????*?If?the?fsync?is?still?in?progress?we?can?try?to?delay?
  • ?????????*?the?write?for?a?couple?of?seconds.?*/??
  • ????????//?如果?aof_fsync?隊列里已經有正在等待的任務??
  • ????????if?(sync_in_progress)?{??
  • ????????????//?上一次沒有推遲沖洗過,記錄推延的當前時間,然后返回??
  • ????????????if?(server.aof_flush_postponed_start?==?0)?{??
  • ????????????????/*?No?previous?write?postponinig,?remember?that?we?are?
  • ?????????????????*?postponing?the?flush?and?return.?*/??
  • ????????????????server.aof_flush_postponed_start?=?server.unixtime;??
  • ????????????????return;??
  • ????????????}?else?if?(server.unixtime?-?server.aof_flush_postponed_start?<?2)?{??
  • ????????????????//?允許在兩秒之內的推延沖洗??
  • ????????????????/*?We?were?already?waiting?for?fsync?to?finish,?but?for?less?
  • ?????????????????*?than?two?seconds?this?is?still?ok.?Postpone?again.?*/??
  • ????????????????return;??
  • ????????????}??
  • ????????????/*?Otherwise?fall?trough,?and?go?write?since?we?can't?wait?
  • ?????????????*?over?two?seconds.?*/??
  • ????????????server.aof_delayed_fsync++;??
  • ????????????redisLog(REDIS_NOTICE,"Asynchronous?AOF?fsync?is?taking?too?long?(disk?is?busy?).?Writing?the?AOF?buffer?without?waiting?for?fsync?to?complete,?this?may?slow?down?Redis.");??
  • ????????}??
  • ????}??
  • ????/*?If?you?are?following?this?code?path,?then?we?are?going?to?write?so?
  • ?????*?set?reset?the?postponed?flush?sentinel?to?zero.?*/??
  • ????server.aof_flush_postponed_start?=?0;??
  • ??
  • ????/*?We?want?to?perform?a?single?write.?This?should?be?guaranteed?atomic?
  • ?????*?at?least?if?the?filesystem?we?are?writing?is?a?real?physical?one.?
  • ?????*?While?this?will?save?us?against?the?server?being?killed?I?don't?think?
  • ?????*?there?is?much?to?do?about?the?whole?server?stopping?for?power?problems?
  • ?????*?or?alike?*/??
  • ????//?將?AOF?緩存寫入到文件,如果一切幸運的話,寫入會原子性地完成??
  • ????nwritten?=?write(server.aof_fd,server.aof_buf,sdslen(server.aof_buf));??
  • ????if?(nwritten?!=?(signed)sdslen(server.aof_buf))?{//出錯??
  • ????????/*?Ooops,?we?are?in?troubles.?The?best?thing?to?do?for?now?is?
  • ?????????*?aborting?instead?of?giving?the?illusion?that?everything?is?
  • ?????????*?working?as?expected.?*/??
  • ????????if?(nwritten?==?-1)?{??
  • ????????????redisLog(REDIS_WARNING,"Exiting?on?error?writing?to?the?append-only?file:?%s",strerror(errno));??
  • ????????}?else?{??
  • ????????????redisLog(REDIS_WARNING,"Exiting?on?short?write?while?writing?to?"??
  • ???????????????????????????????????"the?append-only?file:?%s?(nwritten=%ld,?"??
  • ???????????????????????????????????"expected=%ld)",??
  • ???????????????????????????????????strerror(errno),??
  • ???????????????????????????????????(long)nwritten,??
  • ???????????????????????????????????(long)sdslen(server.aof_buf));??
  • ??
  • ????????????if?(ftruncate(server.aof_fd,?server.aof_current_size)?==?-1)?{??
  • ????????????????redisLog(REDIS_WARNING,?"Could?not?remove?short?write?"??
  • ?????????????????????????"from?the?append-only?file.??Redis?may?refuse?"??
  • ?????????????????????????"to?load?the?AOF?the?next?time?it?starts.??"??
  • ?????????????????????????"ftruncate:?%s",?strerror(errno));??
  • ????????????}??
  • ????????}??
  • ????????exit(1);??
  • ????}??
  • ????server.aof_current_size?+=?nwritten;??
  • ??
  • ????/*?Re-use?AOF?buffer?when?it?is?small?enough.?The?maximum?comes?from?the?
  • ?????*?arena?size?of?4k?minus?some?overhead?(but?is?otherwise?arbitrary).?*/??
  • ????//?如果?aof?緩存不是太大,那么重用它,否則,清空?aof?緩存??
  • ????if?((sdslen(server.aof_buf)+sdsavail(server.aof_buf))?<?4000)?{??
  • ????????sdsclear(server.aof_buf);??
  • ????}?else?{??
  • ????????sdsfree(server.aof_buf);??
  • ????????server.aof_buf?=?sdsempty();??
  • ????}??
  • ??
  • ????/*?Don't?fsync?if?no-appendfsync-on-rewrite?is?set?to?yes?and?there?are?
  • ?????*?children?doing?I/O?in?the?background.?*/??
  • ????//aof?rdb子進程運行中不支持fsync并且aof?rdb子進程正在運行,那么直接返回,??
  • ????//但是數據已經寫到aof文件中,只是沒有刷新到硬盤??
  • ????if?(server.aof_no_fsync_on_rewrite?&&??
  • ????????(server.aof_child_pid?!=?-1?||?server.rdb_child_pid?!=?-1))??
  • ????????????return;??
  • ??
  • ????/*?Perform?the?fsync?if?needed.?*/??
  • ????if?(server.aof_fsync?==?AOF_FSYNC_ALWAYS)?{//總是fsync,那么直接進行fsync??
  • ????????/*?aof_fsync?is?defined?as?fdatasync()?for?Linux?in?order?to?avoid?
  • ?????????*?flushing?metadata.?*/??
  • ????????aof_fsync(server.aof_fd);?/*?Let's?try?to?get?this?data?on?the?disk?*/??
  • ????????server.aof_last_fsync?=?server.unixtime;??
  • ????}?else?if?((server.aof_fsync?==?AOF_FSYNC_EVERYSEC?&&??
  • ????????????????server.unixtime?>?server.aof_last_fsync))?{??
  • ????????if?(!sync_in_progress)?aof_background_fsync(server.aof_fd);//放到后臺線程進行fsync??
  • ????????server.aof_last_fsync?=?server.unixtime;??
  • ????}??
  • }??
  • 上述代碼中請關注server.aof_fsync參數,即設置Redis fsync AOF文件到硬盤的策略,如果設置為AOF_FSYNC_ALWAYS,那么直接在主進程中fsync,如果設置為AOF_FSYNC_EVERYSEC,那么放入后臺線程中fsync,后臺線程的代碼在bio.c中。

    ?

    ?

    小結

    文章寫到這,已經解決的了Redis Server啟動加載AOF文件和如何將客戶端請求產生的新的數據追加到AOF文件中,對于追加數據到AOF文件中,根據fsync的配置策略如何將寫入到AOF文件中的新數據刷新到硬盤中,直接在主進程中fsync或是在后臺線程fsync。

    至此,AOF數據持久化還剩下如何rewrite AOF,接受客戶端發送的BGREWRITEAOF請求,此部分內容待下篇博客中解析。

    感謝此篇博客給我在理解Redis AOF數據持久化方面的巨大幫助,http://chenzhenianqing.cn/articles/786.html

    本人Redis-2.8.2的源碼注釋已經放到Github中,有需要的讀者可以下載,我也會在后續的時間中更新,https://github.com/xkeyideal/annotated-redis-2.8.2

    本人不怎么會使用Git,望有人能教我一下。

    ?

    --------------------------------------------------------------------------------------------------------------------------------------------------------------

    本文所引用的源碼全部來自Redis2.8.2版本。

    Redis AOF數據持久化機制的實現相關代碼是redis.c, redis.h, aof.c, bio.c, rio.c, config.c

    在閱讀本文之前請先閱讀Redis數據持久化機制AOF原理分析之配置詳解文章,了解AOF相關參數的解析,文章鏈接

    http://blog.csdn.net/acceptedxukai/article/details/18135219

    接著上一篇文章,本文將介紹Redis是如何實現AOF rewrite的。

    轉載請注明,文章出自http://blog.csdn.net/acceptedxukai/article/details/18181563

    ?

    AOF rewrite的觸發機制

    ?

    如果Redis只是將客戶端修改數據庫的指令重現存儲在AOF文件中,那么AOF文件的大小會不斷的增加,因為AOF文件只是簡單的重現存儲了客戶端的指令,而并沒有進行合并。對于該問題最簡單的處理方式,即當AOF文件滿足一定條件時就對AOF進行rewrite,rewrite是根據當前內存數據庫中的數據進行遍歷寫到一個臨時的AOF文件,待寫完后替換掉原來的AOF文件即可。

    ?

    Redis觸發AOF rewrite機制有三種:

    1、Redis Server接收到客戶端發送的BGREWRITEAOF指令請求,如果當前AOF/RDB數據持久化沒有在執行,那么執行,反之,等當前AOF/RDB數據持久化結束后執行AOF rewrite

    2、在Redis配置文件redis.conf中,用戶設置了auto-aof-rewrite-percentage和auto-aof-rewrite-min-size參數,并且當前AOF文件大小server.aof_current_size大于auto-aof-rewrite-min-size(server.aof_rewrite_min_size),同時AOF文件大小的增長率大于auto-aof-rewrite-percentage(server.aof_rewrite_perc)時,會自動觸發AOF rewrite

    3、用戶設置“config set appendonly yes”開啟AOF的時,調用startAppendOnly函數會觸發rewrite

    下面分別介紹上述三種機制的處理.

    ?

    接收到BGREWRITEAOF指令

    [cpp]?view plaincopyprint?
  • <span?style="font-size:12px;">void?bgrewriteaofCommand(redisClient?*c)?{??
  • ????//AOF?rewrite正在執行,那么直接返回??
  • ????if?(server.aof_child_pid?!=?-1)?{??
  • ????????addReplyError(c,"Background?append?only?file?rewriting?already?in?progress");??
  • ????}?else?if?(server.rdb_child_pid?!=?-1)?{??
  • ????????//AOF?rewrite未執行,但RDB數據持久化正在執行,那么設置AOF?rewrite狀態為scheduled??
  • ????????//待RDB結束后執行AOF?rewrite??
  • ????????server.aof_rewrite_scheduled?=?1;??
  • ????????addReplyStatus(c,"Background?append?only?file?rewriting?scheduled");??
  • ????}?else?if?(rewriteAppendOnlyFileBackground()?==?REDIS_OK)?{??
  • ????????//直接執行AOF?rewrite??
  • ????????addReplyStatus(c,"Background?append?only?file?rewriting?started");??
  • ????}?else?{??
  • ????????addReply(c,shared.err);??
  • ????}??
  • }</span>??
  • 當AOF rewrite請求被掛起時,在serverCron函數中,會處理。 [cpp]?view plaincopyprint?
  • /*?Start?a?scheduled?AOF?rewrite?if?this?was?requested?by?the?user?while?
  • ?????*?a?BGSAVE?was?in?progress.?*/??
  • ????//?如果用戶執行?BGREWRITEAOF?命令的話,在后臺開始?AOF?重寫??
  • ????//當用戶執行BGREWRITEAOF命令時,如果RDB文件正在寫,那么將server.aof_rewrite_scheduled標記為1??
  • ????//當RDB文件寫完后開啟AOF?rewrite??
  • ????if?(server.rdb_child_pid?==?-1?&&?server.aof_child_pid?==?-1?&&??
  • ????????server.aof_rewrite_scheduled)??
  • ????{??
  • ????????rewriteAppendOnlyFileBackground();??
  • ????}??


  • Server自動對AOF進行rewrite

    在serverCron函數中會周期性判斷 [cpp]?view plaincopyprint?
  • /*?Trigger?an?AOF?rewrite?if?needed?*/??
  • ?????????//滿足一定條件rewrite?AOF文件??
  • ?????????if?(server.rdb_child_pid?==?-1?&&??
  • ?????????????server.aof_child_pid?==?-1?&&??
  • ?????????????server.aof_rewrite_perc?&&??
  • ?????????????server.aof_current_size?>?server.aof_rewrite_min_size)??
  • ?????????{??
  • ????????????long?long?base?=?server.aof_rewrite_base_size????
  • ????????????????????????????server.aof_rewrite_base_size?:?1;??
  • ????????????long?long?growth?=?(server.aof_current_size*100/base)?-?100;??
  • ????????????if?(growth?>=?server.aof_rewrite_perc)?{??
  • ????????????????redisLog(REDIS_NOTICE,"Starting?automatic?rewriting?of?AOF?on?%lld%%?growth",growth);??
  • ????????????????rewriteAppendOnlyFileBackground();??
  • ????????????}??
  • ?????????}??

  • config set appendonly yes

    當客戶端發送該指令時,config.c中的configSetCommand函數會做出響應,startAppendOnly函數會執行AOF rewrite [cpp]?view plaincopyprint?
  • if?(!strcasecmp(c->argv[2]->ptr,"appendonly"))?{??
  • ????int?enable?=?yesnotoi(o->ptr);??
  • ??
  • ????if?(enable?==?-1)?goto?badfmt;??
  • ????if?(enable?==?0?&&?server.aof_state?!=?REDIS_AOF_OFF)?{//appendonly?no?關閉AOF??
  • ????????stopAppendOnly();??
  • ????}?else?if?(enable?&&?server.aof_state?==?REDIS_AOF_OFF)?{//appendonly?yes?rewrite?AOF??
  • ????????if?(startAppendOnly()?==?REDIS_ERR)?{??
  • ????????????addReplyError(c,??
  • ????????????????"Unable?to?turn?on?AOF.?Check?server?logs.");??
  • ????????????return;??
  • ????????}??
  • ????}??
  • }??
  • [cpp]?view plaincopyprint?
  • int?startAppendOnly(void)?{??
  • ????server.aof_last_fsync?=?server.unixtime;??
  • ????server.aof_fd?=?open(server.aof_filename,O_WRONLY|O_APPEND|O_CREAT,0644);??
  • ????redisAssert(server.aof_state?==?REDIS_AOF_OFF);??
  • ????if?(server.aof_fd?==?-1)?{??
  • ????????redisLog(REDIS_WARNING,"Redis?needs?to?enable?the?AOF?but?can't?open?the?append?only?file:?%s",strerror(errno));??
  • ????????return?REDIS_ERR;??
  • ????}??
  • ????if?(rewriteAppendOnlyFileBackground()?==?REDIS_ERR)?{//rewrite??
  • ????????close(server.aof_fd);??
  • ????????redisLog(REDIS_WARNING,"Redis?needs?to?enable?the?AOF?but?can't?trigger?a?background?AOF?rewrite?operation.?Check?the?above?logs?for?more?info?about?the?error.");??
  • ????????return?REDIS_ERR;??
  • ????}??
  • ????/*?We?correctly?switched?on?AOF,?now?wait?for?the?rerwite?to?be?complete?
  • ?????*?in?order?to?append?data?on?disk.?*/??
  • ????server.aof_state?=?REDIS_AOF_WAIT_REWRITE;??
  • ????return?REDIS_OK;??
  • }??

  • Redis AOF rewrite機制的實現

    從上述分析可以看出rewrite的實現全部依靠rewriteAppendOnlyFileBackground函數,下面分析該函數,通過下面的代碼可以看出,Redis是fork出一個子進程來操作AOF rewrite,然后子進程調用rewriteAppendOnlyFile函數,將數據寫到一個臨時文件temp-rewriteaof-bg-%d.aof中。如果子進程完成會通過exit(0)函數通知父進程rewrite結束,在serverCron函數中使用wait3函數接收子進程退出狀態,然后執行后續的AOF rewrite的收尾工作,后面將會分析。 父進程的工作主要包括清楚server.aof_rewrite_scheduled標志,記錄子進程IDserver.aof_child_pid = childpid,記錄rewrite的開始時間server.aof_rewrite_time_start = time(NULL)等。 [cpp]?view plaincopyprint?
  • int?rewriteAppendOnlyFileBackground(void)?{??
  • ????pid_t?childpid;??
  • ????long?long?start;??
  • ??
  • ????//?后臺重寫正在執行??
  • ????if?(server.aof_child_pid?!=?-1)?return?REDIS_ERR;??
  • ????start?=?ustime();??
  • ????if?((childpid?=?fork())?==?0)?{??
  • ????????char?tmpfile[256];??
  • ??
  • ????????/*?Child?*/??
  • ????????closeListeningSockets(0);//??
  • ????????redisSetProcTitle("redis-aof-rewrite");??
  • ????????snprintf(tmpfile,256,"temp-rewriteaof-bg-%d.aof",?(int)?getpid());??
  • ????????if?(rewriteAppendOnlyFile(tmpfile)?==?REDIS_OK)?{??
  • ????????????size_t?private_dirty?=?zmalloc_get_private_dirty();??
  • ??
  • ????????????if?(private_dirty)?{??
  • ????????????????redisLog(REDIS_NOTICE,??
  • ????????????????????"AOF?rewrite:?%zu?MB?of?memory?used?by?copy-on-write",??
  • ????????????????????private_dirty/(1024*1024));??
  • ????????????}??
  • ????????????exitFromChild(0);??
  • ????????}?else?{??
  • ????????????exitFromChild(1);??
  • ????????}??
  • ????}?else?{??
  • ????????/*?Parent?*/??
  • ????????server.stat_fork_time?=?ustime()-start;??
  • ????????if?(childpid?==?-1)?{??
  • ????????????redisLog(REDIS_WARNING,??
  • ????????????????"Can't?rewrite?append?only?file?in?background:?fork:?%s",??
  • ????????????????strerror(errno));??
  • ????????????return?REDIS_ERR;??
  • ????????}??
  • ????????redisLog(REDIS_NOTICE,??
  • ????????????"Background?append?only?file?rewriting?started?by?pid?%d",childpid);??
  • ????????server.aof_rewrite_scheduled?=?0;??
  • ????????server.aof_rewrite_time_start?=?time(NULL);??
  • ????????server.aof_child_pid?=?childpid;??
  • ????????updateDictResizePolicy();??
  • ????????/*?We?set?appendseldb?to?-1?in?order?to?force?the?next?call?to?the?
  • ?????????*?feedAppendOnlyFile()?to?issue?a?SELECT?command,?so?the?differences?
  • ?????????*?accumulated?by?the?parent?into?server.aof_rewrite_buf?will?start?
  • ?????????*?with?a?SELECT?statement?and?it?will?be?safe?to?merge.?*/??
  • ????????server.aof_selected_db?=?-1;??
  • ????????replicationScriptCacheFlush();??
  • ????????return?REDIS_OK;??
  • ????}??
  • ????return?REDIS_OK;?/*?unreached?*/??
  • }??
  • 接下來介紹rewriteAppendOnlyFile函數,該函數的主要工作為:遍歷所有數據庫中的數據,將其寫入到臨時文件temp-rewriteaof-%d.aof中,寫入函數定義在rio.c中,比較簡單,然后將數據刷新到硬盤中,然后將文件名rename為其調用者給定的臨時文件名,注意仔細看代碼,這里并沒有修改為正式的AOF文件名。 在寫入文件時如果設置server.aof_rewrite_incremental_fsync參數,那么在rioWrite函數中fwrite部分數據就會將數據fsync到硬盤中,來保證數據的正確性。 [cpp]?view plaincopyprint?
  • int?rewriteAppendOnlyFile(char?*filename)?{??
  • ????dictIterator?*di?=?NULL;??
  • ????dictEntry?*de;??
  • ????rio?aof;??
  • ????FILE?*fp;??
  • ????char?tmpfile[256];??
  • ????int?j;??
  • ????long?long?now?=?mstime();??
  • ??
  • ????/*?Note?that?we?have?to?use?a?different?temp?name?here?compared?to?the?
  • ?????*?one?used?by?rewriteAppendOnlyFileBackground()?function.?*/??
  • ????snprintf(tmpfile,256,"temp-rewriteaof-%d.aof",?(int)?getpid());??
  • ????fp?=?fopen(tmpfile,"w");??
  • ????if?(!fp)?{??
  • ????????redisLog(REDIS_WARNING,?"Opening?the?temp?file?for?AOF?rewrite?in?rewriteAppendOnlyFile():?%s",?strerror(errno));??
  • ????????return?REDIS_ERR;??
  • ????}??
  • ??
  • ????rioInitWithFile(&aof,fp);?//初始化讀寫函數,rio.c??
  • ????//設置r->io.file.autosync?=?bytes;每32M刷新一次??
  • ????if?(server.aof_rewrite_incremental_fsync)??
  • ????????rioSetAutoSync(&aof,REDIS_AOF_AUTOSYNC_BYTES);??
  • ????for?(j?=?0;?j?<?server.dbnum;?j++)?{//遍歷每個數據庫??
  • ????????char?selectcmd[]?=?"*2\r\n$6\r\nSELECT\r\n";??
  • ????????redisDb?*db?=?server.db+j;??
  • ????????dict?*d?=?db->dict;??
  • ????????if?(dictSize(d)?==?0)?continue;??
  • ????????di?=?dictGetSafeIterator(d);??
  • ????????if?(!di)?{??
  • ????????????fclose(fp);??
  • ????????????return?REDIS_ERR;??
  • ????????}??
  • ??
  • ????????/*?SELECT?the?new?DB?*/??
  • ????????if?(rioWrite(&aof,selectcmd,sizeof(selectcmd)-1)?==?0)?goto?werr;??
  • ????????if?(rioWriteBulkLongLong(&aof,j)?==?0)?goto?werr;??
  • ??
  • ????????/*?Iterate?this?DB?writing?every?entry?*/??
  • ????????while((de?=?dictNext(di))?!=?NULL)?{??
  • ????????????sds?keystr;??
  • ????????????robj?key,?*o;??
  • ????????????long?long?expiretime;??
  • ??
  • ????????????keystr?=?dictGetKey(de);??
  • ????????????o?=?dictGetVal(de);??
  • ????????????initStaticStringObject(key,keystr);??
  • ??
  • ????????????expiretime?=?getExpire(db,&key);??
  • ??
  • ????????????/*?If?this?key?is?already?expired?skip?it?*/??
  • ????????????if?(expiretime?!=?-1?&&?expiretime?<?now)?continue;??
  • ??
  • ????????????/*?Save?the?key?and?associated?value?*/??
  • ????????????if?(o->type?==?REDIS_STRING)?{??
  • ????????????????/*?Emit?a?SET?command?*/??
  • ????????????????char?cmd[]="*3\r\n$3\r\nSET\r\n";??
  • ????????????????if?(rioWrite(&aof,cmd,sizeof(cmd)-1)?==?0)?goto?werr;??
  • ????????????????/*?Key?and?value?*/??
  • ????????????????if?(rioWriteBulkObject(&aof,&key)?==?0)?goto?werr;??
  • ????????????????if?(rioWriteBulkObject(&aof,o)?==?0)?goto?werr;??
  • ????????????}?else?if?(o->type?==?REDIS_LIST)?{??
  • ????????????????if?(rewriteListObject(&aof,&key,o)?==?0)?goto?werr;??
  • ????????????}?else?if?(o->type?==?REDIS_SET)?{??
  • ????????????????if?(rewriteSetObject(&aof,&key,o)?==?0)?goto?werr;??
  • ????????????}?else?if?(o->type?==?REDIS_ZSET)?{??
  • ????????????????if?(rewriteSortedSetObject(&aof,&key,o)?==?0)?goto?werr;??
  • ????????????}?else?if?(o->type?==?REDIS_HASH)?{??
  • ????????????????if?(rewriteHashObject(&aof,&key,o)?==?0)?goto?werr;??
  • ????????????}?else?{??
  • ????????????????redisPanic("Unknown?object?type");??
  • ????????????}??
  • ????????????/*?Save?the?expire?time?*/??
  • ????????????if?(expiretime?!=?-1)?{??
  • ????????????????char?cmd[]="*3\r\n$9\r\nPEXPIREAT\r\n";??
  • ????????????????if?(rioWrite(&aof,cmd,sizeof(cmd)-1)?==?0)?goto?werr;??
  • ????????????????if?(rioWriteBulkObject(&aof,&key)?==?0)?goto?werr;??
  • ????????????????if?(rioWriteBulkLongLong(&aof,expiretime)?==?0)?goto?werr;??
  • ????????????}??
  • ????????}??
  • ????????dictReleaseIterator(di);??
  • ????}??
  • ??
  • ????/*?Make?sure?data?will?not?remain?on?the?OS's?output?buffers?*/??
  • ????fflush(fp);??
  • ????aof_fsync(fileno(fp));//將tempfile文件刷新到硬盤??
  • ????fclose(fp);??
  • ??
  • ????/*?Use?RENAME?to?make?sure?the?DB?file?is?changed?atomically?only?
  • ?????*?if?the?generate?DB?file?is?ok.?*/??
  • ????if?(rename(tmpfile,filename)?==?-1)?{//重命名文件名,注意rename后的文件也是一個臨時文件??
  • ????????redisLog(REDIS_WARNING,"Error?moving?temp?append?only?file?on?the?final?destination:?%s",?strerror(errno));??
  • ????????unlink(tmpfile);??
  • ????????return?REDIS_ERR;??
  • ????}??
  • ????redisLog(REDIS_NOTICE,"SYNC?append?only?file?rewrite?performed");??
  • ????return?REDIS_OK;??
  • ??
  • werr:??
  • ????fclose(fp);??
  • ????unlink(tmpfile);??
  • ????redisLog(REDIS_WARNING,"Write?error?writing?append?only?file?on?disk:?%s",?strerror(errno));??
  • ????if?(di)?dictReleaseIterator(di);??
  • ????return?REDIS_ERR;??
  • }??
  • AOF rewrite工作到這里已經結束一半,上一篇文章提到如果server.aof_state != REDIS_AOF_OFF,那么就會將客戶端請求指令修改的數據通過feedAppendOnlyFile函數追加到AOF文件中,那么此時AOF已經rewrite了,必須要處理此時出現的差異數據,記得在feedAppendOnlyFile函數中有這么一段代碼 [cpp]?view plaincopyprint?
  • if?(server.aof_child_pid?!=?-1)??
  • ????????aofRewriteBufferAppend((unsigned?char*)buf,sdslen(buf));??
  • 如果AOF rewrite正在進行,那么就將修改數據的指令字符串存儲到server.aof_rewrite_buf_blocks鏈表中,等待AOF rewrite子進程結束后處理,處理此部分數據的代碼在serverCron函數中。需要指出的是wait3函數我不了解,可能下面注釋會有點問題。 [cpp]?view plaincopyprint?
  • /*?Check?if?a?background?saving?or?AOF?rewrite?in?progress?terminated.?*/??
  • //如果RDB?bgsave或AOF?rewrite子進程已經執行,通過獲取子進程的退出狀態,對后續的工作進行處理??
  • if?(server.rdb_child_pid?!=?-1?||?server.aof_child_pid?!=?-1)?{//??
  • ????int?statloc;??
  • ????pid_t?pid;??
  • ??
  • ????if?((pid?=?wait3(&statloc,WNOHANG,NULL))?!=?0)?{??
  • ????????int?exitcode?=?WEXITSTATUS(statloc);//獲取退出的狀態??
  • ????????int?bysignal?=?0;??
  • ??
  • ????????if?(WIFSIGNALED(statloc))?bysignal?=?WTERMSIG(statloc);??
  • ??
  • ????????if?(pid?==?server.rdb_child_pid)?{??
  • ????????????backgroundSaveDoneHandler(exitcode,bysignal);??
  • ????????}?else?if?(pid?==?server.aof_child_pid)?{??
  • ????????????backgroundRewriteDoneHandler(exitcode,bysignal);??
  • ????????}?else?{??
  • ????????????redisLog(REDIS_WARNING,??
  • ????????????????"Warning,?detected?child?with?unmatched?pid:?%ld",??
  • ????????????????(long)pid);??
  • ????????}??
  • ????????//?如果?BGSAVE?和?BGREWRITEAOF?都已經完成,那么重新開始?REHASH??
  • ????????updateDictResizePolicy();??
  • ????}??
  • }??
  • 對于AOF rewrite期間出現的差異數據,Server通過backgroundSaveDoneHandler函數將server.aof_rewrite_buf_blocks鏈表中數據追加到新的AOF文件中。 backgroundSaveDoneHandler函數執行步驟:
    1、通過判斷子進程的退出狀態,正確的退出狀態為exit(0),即exitcode為0,bysignal我不清楚具體意義,如果退出狀態正確,backgroundSaveDoneHandler函數才會開始處理 2、通過對rewriteAppendOnlyFileBackground函數的分析,可以知道rewrite后的AOF臨時文件名為temp-rewriteaof-bg-%d.aof(%d=server.aof_child_pid)中,接著需要打開此臨時文件 3、調用aofRewriteBufferWrite函數將server.aof_rewrite_buf_blocks中差異數據寫到該臨時文件中 4、如果舊的AOF文件未打開,那么打開舊的AOF文件,將文件描述符賦值給臨時變量oldfd 5、將臨時的AOF文件名rename為正常的AOF文件名 6、如果舊的AOF文件未打開,那么此時只需要關閉新的AOF文件,此時的server.aof_rewrite_buf_blocks數據應該為空;如果舊的AOF是打開的,那么將server.aof_fd指向newfd,然后根據相應的fsync策略將數據刷新到硬盤上 7、調用aofUpdateCurrentSize函數統計AOF文件的大小,更新server.aof_rewrite_base_size,為serverCron中自動AOF rewrite做相應判斷 8、如果之前是REDIS_AOF_WAIT_REWRITE狀態,則設置server.aof_state為REDIS_AOF_ON,因為只有“config set appendonly yes”指令才會設置這個狀態,也就是需要寫完快照后,立即打開AOF;而BGREWRITEAOF不需要打開AOF 9、調用后臺線程去關閉舊的AOF文件 下面是backgroundSaveDoneHandler函數的注釋代碼 [cpp]?view plaincopyprint?
  • /*?A?background?append?only?file?rewriting?(BGREWRITEAOF)?terminated?its?work.?
  • ?*?Handle?this.?*/??
  • void?backgroundRewriteDoneHandler(int?exitcode,?int?bysignal)?{??
  • ????if?(!bysignal?&&?exitcode?==?0)?{//子進程退出狀態正確??
  • ????????int?newfd,?oldfd;??
  • ????????char?tmpfile[256];??
  • ????????long?long?now?=?ustime();??
  • ??
  • ????????redisLog(REDIS_NOTICE,??
  • ????????????"Background?AOF?rewrite?terminated?with?success");??
  • ??
  • ????????/*?Flush?the?differences?accumulated?by?the?parent?to?the?
  • ?????????*?rewritten?AOF.?*/??
  • ????????snprintf(tmpfile,256,"temp-rewriteaof-bg-%d.aof",??
  • ????????????(int)server.aof_child_pid);??
  • ????????newfd?=?open(tmpfile,O_WRONLY|O_APPEND);??
  • ????????if?(newfd?==?-1)?{??
  • ????????????redisLog(REDIS_WARNING,??
  • ????????????????"Unable?to?open?the?temporary?AOF?produced?by?the?child:?%s",?strerror(errno));??
  • ????????????goto?cleanup;??
  • ????????}??
  • ????????//處理server.aof_rewrite_buf_blocks中DIFF數據??
  • ????????if?(aofRewriteBufferWrite(newfd)?==?-1)?{??
  • ????????????redisLog(REDIS_WARNING,??
  • ????????????????"Error?trying?to?flush?the?parent?diff?to?the?rewritten?AOF:?%s",?strerror(errno));??
  • ????????????close(newfd);??
  • ????????????goto?cleanup;??
  • ????????}??
  • ??
  • ????????redisLog(REDIS_NOTICE,??
  • ????????????"Parent?diff?successfully?flushed?to?the?rewritten?AOF?(%lu?bytes)",?aofRewriteBufferSize());??
  • ??
  • ????????/*?The?only?remaining?thing?to?do?is?to?rename?the?temporary?file?to?
  • ?????????*?the?configured?file?and?switch?the?file?descriptor?used?to?do?AOF?
  • ?????????*?writes.?We?don't?want?close(2)?or?rename(2)?calls?to?block?the?
  • ?????????*?server?on?old?file?deletion.?
  • ?????????*?
  • ?????????*?There?are?two?possible?scenarios:?
  • ?????????*?
  • ?????????*?1)?AOF?is?DISABLED?and?this?was?a?one?time?rewrite.?The?temporary?
  • ?????????*?file?will?be?renamed?to?the?configured?file.?When?this?file?already?
  • ?????????*?exists,?it?will?be?unlinked,?which?may?block?the?server.?
  • ?????????*?
  • ?????????*?2)?AOF?is?ENABLED?and?the?rewritten?AOF?will?immediately?start?
  • ?????????*?receiving?writes.?After?the?temporary?file?is?renamed?to?the?
  • ?????????*?configured?file,?the?original?AOF?file?descriptor?will?be?closed.?
  • ?????????*?Since?this?will?be?the?last?reference?to?that?file,?closing?it?
  • ?????????*?causes?the?underlying?file?to?be?unlinked,?which?may?block?the?
  • ?????????*?server.?
  • ?????????*?
  • ?????????*?To?mitigate?the?blocking?effect?of?the?unlink?operation?(either?
  • ?????????*?caused?by?rename(2)?in?scenario?1,?or?by?close(2)?in?scenario?2),?we?
  • ?????????*?use?a?background?thread?to?take?care?of?this.?First,?we?
  • ?????????*?make?scenario?1?identical?to?scenario?2?by?opening?the?target?file?
  • ?????????*?when?it?exists.?The?unlink?operation?after?the?rename(2)?will?then?
  • ?????????*?be?executed?upon?calling?close(2)?for?its?descriptor.?Everything?to?
  • ?????????*?guarantee?atomicity?for?this?switch?has?already?happened?by?then,?so?
  • ?????????*?we?don't?care?what?the?outcome?or?duration?of?that?close?operation?
  • ?????????*?is,?as?long?as?the?file?descriptor?is?released?again.?*/??
  • ????????if?(server.aof_fd?==?-1)?{??
  • ????????????/*?AOF?disabled?*/??
  • ??
  • ?????????????/*?Don't?care?if?this?fails:?oldfd?will?be?-1?and?we?handle?that.?
  • ??????????????*?One?notable?case?of?-1?return?is?if?the?old?file?does?
  • ??????????????*?not?exist.?*/??
  • ?????????????oldfd?=?open(server.aof_filename,O_RDONLY|O_NONBLOCK);??
  • ????????}?else?{??
  • ????????????/*?AOF?enabled?*/??
  • ????????????oldfd?=?-1;?/*?We'll?set?this?to?the?current?AOF?filedes?later.?*/??
  • ????????}??
  • ??
  • ????????/*?Rename?the?temporary?file.?This?will?not?unlink?the?target?file?if?
  • ?????????*?it?exists,?because?we?reference?it?with?"oldfd".?*/??
  • ????????//把臨時文件改名為正常的AOF文件名。由于當前oldfd已經指向這個之前的正常文件名的文件,??
  • ????????//所以當前不會造成unlink操作,得等那個oldfd被close的時候,內核判斷該文件沒有指向了,就刪除之。??
  • ????????if?(rename(tmpfile,server.aof_filename)?==?-1)?{??
  • ????????????redisLog(REDIS_WARNING,??
  • ????????????????"Error?trying?to?rename?the?temporary?AOF?file:?%s",?strerror(errno));??
  • ????????????close(newfd);??
  • ????????????if?(oldfd?!=?-1)?close(oldfd);??
  • ????????????goto?cleanup;??
  • ????????}??
  • ????????//如果AOF關閉了,那只要處理新文件,直接關閉這個新的文件即可??
  • ????????//但是這里會不會導致服務器卡呢?這個newfd應該是臨時文件的最后一個fd了,不會的,??
  • ????????//因為這個文件在本函數不會寫入數據,因為stopAppendOnly函數會清空aof_rewrite_buf_blocks列表。??
  • ????????if?(server.aof_fd?==?-1)?{??
  • ????????????/*?AOF?disabled,?we?don't?need?to?set?the?AOF?file?descriptor?
  • ?????????????*?to?this?new?file,?so?we?can?close?it.?*/??
  • ????????????close(newfd);??
  • ????????}?else?{??
  • ????????????/*?AOF?enabled,?replace?the?old?fd?with?the?new?one.?*/??
  • ????????????oldfd?=?server.aof_fd;??
  • ????????????//指向新的fd,此時這個fd由于上面的rename語句存在,已經為正常aof文件名??
  • ????????????server.aof_fd?=?newfd;??
  • ????????????//fsync到硬盤??
  • ????????????if?(server.aof_fsync?==?AOF_FSYNC_ALWAYS)??
  • ????????????????aof_fsync(newfd);??
  • ????????????else?if?(server.aof_fsync?==?AOF_FSYNC_EVERYSEC)??
  • ????????????????aof_background_fsync(newfd);??
  • ????????????server.aof_selected_db?=?-1;?/*?Make?sure?SELECT?is?re-issued?*/??
  • ????????????aofUpdateCurrentSize();??
  • ????????????server.aof_rewrite_base_size?=?server.aof_current_size;??
  • ??
  • ????????????/*?Clear?regular?AOF?buffer?since?its?contents?was?just?written?to?
  • ?????????????*?the?new?AOF?from?the?background?rewrite?buffer.?*/??
  • ????????????//rewrite得到的肯定是最新的數據,所以aof_buf中的數據沒有意義,直接清空??
  • ????????????sdsfree(server.aof_buf);??
  • ????????????server.aof_buf?=?sdsempty();??
  • ????????}??
  • ??
  • ????????server.aof_lastbgrewrite_status?=?REDIS_OK;??
  • ??
  • ????????redisLog(REDIS_NOTICE,?"Background?AOF?rewrite?finished?successfully");??
  • ????????/*?Change?state?from?WAIT_REWRITE?to?ON?if?needed?*/??
  • ????????//下面判斷是否需要打開AOF,比如bgrewriteaofCommand就不需要打開AOF。??
  • ????????if?(server.aof_state?==?REDIS_AOF_WAIT_REWRITE)??
  • ????????????server.aof_state?=?REDIS_AOF_ON;??
  • ??
  • ????????/*?Asynchronously?close?the?overwritten?AOF.?*/??
  • ????????//讓后臺線程去關閉這個舊的AOF文件FD,只要CLOSE就行,會自動unlink的,因為上面已經有rename??
  • ????????if?(oldfd?!=?-1)?bioCreateBackgroundJob(REDIS_BIO_CLOSE_FILE,(void*)(long)oldfd,NULL,NULL);??
  • ??
  • ????????redisLog(REDIS_VERBOSE,??
  • ????????????"Background?AOF?rewrite?signal?handler?took?%lldus",?ustime()-now);??
  • ????}?else?if?(!bysignal?&&?exitcode?!=?0)?{??
  • ????????server.aof_lastbgrewrite_status?=?REDIS_ERR;??
  • ??
  • ????????redisLog(REDIS_WARNING,??
  • ????????????"Background?AOF?rewrite?terminated?with?error");??
  • ????}?else?{??
  • ????????server.aof_lastbgrewrite_status?=?REDIS_ERR;??
  • ??
  • ????????redisLog(REDIS_WARNING,??
  • ????????????"Background?AOF?rewrite?terminated?by?signal?%d",?bysignal);??
  • ????}??
  • ??
  • cleanup:??
  • ????aofRewriteBufferReset();??
  • ????aofRemoveTempFile(server.aof_child_pid);??
  • ????server.aof_child_pid?=?-1;??
  • ????server.aof_rewrite_time_last?=?time(NULL)-server.aof_rewrite_time_start;??
  • ????server.aof_rewrite_time_start?=?-1;??
  • ????/*?Schedule?a?new?rewrite?if?we?are?waiting?for?it?to?switch?the?AOF?ON.?*/??
  • ????if?(server.aof_state?==?REDIS_AOF_WAIT_REWRITE)??
  • ????????server.aof_rewrite_scheduled?=?1;??
  • }??
  • 至此,AOF數據持久化已經全部結束了,剩下的就是一些細節的處理,以及一些Linux庫函數的理解,對于rename、unlink、wait3等庫函數的深入認識就去問Google吧。

    小結

    Redis AOF數據持久化的實現機制通過三篇文章基本上比較詳細的分析了,但這只是從代碼層面去看AOF,對于AOF持久化的優缺點網上有很多分析,Redis的官方網站也有英文介紹,Redis的數據持久化還有一種方法叫RDB,更多RDB的內容等下次再分析。 感謝此篇博客給我在理解Redis AOF數據持久化方面的巨大幫助,http://chenzhenianqing.cn/articles/786.html,此篇博客對AOF的分析十分的詳細。

    ?

    轉載于:https://www.cnblogs.com/davidwang456/p/3521182.html

    《新程序員》:云原生和全面數字化實踐50位技術專家共同創作,文字、視頻、音頻交互閱讀

    總結

    以上是生活随笔為你收集整理的Redis数据持久化机制AOF原理分析一---转的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    97超碰在线久草超碰在线观看 | 国产小视频在线观看 | 中文字幕999 | 青青河边草免费视频 | 久久久久久久久久伊人 | 狠狠狠色丁香婷婷综合激情 | 久久国产片 | 国产在线视频一区二区三区 | 四虎天堂| 就要色综合 | 99久久99久久精品免费 | 国产精品久久久777 成人手机在线视频 | 91观看视频 | 久久精品成人 | 午夜色影院 | 免费一级毛毛片 | 国产精品手机在线播放 | 国产精品一区二区久久久 | 日韩av片在线 | 中文字幕视频一区二区 | 国产视频高清 | 天天色天天艹 | 69国产盗摄一区二区三区五区 | 免费看色网站 | 久久这里 | 久久国产精品区 | 六月丁香色婷婷 | 久久精品一级片 | 久久首页 | 久久精品79国产精品 | 欧美成人高清 | 午夜久久电影网 | 精品国产一区在线观看 | 免费男女羞羞的视频网站中文字幕 | 成人免费在线视频观看 | 久久99精品国产麻豆婷婷 | 超碰国产在线 | 亚洲狠狠婷婷 | 久操视频在线观看 | 黄色aaa毛片 | 激情婷婷色 | 九九综合九九综合 | 亚洲精品国精品久久99热 | 天天射天天干天天插 | 天天爽天天搞 | 综合黄色网 | 日韩va欧美va亚洲va久久 | 天天射天天拍 | 成人国产精品免费 | 综合网伊人 | 国产精品女 | 精品国产一区二区三区久久 | 国产亚洲人 | av中文字幕在线播放 | 天天爽夜夜爽人人爽一区二区 | 日日夜夜草 | 精品视频在线免费观看 | 国产在线观看,日本 | 国产精品免费人成网站 | 久久久国产日韩 | 亚洲人成精品久久久久 | 在线观看免费一级片 | 黄色av影院 | 尤物九九久久国产精品的分类 | 久久字幕精品一区 | 91九色国产 | 免费视频久久久 | av高清网站在线观看 | 成人免费观看视频大全 | 久章草在线 | 午夜精品久久久久久久久久久久 | 激情五月看片 | 免费看一级特黄a大片 | 中文在线字幕免费观看 | 狠狠伊人 | 国产精品自拍在线 | 欧美一级高清片 | 97久久久免费福利网址 | 欧美成人在线免费观看 | 不卡的av在线播放 | 久久免费a | 成人在线你懂得 | 亚洲视频网站在线观看 | 青青河边草手机免费 | 国产视频 久久久 | 日本h视频在线观看 | 日日激情 | 久久国产成人午夜av影院宅 | 午夜精品久久久久久久99 | 成人手机在线视频 | 欧美在线观看视频一区二区 | 狠狠操导航| 中日韩男男gay无套 日韩精品一区二区三区高清免费 | 欧美一级片在线 | 97成人精品视频在线观看 | 日韩伦理片一区二区三区 | 西西444www大胆高清视频 | 丁香网五月天 | 久久综合色婷婷 | a成人v | 三级免费黄 | 成人性生交大片免费观看网站 | 亚洲五月 | 成 人 黄 色 视频免费播放 | 中文字幕国产精品 | 深夜免费福利视频 | 久久黄色小说视频 | 久久视讯 | 久久国内精品99久久6app | 婷婷激情五月 | 精品一二区 | 日韩欧美在线视频一区二区三区 | 国产视频一区精品 | 在线观看日韩国产 | 国产精品女人网站 | 国产成人综合图片 | 婷婷色在线播放 | 国产精品一区二区三区观看 | 最近中文字幕国语免费高清6 | 激情综合网色播五月 | 黄色大片免费网站 | 亚洲最大色 | 丰满少妇高潮在线观看 | 欧美日韩精品免费观看视频 | 国产精品久久久久婷婷二区次 | www日韩视频 | 黄a在线看 | 久久er99热精品一区二区三区 | 久久夜色精品亚洲噜噜国4 午夜视频在线观看欧美 | 国产精品永久免费 | 91精品视频观看 | 国产真实精品久久二三区 | 最新av观看| 韩日在线一区 | 久久中文网 | 99r精品视频在线观看 | 99热在线这里只有精品 | 国产艹b视频| 国产在线精品福利 | 日本久久久久久 | 亚洲一区二区精品视频 | 91桃色免费视频 | 久热电影 | 国产精品毛片网 | 狠狠干我 | 粉嫩av一区二区三区入口 | 精品一区精品二区高清 | 国产精品一区免费在线观看 | 色视频在线看 | 黄色成人影院 | 精品免费久久久久久 | 国内久久精品 | www.色的 | 国产在线欧美 | 亚洲第一香蕉视频 | av在线com | 日韩理论片在线 | 九九九视频精品 | 欧美91视频 | 欧洲在线免费视频 | 亚洲精品字幕 | 91免费的视频在线播放 | 亚洲在线视频免费 | 成人久久毛片 | www黄在线 | 天天干.com | 啪啪动态视频 | 久久天堂亚洲 | 麻豆影视在线观看 | 在线看国产视频 | 久久综合色8888 | 国产二区电影 | 黄色免费在线视频 | 国产一区二区三精品久久久无广告 | 九九热久久免费视频 | 激情视频一区二区三区 | 免费观看黄 | 亚州人成在线播放 | 色网址99 | 国产精品都在这里 | 久久精品美女 | 天天操夜操 | 亚洲欧美视频在线播放 | 在线视频 一区二区 | 青青草国产在线 | 91自拍成人 | 九九有精品 | 91在线观 | 色99之美女主播在线视频 | 97在线看片| 97精品国产97久久久久久粉红 | 国产尤物一区二区三区 | 久久99视频免费观看 | 色综合久久久 | 久久精品国产一区二区电影 | 欧美日韩破处 | 久久毛片网| 国产精品免费久久 | 国产主播大尺度精品福利免费 | 久久精品中文字幕一区二区三区 | 国产精品v a免费视频 | 色天天综合久久久久综合片 | 亚洲砖区区免费 | 九九热国产视频 | 国产黄色免费看 | 亚洲三级在线 | 久久久久久久久毛片精品 | 亚洲久久视频 | 日韩欧美一区二区在线播放 | 91欧美视频网站 | 五月婷婷久久丁香 | 免费视频一级片 | 五月天激情视频在线观看 | 99热这里只有精品国产首页 | 久久综合狠狠综合久久狠狠色综合 | 天天躁天天操 | 国产精品夜夜夜一区二区三区尤 | 国产群p | 欧美一区二区在线免费观看 | 一区二区三区四区精品 | 久久免费在线观看 | 免费的成人av | 欧美一二三视频 | 久草精品视频在线看网站免费 | av丁香花 | 久久免费视频7 | 日韩免费电影网站 | 日韩久久视频 | 97超碰免费在线观看 | 国产精品日韩高清 | 超碰97中文| 亚洲视频资源在线 | www.99在线观看 | 婷婷色婷婷 | 久久论理 | 黄污在线观看 | 免费看v片 | 天天干天天操天天搞 | 97**国产露脸精品国产 | 免费a网址 | 丁香色天天 | 日韩免费中文 | 黄色91在线观看 | 99国产精品免费网站 | 精品国产电影一区二区 | 日本精品在线视频 | 日韩av在线影视 | 婷婷成人在线 | 久久久国产精品成人免费 | 亚洲精品午夜aaa久久久 | av福利第一导航 | 日本精品久久久久中文字幕5 | 91免费网 | 激情电影影院 | 国产成人精品亚洲a | 国产亚州精品视频 | 91大神精品视频在线观看 | 欧美韩国日本在线观看 | 午夜精品99久久免费 | 麻豆视频在线观看免费 | 久久 地址 | 99爱视频在线观看 | 亚洲在线黄色 | 国产精选在线 | 中文字幕在线视频一区 | 色香蕉视频 | 亚洲视频在线看 | 一二区电影 | 日韩在线视 | 精品亚洲国产视频 | 97操操 | 欧洲激情综合 | 亚洲精品乱码久久久久v最新版 | 亚洲三级网站 | 欧美日韩国产精品久久 | 欧美在线久久 | 99久久精品国产免费看不卡 | 成片免费| 在线视频欧美日韩 | 久久久国产精品亚洲一区 | 天天超碰 | 在线观看黄色的网站 | 成片免费观看视频大全 | 天天射综合网视频 | 中文字幕国语官网在线视频 | 黄色大全在线观看 | 成片视频在线观看 | 99精品在线免费视频 | av中文在线 | 天天色天天综合 | 久久久精选 | 久热免费在线观看 | 欧美日韩午夜爽爽 | 久久久久久久久亚洲精品 | 久久免费视频这里只有精品 | 欧美一区二视频在线免费观看 | 国产视频一区二区在线观看 | 一级黄色片毛片 | 九色视频网址 | av不卡中文 | 久精品在线观看 | 夜夜夜精品 | 国产成人777777| 欧美在线视频第一页 | 天天操天天拍 | 欧美精品久久久久性色 | 亚洲精品18日本一区app | 日韩有码在线观看视频 | 福利视频一区二区 | 国产高清在线看 | 久久中文字幕导航 | 国产精品久久久久永久免费观看 | 国产免费午夜 | 有码视频在线观看 | 99久久综合国产精品二区 | 在线免费亚洲 | 天堂av在线网址 | 亚洲天堂网在线视频观看 | 精品久久久久久久久中文字幕 | 99性视频 | 精品国产乱码久久久久久三级人 | 国产专区在线播放 | 国产1级毛片 | 天天在线免费视频 | 日韩av黄 | 国产午夜在线 | 91香蕉视频污在线 | 中文字幕av有码 | 亚洲精品免费看 | 免费网站在线观看成人 | 亚洲欧洲国产日韩精品 | 婷婷丁香社区 | 欧美精品被 | 久久精品视频观看 | 亚洲欧洲日韩在线观看 | 99自拍视频在线观看 | 丁香六月婷婷综合 | 欧美性生活久久 | av视屏在线 | 五月天婷婷综合 | www.久艹 | 色偷偷88欧美精品久久久 | 97理论片| www中文在线| 国产三级国产精品国产专区50 | 亚洲国产中文字幕在线视频综合 | 黄色日本片 | 伊人干综合 | 手机在线黄色网址 | 天天射天天干天天爽 | 亚洲国产成人精品在线观看 | 97网在线观看 | 国产成人精品综合 | 国产精品黄色 | 亚洲人久久| 狠狠干.com | 在线亚洲人成电影网站色www | 中文字幕av在线播放 | 丁香久久久 | 成 人 黄 色 视频免费播放 | 国产精品久久久久久久久久久久 | av免费看网站 | 久久久久久久av麻豆果冻 | 久久精品系列 | 黄色精品免费 | 国产免费又爽又刺激在线观看 | 免费不卡中文字幕视频 | 日韩两性视频 | 天天透天天插 | 色天天久久 | 人人dvd| 精品96久久久久久中文字幕无 | 天天爱天天射天天干天天 | 免费看一级特黄a大片 | 国产美女无遮挡永久免费 | 欧美另类一二三四区 | 亚洲春色综合另类校园电影 | 欧美精品在线一区二区 | 友田真希x88av | 97精品国产一二三产区 | 手机av电影在线 | 国产一区二区在线播放视频 | 狠狠色狠狠色综合日日小说 | 午夜丁香视频在线观看 | 久久久天堂 | 精品欧美一区二区三区久久久 | 日本精品视频免费观看 | 亚洲成人精品久久久 | 99久久精品免费看国产免费软件 | 91免费视频黄| 精品福利在线观看 | 国产精品久久久久久久久久久不卡 | 西西444www高清大胆 | 精品久久久久一区二区国产 | 国产精品不卡 | 久久亚洲福利 | 九九视频网站 | 成人在线观看av | www.夜夜骑.com| 成年人国产在线观看 | 精品国产电影一区二区 | 亚洲欧美视频一区二区三区 | 日本性生活免费看 | 黄色一区三区 | 久久久精品久久日韩一区综合 | 日韩欧美国产激情在线播放 | 婷婷色中文网 | 国产高清一级 | 碰天天操天天 | 99在线免费观看 | 四虎永久国产精品 | 在线成人av | 成人三级视频 | 黄色高清视频在线观看 | 久久激情五月激情 | 欧美日韩视频免费 | 免费观看91| 国产免费一区二区三区最新 | 日韩黄色免费在线观看 | 国产一区二区手机在线观看 | 久久久成人精品 | 91免费国产在线观看 | 五月天免费网站 | 亚洲精品1234区 | 国内视频1区 | 碰超人人 | 婷婷午夜激情 | 免费网站看v片在线a | 在线观看v片 | 精品电影一区 | 超碰97成人 | 美女视频是黄的免费观看 | 免费视频三区 | av女优中文字幕在线观看 | 91理论片午午伦夜理片久久 | 激情丁香综合 | 中文字幕av免费在线观看 | 色丁香久久 | 国产亚洲精品成人av久久影院 | 成人久久视频 | 国产高清中文字幕 | 91在线文字幕 | 91视视频在线直接观看在线看网页在线看 | 亚洲在线视频观看 | 黄在线免费观看 | 不卡电影一区二区三区 | 久久久久久久国产精品视频 | 在线观看国产日韩欧美 | 成人黄大片 | 婷婷射五月 | 99自拍视频在线观看 | 人人超碰97| 99久久精品国产观看 | 国产精品久久久久久久久久久久午夜片 | 不卡中文字幕在线 | 久久久精品 一区二区三区 国产99视频在线观看 | 日日干 天天干 | 亚洲精品在线一区二区 | 一区二区三区电影 | 天天色 天天| 狠狠狠狠狠狠天天爱 | 欧美aa一级片 | 亚洲片在线资源 | 亚洲一级片在线观看 | 久久激五月天综合精品 | 狠狠插狠狠干 | 中文字幕一区二区三区乱码不卡 | 国产一区二区三精品久久久无广告 | 波多野结衣在线视频一区 | 久久久夜色 | 日本精品一区二区三区在线观看 | 超碰人人射| 欧美日韩三级在线观看 | 免费观看丰满少妇做爰 | 91亚色视频在线观看 | 国产精品青草综合久久久久99 | 亚洲高清不卡av | 国产日韩中文在线 | 在线免费观看视频一区二区三区 | 亚洲精品在线观看中文字幕 | 中文字幕之中文字幕 | 国产美女视频黄a视频免费 久久综合九色欧美综合狠狠 | 久久久久国产成人精品亚洲午夜 | 欧美性色xo影院 | 人人涩| 国内久久| 超碰在线97免费 | 97色资源 | 国产精品涩涩屋www在线观看 | 亚洲h在线播放在线观看h | 久久久免费观看完整版 | 精品人人人人 | 久久久久久国产精品999 | 国产精品久久久久久久久久久久午夜 | 五月天堂网 | 高清不卡毛片 | 欧美少妇bbwhd | 国产免费一区二区三区最新 | 国产在线观看地址 | 蜜臀精品久久久久久蜜臀 | 亚洲精品456在线播放乱码 | 国产一区二区精品久久 | 国产色区 | 国产一级特黄电影 | 日本精品中文字幕 | 99久久综合精品五月天 | 欧美精品v国产精品v日韩精品 | 日韩精品久久久久久 | 天天做天天爱天天综合网 | 中文字幕超清在线免费 | 久久婷婷一区二区三区 | 免费h在线观看 | 中文字幕韩在线第一页 | 在线91av| 亚洲成人频道 | 国产精品人人做人人爽人人添 | www夜夜 | 久久久久久久影院 | 欧美日韩高清在线一区 | 九九在线精品视频 | 天天色天天上天天操 | www.黄色在线| 韩国av永久免费 | 99r在线| 亚洲人成人天堂h久久 | 欧美在线aaa | 日韩精品一区二区在线观看视频 | 美女性爽视频国产免费app | 久热国产视频 | 国产精品久久久精品 | 日韩av看片 | 亚洲精品乱码久久 | 久久久久国产成人免费精品免费 | 97av视频| 久久国产亚洲视频 | 国内精品久久久久久中文字幕 | 欧美一区二区三区在线看 | 日韩精品高清不卡 | 韩国av在线播放 | 精品在线播放 | 黄色网址国产 | 日韩色综合 | 一区二区精品在线 | 国产日韩欧美中文 | 98精品国产自产在线观看 | .精品久久久麻豆国产精品 亚洲va欧美 | 伊人婷婷久久 | 91视频在线播放视频 | 中文字幕一区二区三区在线播放 | 国产综合在线视频 | 国产精品久久久久999 | 天天色综合久久 | 狠狠操狠狠干天天操 | 国产精品美女久久久免费 | 日韩高清在线看 | 国产又粗又猛又黄又爽 | 一级片黄色片网站 | 99视频免费播放 | 操操操夜夜操 | 色综合天天色综合 | 中文字幕在线观看1 | 天天爱天天操天天爽 | 欧美日韩午夜爽爽 | 亚洲成人黄色在线 | 五月婷婷丁香在线观看 | 久久99国产精品二区护士 | 2022中文字幕在线观看 | 最新国产精品拍自在线播放 | 日韩免费一区二区 | 久久99亚洲精品久久久久 | 国产免费叼嘿网站免费 | 午夜精品一区二区三区在线视频 | 五月婷婷狠狠 | 久久久久久久av麻豆果冻 | 韩日精品在线 | 久久久久亚洲精品 | 狠狠干夜夜爽 | 国产精品涩涩屋www在线观看 | 日本精品在线看 | 国产成人精品久久久 | 99久久精品国产欧美主题曲 | 国产精品成久久久久三级 | 波多野结衣精品视频 | 五月开心六月伊人色婷婷 | 久久这里只有精品视频99 | 日韩精品一区二区三区不卡 | 人人澡超碰碰 | 91伊人久久大香线蕉蜜芽人口 | 日日操狠狠干 | 69国产精品视频免费观看 | 久久狠狠亚洲综合 | www日 | 午夜99| 国产高清精品在线观看 | 亚洲美女在线一区 | 国产伦精品一区二区三区无广告 | 亚洲免费av在线播放 | 在线观看自拍 | 亚洲视频一 | 欧美国产91 | 开心丁香婷婷深爱五月 | 狠狠色丁香久久婷婷综合丁香 | 激情综合久久 | 99中文字幕视频 | 最新日本中文字幕 | 夜色资源站国产www在线视频 | 黄色av电影一级片 | 五月婷婷视频在线观看 | 国内揄拍国内精品 | 美女视频一区 | 久久综合久久综合久久 | 一本一本久久a久久精品综合 | 开心色婷婷 | 国产一区免费 | 91在线看 | 久久久精品国产一区二区 | zzijzzij亚洲日本少妇熟睡 | 国产aa免费视频 | 黄色在线看网站 | 日本资源中文字幕在线 | 日韩欧美xx | 亚洲第一久久久 | 成人资源在线播放 | 色综合久久综合中文综合网 | 在线看国产日韩 | 国产精品久久久久久久久久不蜜月 | 中文字幕一区二区三区在线观看 | 国产精品久久久久久一区二区三区 | 在线视频 你懂得 | 国产又黄又硬又爽 | 激情久久网 | 久久专区 | 国产精品2020 | 91传媒免费在线观看 | 国产精品二区在线 | 免费观看国产精品视频 | 在线精品国产 | 91视频在线看 | 亚欧日韩av | 视色网站 | 香蕉影视在线观看 | 国产精品久久电影网 | 天天做日日爱夜夜爽 | 国产精品久久久久久久电影 | 亚洲欧美一区二区三区孕妇写真 | 国产精品欧美在线 | 欧美淫视频 | 午夜精品在线看 | 国产高清视频免费观看 | 九色福利视频 | 久久久久久国产精品久久 | 狠狠干网址 | 久久国产精品视频免费看 | 美女国内精品自产拍在线播放 | av在线h| 亚洲人成人在线 | 中文字幕成人一区 | 久久夜色精品国产欧美乱极品 | 中文字幕中文字幕 | 国产高清一区二区 | 97福利在线 | 日韩久久片 | 国产一级片免费视频 | 波多野结依在线观看 | 免费在线观看日韩视频 | 亚洲精品在线免费观看视频 | 久久久久成人免费 | 久久男人视频 | 国产高清日韩欧美 | 96视频免费在线观看 | 亚洲美女视频在线观看 | 激情欧美在线观看 | 中文字幕一区二区三区四区在线视频 | 特级毛片爽www免费版 | 亚洲一二三久久 | 国产高清视频在线免费观看 | 亚洲电影影音先锋 | 国产精品一区二区无线 | 久久99国产精品二区护士 | 久久精品com | 正在播放久久 | 免费av成人在线 | 夜夜操天天摸 | 欧美福利在线播放 | 91一区二区在线 | 欧美日本一区 | 国产亚洲一区 | 久热色超碰 | 精品一区二区影视 | 国产精品一区二区三区四 | 亚洲精品男人的天堂 | 91精品国产自产91精品 | 在线观看国产一区二区 | 亚洲精品视频免费在线观看 | 亚洲精品国产第一综合99久久 | 亚洲精品白浆高清久久久久久 | 成人aaa毛片 | 日韩欧在线 | 又黄又爽免费视频 | 天天综合日日夜夜 | 欧美日韩国产成人 | 国产专区欧美专区 | 999热线在线观看 | 91综合色 | 国产精品自拍在线 | 中文字幕av免费 | 欧美激情第十页 | 九九有精品 | a级国产乱理论片在线观看 伊人宗合网 | 天天综合中文 | 久草视频在线观 | 在线观看911视频 | 91在线看免费 | 成人av日韩 | 91av电影网| 美女免费视频一区 | 日韩中文字幕国产精品 | 高清在线观看av | 激情视频一区 | 国产一区av在线 | www.色午夜.com | 99久久精品免费看国产免费软件 | 青草视频网 | 日p视频在线观看 | 日韩精品免费在线 | 国产高清在线一区 | 久久欧美在线电影 | 超碰在线97观看 | 精产嫩模国品一二三区 | 日韩啪啪小视频 | 久久精品国亚洲 | 久久久精品在线观看 | 在线观看国产日韩 | 91日韩在线视频 | 香蕉视频国产在线 | 精品国产aⅴ麻豆 | 亚洲国产精彩中文乱码av | 精品国产精品国产偷麻豆 | 18做爰免费视频网站 | 91人人射 | 日韩v在线91成人自拍 | 国产精品青草综合久久久久99 | 国产成人免费观看 | 色婷婷久久久 | www.久艹| 在线观看中文字幕dvd播放 | 毛片网在线播放 | 精壮的侍卫呻吟h | 国产视频 亚洲精品 | 国产手机视频 | 91丨九色丨国产在线观看 | 国产一级视频在线观看 | 97超碰中文字幕 | 九九九九免费视频 | 亚洲综合在线观看视频 | 国产精品久久久久久久午夜片 | 91超在线 | 国模精品在线 | 精品国产诱惑 | 国内精品久久久久影院优 | 色播99| 福利一区二区在线 | 99产精品成人啪免费网站 | 久久亚洲区 | 国产精选在线 | 久久人人爽人人片av | 婷婷丁香在线视频 | 热久久影视 | 久久精品看 | 日韩在线观看影院 | 韩国一区二区在线观看 | 亚洲午夜av电影 | 日日干干 | 日韩电影在线观看一区 | 91视频这里只有精品 | 久久欧洲视频 | 久久黄色影视 | 视频三区| 亚洲精品午夜久久久久久久 | 区一区二在线 | 久久久www免费电影网 | 白丝av免费观看 | 午夜狠狠干 | 久久伊99综合婷婷久久伊 | 国产成人av在线影院 | 亚洲欧洲国产精品 | 99久久综合狠狠综合久久 | 亚洲精品国精品久久99热一 | 久草影视在线观看 | 99久久er热在这里只有精品66 | 一区二区三区视频 | 国产一级精品视频 | 国产 日韩 欧美 自拍 | 欧美9999| 亚洲尺码电影av久久 | 国产成人精品在线观看 | 亚洲永久字幕 | 中文字幕在线观 | 国产美女免费观看 | 日韩在线中文字幕视频 | 欧美疯狂性受xxxxx另类 | 夜色在线资源 | 日韩av中文字幕在线免费观看 | 精品在线观看一区二区 | 免费中文字幕 | 24小时日本在线www免费的 | 怡春院av| 国产精品18videosex性欧美 | 国产喷水在线 | 国产一区精品在线观看 | 91视频a | 97操操操| 亚洲免费在线观看视频 | 欧美一级片免费播放 | 日韩精品综合在线 | 蜜臀av夜夜澡人人爽人人 | 99精品在线免费视频 | 怡红院久久 | 亚洲在线观看av | 激情综合五月网 | 日韩视频一区二区三区在线播放免费观看 | 久操视频在线观看 | 狠狠激情中文字幕 | 午夜性色 | 精油按摩av| 一二区av | 亚洲免费小视频 | 狠狠色噜噜狠狠狠狠2022 | 在线国产专区 | 黄毛片在线观看 | 亚洲国产av精品毛片鲁大师 | 国产精品久久久久久久久久久杏吧 | 99免费国产| 国产成人无码AⅤ片在线观 日韩av不卡在线 | 色婷婷久久久 | 福利视频精品 | 午夜色婷婷 | 最近最新最好看中文视频 | 国产亚洲精品久久19p | 三级小视频在线观看 | 欧美福利网站 | 不卡中文字幕av | 97人人看| 亚洲另类视频 | 深夜免费福利在线 | 毛片黄色一级 | 色综合久久五月天 | 国产破处视频在线播放 | 久久夜视频 | 91精品国自产在线观看欧美 | 欧美日韩精品在线视频 | 免费午夜视频在线观看 | 亚洲永久字幕 | 91禁在线看 | 91香蕉视频在线下载 | 黄色精品国产 | 欧美精品在线观看一区 | 日日干夜夜骑 | 欧美最猛性xxx | 欧美韩日在线 | 免费看v片 | 久久婷婷色综合 | 成人欧美一区二区三区在线观看 | 三上悠亚一区二区在线观看 | 99久久99| 久久欧美在线电影 | 久久精品视频网 | 免费一区在线 | 亚洲激情精品 | 六月丁香六月婷婷 | 欧美日韩一区二区免费在线观看 | 久久一线| 久久久人人人 | 免费看的视频 | 日韩城人在线 | 激情伊人五月天 | 亚洲九九九在线观看 | 在线观看免费成人 | 午夜视频在线观看一区二区三区 | 久久人人爽人人爽人人片 | 又黄又刺激 | 91九色在线视频观看 | 国产一级视频免费看 | 午夜久久影视 | 色欧美日韩 | 探花视频在线观看+在线播放 | 亚洲精品91天天久久人人 | 日韩在线观看av | 青青看片 | 日本成人免费在线观看 | 九九在线视频免费观看 | 91av免费看 | 91精品啪 | 黄色三级免费网址 | 五月天激情在线 | av中文字幕网 | 中文在线a在线 | 四季av综合网站 | 欧美夫妻生活视频 | 久久久精品网站 | 国产呻吟在线 | 99久久99久国产黄毛片 | 在线免费黄网站 | 人人玩人人添人人澡97 | 免费人成在线观看 | 久久久国产精品电影 | 久久a热6| 最近中文字幕第一页 | 亚洲一区视频在线播放 | 欧美亚洲国产一卡 | 日批视频在线观看免费 | 久久精品一二三区 | 美女视频黄免费的久久 | 欧美激情视频一区 | 久久国产精品视频免费看 | 韩国av免费在线 | 在线观看免费观看在线91 | 久久久久成人精品免费播放动漫 | 日韩成人欧美 | 久久久精品国产一区二区 | 精品视频专区 | 欧美激情第一区 | 精品国产伦一区二区三区观看方式 | 成人免费视频播放 | 久久国产三级 | 国产精品久久久久久久久蜜臀 | 99色 | 日韩久久激情 | 久久免费视频4 | 99久久婷婷国产一区二区三区 | 丝袜制服综合网 | 91在线网址 | 国产精品久久久久免费a∨ 欧美一级性生活片 | 色综合天天狠狠 | 国内精品在线看 | 久草精品在线播放 | 九九久久电影 | 精品国产一区二区三区久久久 | 成人在线视频在线观看 | 亚洲欧美国产精品久久久久 | 日韩精品一区二区在线视频 | 国产在线va | 国产精品精品久久久久久 | www五月天com| 亚洲综合视频在线播放 | 一级黄色电影网站 | 午夜精品久久久久久 | 免费欧美高清视频 | 91在线免费看片 | 美女视频久久 | 天天干.com | 亚洲国产伊人 | 婷婷在线精品视频 | 免费a v观看 | 国产高清视频免费观看 | 天天曰天天爽 | 国产中出在线观看 | 天天色天天综合 | 亚洲最快最全在线视频 | 久久久免费毛片 | 久久精品中文字幕 | 国产一级小视频 | 久草在线免费资源 | 欧美亚洲三级 | 国产日韩一区在线 | 人人爽人人爽人人爽人人爽 | 国产视频久久久久 | 色偷偷97 | 国产精品一区专区欧美日韩 | 天天天色综合 | 国内精品久久久久久久久久 | 国产精品中文 | 国产精品成人a免费观看 | 中文字幕日韩免费视频 | 久久精品国产久精国产 | 久草久视频 | 欧美国产亚洲精品久久久8v | 婷婷激情综合网 | 久久久久久久久久伊人 | 免费观看一级 | 天堂在线一区二区 | 日韩在线短视频 | 在线精品亚洲 | 日日操操 | 久久免费视频一区 | 丝袜少妇在线 | 欧美激情综合五月色丁香 | 国产中文字幕在线 | 亚洲精选99 | 91麻豆操 | 天天摸夜夜添 | 天天天色综合a | 一区二区av | 中文字幕观看视频 | 五月婷婷综合在线 | 丁香花在线观看视频在线 | 免费国产在线观看 | 免费看一级一片 |