日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當(dāng)前位置:首頁 > > 充電吧
[導(dǎo)讀]問題? ? ?我們在生產(chǎn)環(huán)境中使用SQLite時中發(fā)現(xiàn)建表報“table xxx already exists”錯誤,但DB文件中并沒有該表。后面才發(fā)現(xiàn)這個是SQLite在實現(xiàn)過程中的一個bug,而這

問題

? ? ?我們在生產(chǎn)環(huán)境中使用SQLite時中發(fā)現(xiàn)建表報“table xxx already exists”錯誤,但DB文件中并沒有該表。后面才發(fā)現(xiàn)這個是SQLite在實現(xiàn)過程中的一個bug,而這個bug與數(shù)據(jù)字典的一致性相關(guān),下面這篇文章主要討論SQLite的緩存機制,以及緩存一致性實現(xiàn)的策略,希望對大家了解SQLite緩存機制有一定的幫助。

緩存

? ? ??SQLite中緩存主要包括兩方面,數(shù)據(jù)字典緩存和數(shù)據(jù)頁緩存。SQLite本身是一個文件數(shù)據(jù)庫,所有的數(shù)據(jù)都在一個DB文件中,文件以塊(page)的形式存放,默認(rèn)情況下每個page是1024個字節(jié)。為了避免每次訪問都產(chǎn)生磁盤IO,針對數(shù)據(jù)塊在SQLite內(nèi)部實現(xiàn)了一層緩存
pagecache,pagecache的作用就是緩存頁數(shù)據(jù)。在SQLite內(nèi)部,除了用戶數(shù)據(jù),還有一部分內(nèi)容是元數(shù)據(jù)信息,包括表,視圖,索引和觸發(fā)器等,這部分元數(shù)據(jù)信息在數(shù)據(jù)庫領(lǐng)域一般稱為數(shù)據(jù)字典,這部分信息也存在DB文件中。由于每次執(zhí)行語句時,都需要數(shù)據(jù)字典進行語義分析和執(zhí)行計劃優(yōu)化(表是否存在,列是否存在,是否有索引可用,是否存在觸發(fā)器等),如果每次獲取這些信息時,都需要從DB文件中獲取,則非常影響性能。你可能會說,不是已經(jīng)有pagecache了嗎?對的,數(shù)據(jù)字典的內(nèi)容也緩存在pagecahce中,但是,要知道page中的數(shù)據(jù)都是二進制的,需要對內(nèi)容進行解析產(chǎn)生結(jié)構(gòu)化數(shù)據(jù)才能使用。為此,為了避免分析語句時,頻繁解析獲取數(shù)據(jù)字典,將解析好的數(shù)據(jù)進行緩存,以供多次使用,提高效率。

數(shù)據(jù)頁緩存一致性
? ? ?我們這里討論的數(shù)據(jù)頁緩存對應(yīng)MySQL的概念就是BufferPool,當(dāng)然其它數(shù)據(jù)庫Oracle,SQLServer都有類似的概念。
傳統(tǒng)PC上面的數(shù)據(jù)庫,都是在數(shù)據(jù)庫服務(wù)啟動時,根據(jù)參數(shù)設(shè)定值一次性分配特定大小的BufferPool。而SQLite采用懶分配策略,即“用多少則分配多少”,pagecache默認(rèn)大小是2000個page,2000個page可以認(rèn)為是一個緩存的上限。一次性分配的好處是,內(nèi)存在物理是連續(xù)的,不容易產(chǎn)生內(nèi)存碎片;而懶分配則更節(jié)約內(nèi)存,由于SQLite一般用于端設(shè)備,采用懶分配方式可能更經(jīng)濟實惠。SQLite的緩存分配策略采用LRU,保留最近訪問的page,淘汰最老的page。
? ? ? SQLite中每個數(shù)據(jù)庫連接對應(yīng)一個DB句柄,應(yīng)用通過DB句柄來操作數(shù)據(jù)庫,而pagecache實際上就作為一個成員掛在DB句柄中,因此每個DB句柄都有自己獨立的緩存,這點與傳統(tǒng)的PC數(shù)據(jù)庫不同(比如MySQL中,所有連接共享BufferPool)。既然每個DB句柄有獨立的緩存,那么緩存之間如何同步?比如有Connection1和Connection2兩個連接,Connection1首先從文件中讀取了page_A并加入到了緩存;隨后Connection2也從文件中讀取Page_A,并進行了更新;那么當(dāng)Connection1再次讀取page_A時,Connection1如何知道自己緩存的page_A已經(jīng)不是最新了,需要重新到DB文件中讀?。?br />SQLite為了處理這個問題,在DB的文件控制頭中存放的DB的版本信息,開始執(zhí)行SQL時會讀取DB的版本信息并緩存,如何發(fā)現(xiàn)本次的版本信息與之前的不同,則確認(rèn)DB文件已經(jīng)被修改,清理自身的緩存。每次事務(wù)提交時,都會調(diào)用pager_write_changecounter進行更新,具體位置在第一頁的第24個字節(jié),占4個字節(jié)。

數(shù)據(jù)字典緩存一致性
? ? ?我們這里討論的數(shù)據(jù)字典對應(yīng)MySQL的概念就是information_schema的系統(tǒng)表,字典緩存就是對系統(tǒng)表信息的結(jié)構(gòu)化信息存儲。在SQLite中字典信息采用Hash表存儲,包括(tblHash,idxHash,trigHash和fkeyHash等)判斷一個對象是否存在的依據(jù)是Hash表中對象是否存在。openDatabase函數(shù)通過調(diào)用sqlite3Init對數(shù)據(jù)字典進行初始化,并設(shè)置標(biāo)記。與數(shù)據(jù)頁緩存一樣,字典緩存也是每個DB句柄有單獨的一份數(shù)據(jù),同樣的,SQLite文件頭中同樣存放了數(shù)據(jù)字典的版本信息,具體位置在第一頁的第40個字節(jié),占4個字節(jié)。進行DDL操作時(CREATE,DROP,ALTER等),會調(diào)用sqlite3ChangeCookie更新字典版本號(Schema cookie)。在Prepare階段分析語句時,若發(fā)現(xiàn)對象不存在,會觸發(fā)一次Schema cookie檢查,如果數(shù)據(jù)字典不是最新,則會調(diào)用sqlite3SchemaClear進行清理,并重新加載數(shù)據(jù)字典。另外,SQLite的數(shù)據(jù)字典表非常簡單,主要在sqlite_master表中,每個對象都是一行記錄,記錄中包含了表定義,加載字典時,實際就是將表定義語句分析一遍,通過調(diào)用sqlite3EndTable將對象加入Hash表,非常方便。

小結(jié)
? ? ?可以看到,無論數(shù)據(jù)頁緩存也好,數(shù)據(jù)字典緩存也好,SQLite都是采用一個版本號來控制版本信息,非常簡單實用,但缺點是粒度非常大。如果DB寫非常頻繁,那么每次讀基本都會導(dǎo)致物理IO,可能修改的是A表,訪問B表也需要將緩存清空。這里也可以解釋為什么頁緩存是“懶加載”模式,這樣清空緩存的代價也相對較小。對于數(shù)據(jù)字典緩存,粒度同樣很粗,每修改一個表,視圖,觸發(fā)器等對象,都會觸發(fā)數(shù)據(jù)字典版本更新。當(dāng)然SQLite不會傻傻的每次執(zhí)行SQL時都去判斷自己的版本是否最新,只是在訪問對象時,對象不存在的情況才去檢查版本,這樣在一定程度上減少了加載的次數(shù),但這樣也帶來了問題,下面回到問題本身。

回到問題
? ? ?前面我們拋出了一個SQLite的bug,這里來細說來龍去脈。假設(shè)有兩個DB句柄,分別稱為A和B。執(zhí)行如下序列: A:create table t(id int); B:DROP table if exists t; A: create table t(id int); 第二次A建表時會報“table t already exists”錯誤,而實際上表已經(jīng)不存在了。這主要原因就是第3步A建表時發(fā)現(xiàn)表存在并沒有觸發(fā)去判斷數(shù)據(jù)字典是否最新的邏輯,導(dǎo)致誤報。復(fù)現(xiàn)該問題時要注意關(guān)閉sharecache,因為在sharecache模式下,所有的DB句柄共享一個緩存區(qū)。其實問題很簡單,但猜測復(fù)現(xiàn)問題還是花了一點精力。


本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動電源

在工業(yè)自動化蓬勃發(fā)展的當(dāng)下,工業(yè)電機作為核心動力設(shè)備,其驅(qū)動電源的性能直接關(guān)系到整個系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動勢抑制與過流保護是驅(qū)動電源設(shè)計中至關(guān)重要的兩個環(huán)節(jié),集成化方案的設(shè)計成為提升電機驅(qū)動性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機 驅(qū)動電源

LED 驅(qū)動電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個照明設(shè)備的使用壽命。然而,在實際應(yīng)用中,LED 驅(qū)動電源易損壞的問題卻十分常見,不僅增加了維護成本,還影響了用戶體驗。要解決這一問題,需從設(shè)計、生...

關(guān)鍵字: 驅(qū)動電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動電源的公式,電感內(nèi)電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計 驅(qū)動電源

電動汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動汽車的核心技術(shù)之一是電機驅(qū)動控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機驅(qū)動系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動汽車的動力性能和...

關(guān)鍵字: 電動汽車 新能源 驅(qū)動電源

在現(xiàn)代城市建設(shè)中,街道及停車場照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進步,高亮度白光發(fā)光二極管(LED)因其獨特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動電源 LED

LED通用照明設(shè)計工程師會遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動電源的電磁干擾(EMI)問題成為了一個不可忽視的挑戰(zhàn)。電磁干擾不僅會影響LED燈具的正常工作,還可能對周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動電源

開關(guān)電源具有效率高的特性,而且開關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機重量也有所下降,所以,現(xiàn)在的LED驅(qū)動電源

關(guān)鍵字: LED 驅(qū)動電源 開關(guān)電源

LED驅(qū)動電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動電源
關(guān)閉