真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

使用Redis優(yōu)化查詢性能的實(shí)踐是怎樣的

使用redis優(yōu)化查詢性能的實(shí)踐是怎樣的,很多新手對(duì)此不是很清楚,為了幫助大家解決這個(gè)難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。

創(chuàng)新互聯(lián)公司長期為千余家客戶提供的網(wǎng)站建設(shè)服務(wù),團(tuán)隊(duì)從業(yè)經(jīng)驗(yàn)10年,關(guān)注不同地域、不同群體,并針對(duì)不同對(duì)象提供差異化的產(chǎn)品和服務(wù);打造開放共贏平臺(tái),與合作伙伴共同營造健康的互聯(lián)網(wǎng)生態(tài)環(huán)境。為鄂州企業(yè)提供專業(yè)的網(wǎng)站制作、網(wǎng)站建設(shè),鄂州網(wǎng)站改版等技術(shù)服務(wù)。擁有十年豐富建站經(jīng)驗(yàn)和眾多成功案例,為您定制開發(fā)。

應(yīng)用背景

有一個(gè)應(yīng)用需要上傳一組ID到服務(wù)器來查詢這些ID所對(duì)應(yīng)的數(shù)據(jù),數(shù)據(jù)庫中存儲(chǔ)的數(shù)據(jù)量是7千萬,每次上傳的ID數(shù)量一般都是幾百至上千數(shù)量級(jí)別。

以前的解決方案

  1. 數(shù)據(jù)存儲(chǔ)在Oracle中,為ID建立了索引;

  2. 查詢時(shí),先將這些上傳的ID數(shù)據(jù)存儲(chǔ)到臨時(shí)表中,然后用表關(guān)聯(lián)的方法來查詢。

這樣做的優(yōu)點(diǎn)是減少了查詢次數(shù)(不用每個(gè)ID都查詢一次),減少了解析SQL的時(shí)間(只需要執(zhí)行1次查詢SQL,但是多了插入數(shù)據(jù)的SQL處理時(shí)間)。

但是這樣的設(shè)計(jì)仍然存在巨大的提升空間,當(dāng)并發(fā)查詢的數(shù)量增加時(shí),數(shù)據(jù)庫的響應(yīng)就會(huì)很久。雖然建立了索引,但是每個(gè)ID查詢的時(shí)間復(fù)雜度仍是O(logn)級(jí)別的,那么總的查詢時(shí)間復(fù)雜度就應(yīng)該是m*O(logn)。不知道Oracle對(duì)表關(guān)聯(lián)查詢有做了哪些優(yōu)化,但應(yīng)該也是改變不了時(shí)間復(fù)雜度的級(jí)別。

解決方法

一遇到讀數(shù)據(jù)庫存在瓶頸的問題,首先想到的就是要用內(nèi)存數(shù)據(jù)庫,用緩存來解決。首選 Redis,因?yàn)镽edis是一種提供了豐富數(shù)據(jù)結(jié)構(gòu)的key-value數(shù)據(jù)庫,value可以存儲(chǔ)STRING(字符串)、HASH(哈希),LIST(列表),ZSET(有序集)。

首先需要將數(shù)據(jù)的存儲(chǔ)改成 key-value 架構(gòu)。簡單的做法就是一個(gè)ID對(duì)應(yīng)一個(gè)字符串的 Value。但是一個(gè) ID 可以對(duì)應(yīng)多條數(shù)據(jù),而且一條數(shù)據(jù)內(nèi)又可以包含多個(gè)字段。這時(shí)候就需要將這些數(shù)據(jù)重新組合一下,拼在一起,或者是采用列表、哈?;蚣蟻泶鎯?chǔ) Value。

Redis內(nèi)部采用 HashTable(哈希表)來實(shí)現(xiàn)key-value的數(shù)據(jù)結(jié)構(gòu),是一種空間占用較高的數(shù)據(jù)結(jié)構(gòu)。而我的應(yīng)用場景又是ID有幾千萬規(guī)模的,如果按上述方法,使用每個(gè)ID作為key,那么內(nèi)存的消耗將是巨大的。每個(gè)key-vaulue結(jié)構(gòu),Redis本身的維護(hù)開銷就要80幾字節(jié),即便value存儲(chǔ)的是純數(shù)字(會(huì)使用long類型,占用4個(gè)字節(jié)),也依然很大,1000萬的數(shù)據(jù),就要占用快1G內(nèi)存。

使用兩級(jí)Hash優(yōu)化內(nèi)存

依據(jù)官方文檔的內(nèi)存優(yōu)化方法,以及這篇文章 節(jié)約內(nèi)存:Instagram的Redis實(shí)踐,建議對(duì)ID分段作為key,并使用 hash 來存儲(chǔ)第一級(jí) key 的 value,第二級(jí)存儲(chǔ)較少的數(shù)據(jù)量(推薦1000),因此第二級(jí)的key使用ID的后3位。

為了節(jié)約內(nèi)存,Redis默認(rèn)使用ziplist(壓縮列表)來存儲(chǔ)HASH(哈希),LIST(列表),ZSET(有序集)這些數(shù)據(jù)結(jié)構(gòu)。當(dāng)某些條件被滿足時(shí),自動(dòng)轉(zhuǎn)換成 hash table(哈希表),linkedlist(雙端列表),skiplist(跳表)。

ziplist是用一個(gè)數(shù)組來實(shí)現(xiàn)的雙向鏈表結(jié)構(gòu),顧名思義,使用ziplist可以減少雙向鏈表的存儲(chǔ)空間,主要是節(jié)省了鏈表指針的存儲(chǔ),如果存儲(chǔ)指向上一個(gè)鏈表結(jié)點(diǎn)和指向下一個(gè)鏈表結(jié)點(diǎn)的指針需要8個(gè)字節(jié),而轉(zhuǎn)化成存儲(chǔ)上一個(gè)結(jié)點(diǎn)長度和當(dāng)前結(jié)點(diǎn)長度在大多數(shù)情況下可以節(jié)省很多空間(最好的情況下只需2個(gè)字節(jié))。但是每次向鏈表增加元素都需要重新分配內(nèi)存?!?引用自這里的描述

ziplist的詳細(xì)信息請看 Redis book ziplist 章節(jié)

查看 Redis 的 .conf 文件,可以查看到轉(zhuǎn)換條件的設(shè)置信息。

# Hashes are encoded using a memory efficient data structure when they have a# small number of entries, and the biggest entry does not exceed a given# threshold. These thresholds can be configured using the following directives.hash-max-ziplist-entries 512hash-max-ziplist-value 64# Similarly to hashes, small lists are also encoded in a special way in order# to save a lot of space. The special representation is only used when# you are under the following limits:list-max-ziplist-entries 512list-max-ziplist-value 64# Similarly to hashes and lists, sorted sets are also specially encoded in# order to save a lot of space. This encoding is only used when the length and# elements of a sorted set are below the following limits:zset-max-ziplist-entries 128zset-max-ziplist-value 64

ziplist 查找的時(shí)間復(fù)雜度是 O(N),但是數(shù)據(jù)量較少,第二級(jí)Hash的查詢速度依然在O(1)級(jí)別。

對(duì)第二級(jí)Hash存儲(chǔ)的數(shù)據(jù)再編碼

在我的應(yīng)用場景中每個(gè)ID對(duì)應(yīng)的數(shù)據(jù)可以有很多個(gè)字段,這些字段有很多實(shí)際上是類型數(shù)據(jù),存儲(chǔ)的也是ID。為了進(jìn)一步節(jié)約內(nèi)存,對(duì)這些使用數(shù)字作為ID的字段,采用base62編碼(0-9,A-Z,a-z),這樣可以使這些ID的字符長度變短,進(jìn)一步減少在Redis中第二級(jí)hash需要存儲(chǔ)的數(shù)據(jù)量,從而減少Redis占用的內(nèi)存。

使用Lua腳本來處理批量操作

由于每次查詢都上傳幾百上千個(gè)ID,如果對(duì)這些ID,都單獨(dú)調(diào)用一次HGET命令,那么一次查詢就需要上千次TCP通信,速度很慢。這個(gè)時(shí)候最好的方法就是一次性將所有的查詢都發(fā)送到 Redis Server,然后在 Redis Server 處再依次執(zhí)行HGET命令,這個(gè)時(shí)候就要用到 Redis 的Pipelining(管道),Lua 腳本(需要 Redis 2.6以上版本)。這兩項(xiàng)功能可以用來處理批量操作。由于Lua腳本更簡單好用,因此我就直接選用Lua腳本。

Redis Lua 腳本具有原子性,執(zhí)行過程會(huì)鎖住 Redis Server,因此 Redis Server 會(huì)全部執(zhí)行完 Lua 腳本里面的所有命令,才會(huì)去處理其他命令請求,不用擔(dān)心并發(fā)帶來的共享資源讀寫需要加鎖問題。實(shí)際上所有的 Redis 命令都是原子的,執(zhí)行任何 Redis 命令,包括 info,都會(huì)鎖住 Redis Server。

不過需要注意的是:

為了防止某個(gè)腳本執(zhí)行時(shí)間過長導(dǎo)致Redis無法提供服務(wù)(比如陷入死循環(huán)),Redis提供了lua-time-limit參數(shù)限制腳本的最長運(yùn)行時(shí)間,默認(rèn)為5秒鐘(見.conf配置文件)。當(dāng)腳本運(yùn)行時(shí)間超過這一限制后,Redis將開始接受其他命令但不會(huì)執(zhí)行(以確保腳本的原子性,因?yàn)榇藭r(shí)腳本并沒有被終止),而是會(huì)返回"BUSY"錯(cuò)誤——引用自這里的描述

遇到這種情況,就需要使用 SCRIPT KILL 命令來終止 Lua 腳本的執(zhí)行。因此,千萬要注意 Lua 腳本不能出現(xiàn)死循環(huán),也不要用來執(zhí)行費(fèi)時(shí)的操作。

性能分析

測試環(huán)境:

  • 內(nèi)存:1333MHz

  • CPU:Intel Core i3 2330M 2.2GHz

  • 硬盤:三星 SSD

實(shí)驗(yàn)基本設(shè)置:

  1. 將7000萬數(shù)據(jù)按照上面描述的方法,使用兩級(jí)Hash以及對(duì)數(shù)據(jù)再編碼,存儲(chǔ)到Redis中。

  2. 模擬數(shù)據(jù)請求(沒有通過HTTP請求,直接函數(shù)調(diào)用),查詢數(shù)據(jù),生成響應(yīng)的JSON數(shù)據(jù)。

(數(shù)據(jù)僅供參考,因?yàn)槲凑嬲Y(jié)合Web服務(wù)器進(jìn)行測試)

使用上述方法,對(duì)Redis的內(nèi)存優(yōu)化效果非常好。

實(shí)驗(yàn)設(shè)置:

  1. 模擬每次查詢500個(gè)ID,分批次連續(xù)查詢。用于模擬測試并發(fā)情況下的查詢性能。

響應(yīng)速度與查詢的數(shù)據(jù)量,幾乎是線性相關(guān)。30s 的時(shí)間就可以處理2000次請求,100W個(gè)ID的查詢。由于Oracle速度實(shí)在太慢,就不做測試了。

實(shí)驗(yàn)設(shè)置:

  1. 連續(xù)查詢1W個(gè)ID,每次500個(gè),分20次。用于測試Redis中存儲(chǔ)的數(shù)據(jù)量對(duì)查詢性能的影響。

查詢速度受存儲(chǔ)數(shù)據(jù)量的影響較小。當(dāng)存儲(chǔ)的數(shù)據(jù)量較多時(shí),第二級(jí)hash存儲(chǔ)的數(shù)據(jù)量就會(huì)更多,因此查詢時(shí)間會(huì)有略微的上升,但依然很快。

看完上述內(nèi)容是否對(duì)您有幫助呢?如果還想對(duì)相關(guān)知識(shí)有進(jìn)一步的了解或閱讀更多相關(guān)文章,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝您對(duì)創(chuàng)新互聯(lián)的支持。


文章名稱:使用Redis優(yōu)化查詢性能的實(shí)踐是怎樣的
文章出自:http://weahome.cn/article/iegdop.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部