关于memecache的使用及清楚示意
Memcache是danga.com的一個項目,最早是為 LiveJournal 服務(wù)的,目前全世界不少人使用這個緩存項目來構(gòu)建自己大負(fù)載的網(wǎng)站,來分擔(dān)數(shù)據(jù)庫的壓力。它可以應(yīng)對任意多個連接,使用非阻塞的網(wǎng)絡(luò)IO。由于它的工作機 制是在內(nèi)存中開辟一塊空間,然后建立一個HashTable,Memcached自管理這些HashTable。Memcache官方網(wǎng) 站:http://www.danga.com/memcached,更多詳細(xì)的信息可以來這里了解。
為什么會有Memcache和memcached兩種名稱?其實Memcache是這個項目的名稱,而memcached是它服務(wù)器端的主程序文件名,知道我的意思了把~~~~。一個是項目名稱,一個是主程序文件名,在網(wǎng)上看到了很多人不明白,于是混用了。
Memcache的安裝
分為兩個過程:memcache服務(wù)器端的安裝和memcached客戶端的安裝。
所謂服務(wù)器端的安裝就是在服務(wù)器(一般都是linux系統(tǒng))上安裝Memcache實現(xiàn)數(shù)據(jù)的存儲。
所謂客戶端的安裝就是指php(或者其他程序,Memcache還有其他不錯的api接口提供)去使用服務(wù)器端的Memcache提供的函數(shù),需要php添加擴展。
PHP的Memcache
| 01 | <?php |
| 02 | ????//連接 |
| 03 | ????$mem = new Memcache; |
| 04 | ????$mem->connect("db.nowamagic.net", 12000); |
| 05 | ????//保存數(shù)據(jù) |
| 06 | ????$mem->set('key1', 'This is first value', 0, 60); |
| 07 | ????$val = $mem->get('key1'); |
| 08 | ????echo "Get key1 value: " . $val ."<br />"; |
| 09 | ????//替換數(shù)據(jù) |
| 10 | ????$mem->replace('key1', 'This is replace value', 0, 60); |
| 11 | ????$val = $mem->get('key1'); |
| 12 | ????echo "Get key1 value: " . $val . "<br />"; |
| 13 | ????//保存數(shù)組 |
| 14 | ????$arr = array('aaa', 'bbb', 'ccc', 'ddd'); |
| 15 | ????$mem->set('key2', $arr, 0, 60); |
| 16 | ????$val2 = $mem->get('key2'); |
| 17 | ????echo "Get key2 value: "; |
| 18 | ????print_r($val2); |
| 19 | ????echo "<br />"; |
| 20 | ????//刪除數(shù)據(jù) |
| 21 | ????$mem->delete('key1'); |
| 22 | ????$val = $mem->get('key1'); |
| 23 | ????echo "Get key1 value: " . $val . "<br />"; |
| 24 | ????//清除所有數(shù)據(jù) |
| 25 | ????$mem->flush(); |
| 26 | ????$val2 = $mem->get('key2'); |
| 27 | ????echo "Get key2 value: "; |
| 28 | ????print_r($val2); |
| 29 | ????echo "<br />"; |
| 30 | ????//關(guān)閉連接 |
| 31 | ????$mem->close(); |
| 32 | ?> |
如果正常的話,瀏覽器將輸出:
| 1 | Get key1 value: This is first value |
| 2 | Get key1 value: This is replace value |
| 3 | Get key2 value: Array ( [0] => aaa [1] => bbb [2] => ccc [3] => ddd ) |
| 4 | Get key1 value: |
| 5 | Get key2 value: |
程序代碼分析
初始化一個Memcache的對象:$mem = new Memcache;
連接到我們的Memcache服務(wù)器端,第一個參數(shù)是服務(wù)器的IP地址,也可以是主機名,第二個參數(shù)是Memcache的開放的端口:$mem->connect("192.168.0.200", 12000);
保存一個數(shù)據(jù)到Memcache服務(wù)器上,第一個參數(shù)是數(shù)據(jù)的key,用來定位一個數(shù)據(jù),第二個參數(shù)是需要保存的數(shù)據(jù)內(nèi)容,這里是一個字符 串,第三個參數(shù)是一個標(biāo)記,一般設(shè)置為0或者MEMCACHE_COMPRESSED就行了,第四個參數(shù)是數(shù)據(jù)的有效期,就是說數(shù)據(jù)在這個時間內(nèi)是有效 的,如果過去這個時間,那么會被Memcache服務(wù)器端清除掉這個數(shù)據(jù),單位是秒,如果設(shè)置為0,則是永遠(yuǎn)有效,我們這里設(shè)置了60,就是一分鐘有效時 間:$mem->set(‘key1‘, ‘This is first value’, 0, 60);
從Memcache服務(wù)器端獲取一條數(shù)據(jù),它只有一個參數(shù),就是需要獲取數(shù)據(jù)的key,我們這里是上一步設(shè)置的key1,現(xiàn)在獲取這個數(shù)據(jù)后輸出輸出:
| 1 | $val = $mem->get(’key1′); |
| 2 | echo "Get key1 value: " . $val; |
現(xiàn)在是使用replace方法來替換掉上面key1的值,replace方法的參數(shù)跟set是一樣的,不過第一個參數(shù)key1是必須是要替換數(shù)據(jù)內(nèi)容的key,最后輸出了:
| 1 | $mem->replace('key1', 'This is replace value', 0, 60); |
| 2 | $val = $mem->get('key1'); |
| 3 | echo "Get key1 value: " . $val; |
同樣的,Memcache也是可以保存數(shù)組的,下面是在Memcache上面保存了一個數(shù)組,然后獲取回來并輸出:
| 1 | $arr = array('aaa', 'bbb', 'ccc', 'ddd'); |
| 2 | $mem->set('key2', $arr, 0, 60); |
| 3 | $val2 = $mem->get('key2'); |
| 4 | print_r($val2); |
現(xiàn)在刪除一個數(shù)據(jù),使用delte接口,參數(shù)就是一個key,然后就能夠把Memcache服務(wù)器這個key的數(shù)據(jù)刪除,最后輸出的時候沒有結(jié)果:
| 1 | $mem->delete('key1'); |
| 2 | $val = $mem->get('key1'); |
| 3 | echo "Get key1 value: " . $val . "<br />"; |
最后我們把所有的保存在Memcache服務(wù)器上的數(shù)據(jù)都清除,會發(fā)現(xiàn)數(shù)據(jù)都沒有了,最后輸出key2的數(shù)據(jù)為空,最后關(guān)閉連接:
| 1 | $mem->flush(); |
| 2 | $val2 = $mem->get('key2'); |
| 3 | echo "Get key2 value: "; |
| 4 | print_r($val2); |
| 5 | echo "<br />"; |
Memcache的使用
使用Memcache的網(wǎng)站一般流量都是比較大的,為了緩解數(shù)據(jù)庫的壓力,讓Memcache作為一個緩存區(qū)域,把部分信息保存在內(nèi)存中,在 前端能夠迅速的進行存取。那么一般的焦點就是集中在如何分擔(dān)數(shù)據(jù)庫壓力和進行分布式,畢竟單臺Memcache的內(nèi)存容量的有限的。我這里簡單提出我的個 人看法,未經(jīng)實踐,權(quán)當(dāng)參考。
分布式應(yīng)用
Memcache本來支持分布式,我們客戶端稍加改造,更好的支持。我們的key可以適當(dāng)進行有規(guī)律的封裝,比如以user為主的網(wǎng)站來 說,每個用戶都有User ID,那么可以按照固定的ID來進行提取和存取,比如1開頭的用戶保存在第一臺Memcache服務(wù)器上,以2開頭的用戶的數(shù)據(jù)保存在第二胎 Mecache服務(wù)器上,存取數(shù)據(jù)都先按照User ID來進行相應(yīng)的轉(zhuǎn)換和存取。
但是這個有缺點,就是需要對User ID進行判斷,如果業(yè)務(wù)不一致,或者其他類型的應(yīng)用,可能不是那么合適,那么可以根據(jù)自己的實際業(yè)務(wù)來進行考慮,或者去想更合適的方法。
減少數(shù)據(jù)庫壓力
這個算是比較重要的,所有的數(shù)據(jù)基本上都是保存在數(shù)據(jù)庫當(dāng)中的,每次頻繁的存取數(shù)據(jù)庫,導(dǎo)致數(shù)據(jù)庫性能極具下降,無法同時服務(wù)更多的用戶, 比如MySQL,特別頻繁的鎖表,那么讓Memcache來分擔(dān)數(shù)據(jù)庫的壓力吧。我們需要一種改動比較小,并且能夠不會大規(guī)模改變前端的方式來進行改變目 前的架構(gòu)。
我考慮的一種簡單方法:
后端的數(shù)據(jù)庫操作模塊,把所有的Select操作提取出來(update/delete/insert不管),然后把對應(yīng)的SQL進行相應(yīng) 的hash算法計算得出一個hash數(shù)據(jù)key(比如MD5或者SHA),然后把這個key去Memcache中查找數(shù)據(jù),如果這個數(shù)據(jù)不存在,說明還沒 寫入到緩存中,那么從數(shù)據(jù)庫把數(shù)據(jù)提取出來,一個是數(shù)組類格式,然后把數(shù)據(jù)在set到Memcache中,key就是這個SQL的hash值,然后相應(yīng)的 設(shè)置一個失效時間,比如一個小時,那么一個小時中的數(shù)據(jù)都是從緩存中提取的,有效減少數(shù)據(jù)庫的壓力。缺點是數(shù)據(jù)不實時,當(dāng)數(shù)據(jù)做了修改以后,無法實時到前 端顯示,并且還有可能對內(nèi)存占用比較大,畢竟每次select出來的數(shù)據(jù)數(shù)量可能比較巨大,這個是需要考慮的因素。
Memcache的安全
我們上面的Memcache服務(wù)器端都是直接通過客戶端連接后直接操作,沒有任何的驗證過程,這樣如果服務(wù)器是直接暴露在互聯(lián)網(wǎng)上的話是比 較危險,輕則數(shù)據(jù)泄露被其他無關(guān)人員查看,重則服務(wù)器被入侵,因為Mecache是以root權(quán)限運行的,況且里面可能存在一些我們未知的bug或者是緩 沖區(qū)溢出的情況,這些都是我們未知的,所以危險性是可以預(yù)見的。為了安全起見,我做兩點建議,能夠稍微的防止黑客的入侵或者數(shù)據(jù)的泄露。
內(nèi)網(wǎng)訪問
最好把兩臺服務(wù)器之間的訪問是內(nèi)網(wǎng)形態(tài)的,一般是Web服務(wù)器跟Memcache服務(wù)器之間。普遍的服務(wù)器都是有兩塊網(wǎng)卡,一塊指向互聯(lián) 網(wǎng),一塊指向內(nèi)網(wǎng),那么就讓W(xué)eb服務(wù)器通過內(nèi)網(wǎng)的網(wǎng)卡來訪問Memcache服務(wù)器,我們Memcache的服務(wù)器上啟動的時候就監(jiān)聽內(nèi)網(wǎng)的IP地址和 端口,內(nèi)網(wǎng)間的訪問能夠有效阻止其他非法的訪問。
# memcached -d -m 1024 -u root -l 192.168.0.200 -p 11211 -c 1024 -P /tmp/memcached.pid
Memcache服務(wù)器端設(shè)置監(jiān)聽通過內(nèi)網(wǎng)的192.168.0.200的ip的11211端口,占用1024MB內(nèi)存,并且允許最大1024個并發(fā)連接。
設(shè)置防火墻
防火墻是簡單有效的方式,如果卻是兩臺服務(wù)器都是掛在網(wǎng)的,并且需要通過外網(wǎng)IP來訪問Memcache的話,那么可以考慮使用防火墻或者 代理程序來過濾非法訪問。一般我們在Linux下可以使用iptables或者FreeBSD下的ipfw來指定一些規(guī)則防止一些非法的訪問,比如我們可 以設(shè)置只允許我們的Web服務(wù)器來訪問我們Memcache服務(wù)器,同時阻止其他的訪問。
| 1 | # iptables -F |
| 2 | # iptables -P INPUT DROP |
| 3 | # iptables -A INPUT -p tcp -s 192.168.0.2 –dport 11211 -j ACCEPT |
| 4 | # iptables -A INPUT -p udp -s 192.168.0.2 –dport 11211 -j ACCEPT |
上面的iptables規(guī)則就是只允許192.168.0.2這臺Web服務(wù)器對Memcache服務(wù)器的訪問,能夠有效的阻止一些非法訪問,相應(yīng)的也可以增加一些其他的規(guī)則來加強安全性,這個可以根據(jù)自己的需要來做。
?
?
?
二、Memcache和memcached的區(qū)別
為什么會有Memcache和memcached兩種名稱?其實Memcache是這個項目的名稱,而memcached是它服務(wù)器端的主程序文件名,知道我的意思了把~~~~。一個是項目名稱,一個是主程序文件名,在網(wǎng)上看到了很多人不明白,于是混用了。
三、Memcache的服務(wù)器端和客戶端安裝
分為兩個過程:memcache服務(wù)器端的安裝和memcached客戶端的安裝。
所謂服務(wù)器端的安裝就是在服務(wù)器(一般都是linux系統(tǒng))上安裝Memcache實現(xiàn)數(shù)據(jù)的存儲。
所謂客戶端的安裝就是指php(或者其他程序,Memcache還有其他不錯的api接口提供)去使用服務(wù)器端的Memcache提供的函數(shù),需要php添加擴展。
四、PHP的Memcache客戶端所有方法總結(jié)
memcache函數(shù)所有的方法列表如下:
Memcache::add – 添加一個值,如果已經(jīng)存在,則返回false
Memcache::addServer – 添加一個可供使用的服務(wù)器地址
Memcache::close – 關(guān)閉一個Memcache對象
Memcache::connect – 創(chuàng)建一個Memcache對象
memcache_debug – 控制調(diào)試功能
Memcache::decrement – 對保存的某個key中的值進行減法操作
Memcache::delete – 刪除一個key值
Memcache::flush – 清除所有緩存的數(shù)據(jù)
Memcache::get – 獲取一個key值
Memcache::getExtendedStats – 獲取進程池中所有進程的運行系統(tǒng)統(tǒng)計
Memcache::getServerStatus – 獲取運行服務(wù)器的參數(shù)
Memcache::getStats – 返回服務(wù)器的一些運行統(tǒng)計信息
Memcache::getVersion – 返回運行的Memcache的版本信息
Memcache::increment – 對保存的某個key中的值進行加法操作
Memcache::pconnect – 創(chuàng)建一個Memcache的持久連接對象
Memcache::replace -對一個已有的key進行覆寫操作
Memcache::set – 添加一個值,如果已經(jīng)存在,則覆寫
Memcache::setCompressThreshold – 對大于某一大小的數(shù)據(jù)進行壓縮
Memcache::setServerParams – 在運行時修改服務(wù)器的參數(shù)
五、PHP的Memcache操作方法分解
Memcache::add用法
說明:
如果$key不存在的時候,使用這個函數(shù)來存儲$var的值。功能相同的函數(shù)是memcache_add()。
參數(shù):
$key :將要存儲的鍵值。
$var :存儲的值,字符型和整型會按原值保存,其他類型自動序列化以后保存。
$flag:是否用MEMCACHE_COMPRESSED來壓縮存儲的值,true表示壓縮,false表示不壓縮。
$expire:存儲值的過期時間,如果為0表示不會過期,你可以用unix時間戳或者描述來表示從現(xiàn)在開始的時間,但是你在使用秒數(shù)表示的時候,不要超過2592000秒 (表示30天)。
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE。如果$key值已經(jīng)存在,則會返回FALSE。 其他情況下Memcache::add()的用法類似于Memcache::set()。
例子:
<?php
$memcache_obj = memcache_connect(”localhost”, 11211);
memcache_add($memcache_obj, 'var_key', 'test variable', false, 30);
$memcache_obj->add('var_key', 'test variable', false, 30);
?>
Memcache::addServer用法
說明:
添加一個可供使用的服務(wù)器地址到連接池中,連接用Memcache::addServer打開,腳本執(zhí)行完后自動關(guān)閉,或者可以用Memcache::close()手動關(guān)閉。相同函數(shù)是memcache_add_server()。
當(dāng)用這個方法的時候(相對于Memcache::connect()和 Memcache::pconnect()方法),網(wǎng)絡(luò)連接只有等需要的時候才會建立,因此不會因為增加很多的服務(wù)器到連接池而增加系統(tǒng)負(fù)擔(dān),因為很多服務(wù)器可能沒有使用。
故障恢復(fù)會發(fā)生在這個方法執(zhí)行的任何階段,只要其他的服務(wù)器是正常的,這些連接請求的失敗用戶不會注意到。任何一種socket或者memcached服務(wù)器級的錯誤可以觸發(fā)故障恢復(fù)。正常的客戶端錯誤比如增加一個存在的鍵值不會引發(fā)故障恢復(fù)。
參數(shù):
$host服務(wù)器的地址
$port服務(wù)器端口
$persistent是否是一個持久連接
$weight這臺服務(wù)器在所有服務(wù)器中所占的權(quán)重
$timeout連接的持續(xù)時間
$retry_interval連接重試的間隔時間,默認(rèn)為15,設(shè)置為-1表示不進行重試
$status控制服務(wù)器的在線狀態(tài)
$failure_callback允許設(shè)置一個回掉函數(shù)來處理錯誤信息。
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE。
例子:
$memcache = new Memcache;
$memcache->addServer('memcache_host', 11211);
$memcache->addServer('memcache_host2′, 11211);
$memcache_obj = memcache_connect('memcache_host', 11211);
memcache_add_server($memcache_obj, 'memcache_host2′, 11211);
?>
Memcache::close用法
bool Memcache::close ( void )
說明:
關(guān)閉memcache服務(wù)器連接。這個函數(shù)不會關(guān)閉長連接,長連接只有在web服務(wù)器關(guān)閉或者重啟的時候才會關(guān)閉。相同的函數(shù)memcache_close()
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE。
例子:
<?php
$memcache_obj = memcache_connect('memcache_host', 11211);
memcache_close($memcache_obj);
$memcache_obj = new Memcache;
$memcache_obj->connect('memcache_host', 11211);
$memcache_obj->close();
?>
Memcache::connect用法
說明:
打 開memcached服務(wù)器連接,建立一個到memcached服務(wù)器的連接,用Memcache::connect打開的連接會在腳本執(zhí)行完畢后自動關(guān) 閉。你也可以用Memcache::close()去關(guān)閉連接。相同的函數(shù)是memcache_connect()。
參數(shù):
$host:指向memcached正在收聽的鏈接的主機,這個參數(shù)會有另一種特殊的連接方式unix:///path/to/memcached.sock,即用unix的域名sockets,這種情況下,端口必須設(shè)置為0
$port:指向memcached正在收聽的鏈接的端口,用unix的域名sockets的情況下,端口必須設(shè)置為0
$timeout:用于連接守護進程的秒數(shù),當(dāng)你改變默認(rèn)的1秒的值的時候,你需要考慮一下,如果你的連接太慢的話,你可能會失去緩存的優(yōu)勢。
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE。
例子:
<?php
$memcache_obj = memcache_connect('memcache_host', 11211);
$memcache = new Memcache;
$memcache->connect('memcache_host', 11211);
?>
memcache::debug
說明:
控制調(diào)試功能,前提是php在編譯的時候使用了-enable-debug選項,否則這個函數(shù)不會有作用。
參數(shù):
$on_off:true表示開啟調(diào)試,false表示關(guān)閉調(diào)試
返回值:
如果php在編譯的時候使用了-enable-debug選項,返回true,否則返回false
Memcache::decrement用法
說明:
Memcache::decremen方法的作用是對保存的某個key中的值進行減法操作,用法跟Memcache::increment類似。
你也可以用memcache_decrement()函數(shù)。
參數(shù):
Key:想要減少的鍵的名字
Value:想要減少的值。
返回值:
如果成功,返回被減少后的值,如果失敗返回false。
例子:
<?php
$memcache = new Memcache;
$memcache->connect('localhost', 11211);
$memcache->set('test_item', 8);
$memcache->increment('test_item', 4);
echo $memcache->decrement('test_item', 7);
// 顯示 5
?>
這個例子連Memcache::increment函數(shù)都一塊演示了。
Memcache::delete用法
說明:
刪除一個key值,如果參數(shù)$timeout被設(shè)置,那么存儲的值會在設(shè)置的秒數(shù)以后過期,你也可以用函數(shù)memcache_delete()
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE。
例子:
<?php
$memcache_obj = memcache_connect('memcache_host', 11211);
memcache_delete($memcache_obj, 'key_to_delete', 10);
$memcache_obj = new Memcache;
$memcache_obj->connect('memcache_host', 11211);
$memcache_obj->delete('key_to_delete', 10);
?>
Memcache::flush
說明:
清除所有緩存的數(shù)據(jù)。Memcache::flush實際上沒有釋放資源,它僅僅將所有的緩存標(biāo)記為過期,這樣可以使新的緩存來覆蓋被占的內(nèi)存空間。一樣的函數(shù)是memcache_flush()
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE。
例子:
<?php
$memcache_obj = memcache_connect('memcache_host', 11211);
memcache_flush($memcache_obj);
$memcache_obj = new Memcache;
$memcache_obj->connect('memcache_host', 11211);
$memcache_obj->flush();
?>
Memcache::get
string Memcache::get ( string $key [, int &$flags ] )
array Memcache::get ( array $keys [, array &$flags ] )
說明:
方法的作用是獲取一個key值,key值可以是一個數(shù)組,結(jié)果會包含鍵值對。
參數(shù):
$key是鍵值或者一個鍵的數(shù)組值。
$flags如果這個參數(shù)存在,那么$flags跟寫入這個參數(shù)的值相關(guān),這些$flags 類似于Memcache::set()函數(shù)里的$flags。
返回值:
如果成功,則返回key對應(yīng)的值,如果失敗則返回false.
例子:
<?php
$memcache_obj = memcache_connect('memcache_host', 11211);
$var = memcache_get($memcache_obj, 'some_key');
$memcache_obj = new Memcache;
$memcache_obj->connect('memcache_host', 11211);
$var = $memcache_obj->get('some_key');
$memcache_obj = memcache_connect('memcache_host', 11211);
$var = memcache_get($memcache_obj, Array('some_key', 'another_key'));
$memcache_obj = new Memcache;
$memcache_obj->connect('memcache_host', 11211);
$var = $memcache_obj->get(Array('some_key', 'second_key'));
?>
Memcache::getExtendedStats
說明:
獲取進程池中所有進程的運行系統(tǒng)統(tǒng)計。相同函數(shù)是memcache_get_extended_stats()
參數(shù):
$type表示要求返回的類型:reset, malloc, maps, cachedump, slabs, items, sizes;
$slabid第一個參數(shù)設(shè)置為”cachedump”時使用的。
$limit第一個參數(shù)設(shè)置為”cachedump”時使用的。
返回值:
如果成功,返回統(tǒng)計信息,失敗會返回false
例子:
<?php
$memcache_obj = new Memcache;
$memcache_obj->addServer('memcache_host', 11211);
$memcache_obj->addServer('failed_host', 11211);
$stats = $memcache_obj->getExtendedStats();
//slabs機制分配管理內(nèi)存的情況
$statsslab = $memcache_obj->getExtendedStats(slabs);
?>
Memcache::getServerStatus
說明:
獲取運行服務(wù)器的參數(shù)。返回一個服務(wù)器在線或者離線的狀態(tài)。相同的函數(shù)是memcache_get_server_status()
參數(shù):
$host:正在收聽的連接的主機
$port正在收聽的連接的主機的端口,默認(rèn)是11211
返回值:
成功返回服務(wù)器狀態(tài),服務(wù)器沒有啟動會返回0,其他數(shù)字的時候表示服務(wù)器是啟動狀態(tài)的。
例子:
<?php
$memcache = new Memcache;
$memcache->addServer('memcache_host', 11211);
echo $memcache->getServerStatus('memcache_host', 11211);
$memcache = memcache_connect('memcache_host', 11211);
echo memcache_get_server_status($memcache, 'memcache_host', 11211);
?>
Memcache::getStats
說明:
返回服務(wù)器的一些運行統(tǒng)計信息。相同的函數(shù)是memcache_get_stats()
參數(shù):
$type表示要求返回的類型:reset, malloc, maps, cachedump, slabs, items, sizes;
$slabid第一個參數(shù)設(shè)置為”cachedump”時使用的。
$limit第一個參數(shù)設(shè)置為”cachedump”時使用的。
Memcache::getVersion
說明:
返回運行的Memcache的版本信息。相同函數(shù)memcache_get_version()
返回值:
成功返回服務(wù)器的版本信息,失敗的時候返回false。
例子:
<?php
$memcache = new Memcache;
$memcache->connect('memcache_host', 11211);
echo $memcache->getVersion();
$memcache = memcache_connect('memcache_host', 11211);
echo memcache_get_version($memcache);
?>
Memcache::increment
對保存的某個key中的值進行加法操作
用法參考Memcache::decrement
Memcache::pconnect
說明:
創(chuàng)建一個Memcache的持久連接對象
用法與Memcache::connect()相似,不同點地方是Memcache::pconnect是建立的持久連接。這個連接在腳本執(zhí)行完或者Memcache::close()函數(shù)運行也不會被關(guān)閉。與它相同的函數(shù)是memcache_pconnect()
參數(shù):
$host:指向memcached正在收聽的鏈接的主機,這個參數(shù)會有另一種特殊的連接方式unix:///path/to/memcached.sock,即用unix的域名sockets,這種情況下,端口必須設(shè)置為0
$port:指向memcached正在收聽的鏈接的端口,用unix的域名sockets的情況下,端口必須設(shè)置為0
$timeout:用于連接守護進程的秒數(shù),當(dāng)你改變默認(rèn)的1秒的值的時候,你需要考慮一下,如果你的連接太慢的話,你可能會失去緩存的優(yōu)勢。
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE
<?php
$memcache_obj = memcache_pconnect('memcache_host', 11211);
$memcache_obj = new Memcache;
$memcache_obj->pconnect('memcache_host', 11211);
?>
Memcache::replace
說明:
對一個已有的key進行覆寫操作。相同函數(shù)是memcache_replace()
參數(shù):
$key :將要存儲的鍵值。
$var :存儲的值,字符型和整型會按原值保存,其他類型自動序列化以后保存。
$flag:是否用MEMCACHE_COMPRESSED來壓縮存儲的值,true表示壓縮,false表示不壓縮。
$expire:存儲值的過期時間,如果為0表示不會過期,你可以用unix時間戳或者描述來表示從現(xiàn)在開始的時間,但是你在使用秒數(shù)表示的時候,不要超過2592000秒 (表示30天)。
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE。如果$key值已經(jīng)存在,則會返回FALSE。
<?php
$memcache_obj = memcache_connect('memcache_host', 11211);
memcache_replace($memcache_obj, "test_key", "some variable", false, 30);
$memcache_obj->replace("test_key", "some variable", false, 30);
?>
Memcache::set
說明:
添加一個值,如果已經(jīng)存在,則覆寫。相同函數(shù)是memcache_set()
參數(shù):
$key :將要存儲的鍵值。
$var :存儲的值,字符型和整型會按原值保存,其他類型自動序列化以后保存。
$flag:是否用MEMCACHE_COMPRESSED來壓縮存儲的值,true表示壓縮,false表示不壓縮。
$expire:存儲值的過期時間,如果為0表示不會過期,你可以用unix時間戳或者描述來表示從現(xiàn)在開始的時間,但是你在使用秒數(shù)表示的時候,不要超過2592000秒 (表示30天)。
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE。
例子:
$memcache_obj = new Memcache;
$memcache_obj->connect('memcache_host', 11211);
$memcache_obj->set('var_key', 'some really big variable', MEMCACHE_COMPRESSED, 50);
echo $memcache_obj->get('var_key');
Memcache::setCompressThreshold
說明:
對大于某一大小的數(shù)據(jù)進行壓縮。相同的函數(shù)是memcache_set_compress_threshold()
參數(shù):
setCompressThreshold方法有兩個參數(shù),第一個參數(shù)表示處理數(shù)據(jù)大小的臨界點,第二個參數(shù)表示壓縮的比例,默認(rèn)為0.2。
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE。
例子:
<?php
$memcache_obj = new Memcache;
$memcache_obj->addServer('memcache_host', 11211);
$memcache_obj->setCompressThreshold(20000, 0.2);
$memcache_obj = memcache_connect('memcache_host', 11211);
memcache_set_compress_threshold($memcache_obj, 20000, 0.2);
?>
Memcache::setServerParams
說明:
在運行時修改服務(wù)器的參數(shù)。相同函數(shù)是memcache_set_server_params()。
參數(shù):
$host服務(wù)器的地址
$port服務(wù)器端口
$timeout連接的持續(xù)時間
$retry_interval連接重試的間隔時間,默認(rèn)為15,設(shè)置為-1表示不進行重試
$status控制服務(wù)器的在線狀態(tài)
$failure_callback允許設(shè)置一個回掉函數(shù)來處理錯誤信息。
返回值:
如果成功則返回 TRUE,失敗則返回 FALSE。
例子:
<?php
function _callback_memcache_failure($host, $port) {
print "memcache '$host:$port' failed";
}
$memcache = new Memcache;
// 離線模式增加一個服務(wù)器
$memcache->addServer('memcache_host', 11211, false, 1, 1, -1, false);
// 把服務(wù)器設(shè)成在線
$memcache->setServerParams('memcache_host', 11211, 1, 15, true, '_callback_memcache_failure');
$memcache_obj = memcache_connect('memcache_host', 11211);
memcache_set_server_params($memcache_obj, 'memcache_host', 11211, 1, 15, true, '_callback_memcache_failure');
?>
六、綜合使用實例
<?php
//連接
$mem = new Memcache;
$mem->connect("db.nowamagic.net", 12000);
//保存數(shù)據(jù)
$mem->set('key1', 'This is first value', 0, 60);
$val = $mem->get('key1');
echo "Get key1 value: " . $val ."<br />";
//替換數(shù)據(jù)
$mem->replace('key1', 'This is replace value', 0, 60);
$val = $mem->get('key1');
echo "Get key1 value: " . $val . "<br />";
//保存數(shù)組
$arr = array('aaa', 'bbb', 'ccc', 'ddd');
$mem->set('key2', $arr, 0, 60);
$val2 = $mem->get('key2');
echo "Get key2 value: ";
print_r($val2);
echo "<br />";
//刪除數(shù)據(jù)
$mem->delete('key1');
$val = $mem->get('key1');
echo "Get key1 value: " . $val . "<br />";
//清除所有數(shù)據(jù)
$mem->flush();
$val2 = $mem->get('key2');
echo "Get key2 value: ";
print_r($val2);
echo "<br />";
//關(guān)閉連接
$mem->close();
?>
如果正常的話,瀏覽器將輸出:
Get key1 value: This is first value
Get key1 value: This is replace value
Get key2 value: Array ( [0] => aaa [1] => bbb [2] => ccc [3] => ddd )
Get key1 value:
Get key2 value:
七、實例程序代碼分析
初始化一個Memcache的對象:$mem = new Memcache;
連接到我們的Memcache服務(wù)器端,第一個參數(shù)是服務(wù)器的IP地址,也可以是主機名,第二個參數(shù)是Memcache的開放的端口:$mem->connect("192.168.0.200", 12000);
保 存一個數(shù)據(jù)到Memcache服務(wù)器上,第一個參數(shù)是數(shù)據(jù)的key,用來定位一個數(shù)據(jù),第二個參數(shù)是需要保存的數(shù)據(jù)內(nèi)容,這里是一個字符串,第三個參數(shù)是 一個標(biāo)記,一般設(shè)置為0或者MEMCACHE_COMPRESSED就行了,第四個參數(shù)是數(shù)據(jù)的有效期,就是說數(shù)據(jù)在這個時間內(nèi)是有效的,如果過去這個時 間,那么會被Memcache服務(wù)器端清除掉這個數(shù)據(jù),單位是秒,如果設(shè)置為0,則是永遠(yuǎn)有效,我們這里設(shè)置了60,就是一分鐘有效時 間:$mem->set(‘key1‘, ‘This is first value', 0, 60);
從Memcache服務(wù)器端獲取一條數(shù)據(jù),它只有一個參數(shù),就是需要獲取數(shù)據(jù)的key,我們這里是上一步設(shè)置的key1,現(xiàn)在獲取這個數(shù)據(jù)后輸出輸出:
$val = $mem->get('key1′);
echo "Get key1 value: " . $val;
現(xiàn)在是使用replace方法來替換掉上面key1的值,replace方法的參數(shù)跟set是一樣的,不過第一個參數(shù)key1是必須是要替換數(shù)據(jù)內(nèi)容的key,最后輸出了:
$mem->replace('key1', 'This is replace value', 0, 60);
$val = $mem->get('key1');
echo "Get key1 value: " . $val;
同樣的,Memcache也是可以保存數(shù)組的,下面是在Memcache上面保存了一個數(shù)組,然后獲取回來并輸出:
$arr = array('aaa', 'bbb', 'ccc', 'ddd');
$mem->set('key2', $arr, 0, 60);
$val2 = $mem->get('key2');
print_r($val2);
現(xiàn)在刪除一個數(shù)據(jù),使用delte接口,參數(shù)就是一個key,然后就能夠把Memcache服務(wù)器這個key的數(shù)據(jù)刪除,最后輸出的時候沒有結(jié)果:
$val = $mem->get('key1');
echo "Get key1 value: " . $val . "<br />";
最后我們把所有的保存在Memcache服務(wù)器上的數(shù)據(jù)都清除,會發(fā)現(xiàn)數(shù)據(jù)都沒有了,最后輸出key2的數(shù)據(jù)為空,最后關(guān)閉連接:
$val2 = $mem->get('key2');
echo "Get key2 value: ";
print_r($val2);
echo "<br />";
八、什么時候使用Memcache和Memcache的使用環(huán)境
使 用Memcache的網(wǎng)站一般流量都是比較大的,為了緩解數(shù)據(jù)庫的壓力,讓Memcache作為一個緩存區(qū)域,把部分信息保存在內(nèi)存中,在前端能夠迅速的 進行存取。那么一般的焦點就是集中在如何分擔(dān)數(shù)據(jù)庫壓力和進行分布式,畢竟單臺Memcache的內(nèi)存容量的有限的。我這里簡單提出我的個人看法,未經(jīng)實 踐,權(quán)當(dāng)參考。
分布式應(yīng)用
Memcache本來支持分布式,我們客戶端稍加改 造,更好的支持。我們的key可以適當(dāng)進行有規(guī)律的封裝,比如以user為主的網(wǎng)站來說,每個用戶都有User ID,那么可以按照固定的ID來進行提取和存取,比如1開頭的用戶保存在第一臺Memcache服務(wù)器上,以2開頭的用戶的數(shù)據(jù)保存在第二胎 Mecache服務(wù)器上,存取數(shù)據(jù)都先按照User ID來進行相應(yīng)的轉(zhuǎn)換和存取。
但是這個有缺點,就是需要對User ID進行判斷,如果業(yè)務(wù)不一致,或者其他類型的應(yīng)用,可能不是那么合適,那么可以根據(jù)自己的實際業(yè)務(wù)來進行考慮,或者去想更合適的方法。
減少數(shù)據(jù)庫壓力
這 個算是比較重要的,所有的數(shù)據(jù)基本上都是保存在數(shù)據(jù)庫當(dāng)中的,每次頻繁的存取數(shù)據(jù)庫,導(dǎo)致數(shù)據(jù)庫性能極具下降,無法同時服務(wù)更多的用戶,比如MySQL, 特別頻繁的鎖表,那么讓Memcache來分擔(dān)數(shù)據(jù)庫的壓力吧。我們需要一種改動比較小,并且能夠不會大規(guī)模改變前端的方式來進行改變目前的架構(gòu)。
我考慮的一種簡單方法:
后 端的數(shù)據(jù)庫操作模塊,把所有的Select操作提取出來(update/delete/insert不管),然后把對應(yīng)的SQL進行相應(yīng)的hash算法計 算得出一個hash數(shù)據(jù)key(比如MD5或者SHA),然后把這個key去Memcache中查找數(shù)據(jù),如果這個數(shù)據(jù)不存在,說明還沒寫入到緩存中,那 么從數(shù)據(jù)庫把數(shù)據(jù)提取出來,一個是數(shù)組類格式,然后把數(shù)據(jù)在set到Memcache中,key就是這個SQL的hash值,然后相應(yīng)的設(shè)置一個失效時 間,比如一個小時,那么一個小時中的數(shù)據(jù)都是從緩存中提取的,有效減少數(shù)據(jù)庫的壓力。缺點是數(shù)據(jù)不實時,當(dāng)數(shù)據(jù)做了修改以后,無法實時到前端顯示,并且還 有可能對內(nèi)存占用比較大,畢竟每次select出來的數(shù)據(jù)數(shù)量可能比較巨大,這個是需要考慮的因素。
九、Memcache的安全
我 們上面的Memcache服務(wù)器端都是直接通過客戶端連接后直接操作,沒有任何的驗證過程,這樣如果服務(wù)器是直接暴露在互聯(lián)網(wǎng)上的話是比較危險,輕則數(shù)據(jù) 泄露被其他無關(guān)人員查看,重則服務(wù)器被入侵,因為Mecache是以root權(quán)限運行的,況且里面可能存在一些我們未知的bug或者是緩沖區(qū)溢出的情況, 這些都是我們未知的,所以危險性是可以預(yù)見的。為了安全起見,我做兩點建議,能夠稍微的防止黑客的入侵或者數(shù)據(jù)的泄露。
內(nèi)網(wǎng)訪問
最 好把兩臺服務(wù)器之間的訪問是內(nèi)網(wǎng)形態(tài)的,一般是Web服務(wù)器跟Memcache服務(wù)器之間。普遍的服務(wù)器都是有兩塊網(wǎng)卡,一塊指向互聯(lián)網(wǎng),一塊指向內(nèi)網(wǎng), 那么就讓W(xué)eb服務(wù)器通過內(nèi)網(wǎng)的網(wǎng)卡來訪問Memcache服務(wù)器,我們Memcache的服務(wù)器上啟動的時候就監(jiān)聽內(nèi)網(wǎng)的IP地址和端口,內(nèi)網(wǎng)間的訪問 能夠有效阻止其他非法的訪問。
Memcache服務(wù)器端設(shè)置監(jiān)聽通過內(nèi)網(wǎng)的192.168.0.200的ip的11211端口,占用1024MB內(nèi)存,并且允許最大1024個并發(fā)連接。
設(shè)置防火墻
防 火墻是簡單有效的方式,如果卻是兩臺服務(wù)器都是掛在網(wǎng)的,并且需要通過外網(wǎng)IP來訪問Memcache的話,那么可以考慮使用防火墻或者代理程序來過濾非 法訪問。一般我們在Linux下可以使用iptables或者FreeBSD下的ipfw來指定一些規(guī)則防止一些非法的訪問,比如我們可以設(shè)置只允許我們 的Web服務(wù)器來訪問我們Memcache服務(wù)器,同時阻止其他的訪問。
# iptables -P INPUT DROP
# iptables -A INPUT -p tcp -s 192.168.0.2 –dport 11211 -j ACCEPT
# iptables -A INPUT -p udp -s 192.168.0.2 –dport 11211 -j ACCEPT
轉(zhuǎn)載于:https://www.cnblogs.com/try-better-tomorrow/p/4936819.html
總結(jié)
以上是生活随笔為你收集整理的关于memecache的使用及清楚示意的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: [redis] 分布式 Redis 的
- 下一篇: PHP面试题之设计模式