前端學PHP之PHP操作memcache

来源:http://www.cnblogs.com/xiaohuochai/archive/2016/12/06/6137453.html
-Advertisement-
Play Games

[1]安裝 [2]連接 [3]增刪改查 [4]分散式 [5]狀態 [6]安全 [7]應用 ...


×
目錄
[1]安裝 [2]連接 [3]增刪改查[4]分散式[5]狀態[6]安全[7]應用

前面的話

  和訪問mysql伺服器類似,PHP也是作為客戶端API訪問memcached伺服器的,所以同樣需要為PHP程式安裝memcache的擴展介面,比較常用的有memcache和memcached兩種擴展。而memcached和memcache的守護進程memcached同名,比較容易引起混淆,甚至提到memcached,有些人第一想到的是後臺的守護進程,這裡還是有必要分析一下兩者之間的區別。memcache是完全在PHP框架內開發的,而memecached是使用libmemcached的。從手冊上看,memcached 會比 memcache 多幾個方法,使用方式上都差不多。memcache是原生實現的,但是使用libmemcached的memached只支持OO介面,而 memcache則是OO和非OO兩套介面並存,以後隨著memcached伺服器端的改進,這個lib也必定會馬上跟進的。而memcache卻不一定能做到按時跟進。memcached,還有個非常稱贊的地方,就是flag不是在操作的時候設置了。而是有一個統一的setOption()。memcached 實現了更多的 memcached 協議(畢竟是基於 libmemcached庫)。本文選擇較簡單的memcache擴展作介紹

 

安裝

  在window系統下安裝memcache擴展比較簡單,下載一個與PHP版本一致的memcache擴展庫即可

  將下載的php_memcache.dll文件保存到PHP的應用程式擴展ext目錄中

  在php.ini文件添加擴展的位置,加入一行"extension=php_memcache.dll"

  重新啟動apache伺服器即可,通過phpInfo()可以找到memcache服務已經安裝

 

連接

  PHP的Memcache應用程式介面既然是作為memcached伺服器的客戶端,就需要先連接到memcached伺服器。Memcache::connect()方法就用於連接到一個memcached伺服器,如果連接成功則返回true,否則返回false

bool Memcache::connect ( string $host [, int $port [, int $timeout ]] )

  該方法有三個參數。第一個參數host(必須項)表示memcached服務端監聽主機地址;第二個參數port表示memcached服務端監聽埠(可選項),預設值為11211;第三個參數timeout表示連接持續(超時)時間,單位秒。預設值1秒

  [註意]過長的連接持續時間可能會導致失去所有的緩存優勢

  使用Memcache::connect()連接到memcached伺服器,並完成操作後,可以使用Memcache::close()方法關閉連接,完成一些會話過程。如果需要以長連接方式連接memcached伺服器,可以使用Memcache::pconnect()方法實現,該方法的調用方法和Memcache::connect()完全相同,但長連接不能被Memcache::close()方法關閉

//實例化對象
$memcache = new Memcache;
//連接memcache伺服器
$memcache->connect('localhost');
/*其他操作*/
//關閉連接
$memcache->close();

 

增刪改查

增加

  連接memcached伺服器成功後,就可以添加一個要緩存的數據(add),通過Memcache::add()來完成

bool Memcache::add ( string $key , mixed $var [, int $flag [, int $expire ]] )

  在add()方法中,參數key表示將要分配給變數的key;參數var表示將要被存儲的變數。字元串和整型被以原文存儲,其他類型序列化後存儲;參數flag(可選項)表示使用MEMCACHE_COMPRESSED標記對數據進行壓縮(使用zlib);參數expire(可選項)表示當前寫入緩存的數據的失效時間。如果此值設置為0表明此數據永不過期。你可以設置一個UNIX時間戳或 以秒為單位的整數(從當前算起的時間差)來說明此數據的過期時間,但是在後一種設置方式中,不能超過2592000秒(30天)

$memcache->add('id1','11111');

刪除

Memcache::delete()

  Memcache::delete — 從服務端刪除一個元素

bool Memcache::delete ( string $key [, int $timeout = 0 ] )

  在delete方法中,有兩個參數。參數key表示要刪除的元素的key;參數timeout表示刪除該元素的執行時間。如果值為0,則該元素立即刪除,如果值為30,元素會在30秒內被刪除

$memcache->delete('id1');

Memcache::flush

  Memcache::flush — 清洗(刪除)已經存儲的所有的元素。成功時返回 TRUE, 或者在失敗時返回 FALSE

bool Memcache::flush ( void )

  [註意]Memcache::flush()立即使所有已經存在的元素失效。方法Memcache::flush()並不會真正的釋放任何資源,而是僅僅標記所有元素都失效了,因此已經被使用的記憶體會被新的元素覆寫

修改

Memcache::set

  Memcache::set — 設置一個指定key的緩存內容

bool Memcache::set ( string $key , mixed $var [, int $flag [, int $expire ]] )

  [註意]set()有一個別名是replace()

$memcache->set('id1','12345');

查看

Memcache::get

  Memcache::get — 從服務端檢回一個元素,返回key對應的存儲元素的字元串值或者在失敗或key未找到的時候返回FALSE

  get()有以下兩種用法

string Memcache::get ( string $key [, int &$flags ] )
array Memcache::get ( array $keys [, array &$flags ] )

  如果服務端之前有以key作為key存儲的元素,Memcache::get()方法此時返回之前存儲的值;也可以給Memcache::get()方法傳遞一個數組(多個key)來獲取一個數組的元素值,返回的數組僅僅包含從服務端查找到的key-value對

  get方法包括兩個參數。第一個參數key表示要獲取值的key或key數組;第二個參數為flags,如果給定這個參數(以引用方式傳遞),該參數會被寫入一些key對應的信息。這些標記和Memcache::set()方法中的同名參數 意義相同。用int值的低位保留了pecl/memcache的內部用法(比如:用來說明壓縮和序列化狀態)

$memcache->set('id1','11111');
$memcache->add('id2','22222');
echo $memcache->get('id1') ."<br>";//11111
echo $memcache->get(['id1','id2'])['id2'];//22222

 

分散式

  如果有多台memcached伺服器端,最好使用Memcache::addServer()來連接伺服器,而不是使用Memcache::connect()去連接memcached伺服器,因為PHP客戶端是利用伺服器池,根據"crc32(key) % current_server_num"哈希演算法將key哈希到不同的伺服器中。Memcache::addServer()方法的格式如下所示

bool Memcache::addServer ( string $host [, int $port = 11211 [, bool $persistent [, int $weight [, int $timeout [, int $retry_interval [, bool $status [, callback $failure_callback [, int $timeoutms ]]]]]]]] )

  Memcache::addServer()增加一個伺服器到連接池中。通過Memcache::addServer()打開的連接將會在腳本執行結束後自動關閉,也可以使用Memcache::close()進行手動關閉,也可以使用memcache_add_server()來添加伺服器

  當使用這個方法時(與Memcache::connect()和Memcache::pconnect()相反),網路連接並不會立刻建立,而是直到真正使用的時候才建立。因此在加入大量伺服器到連接池中時也是沒有開銷的,因為它們可能並不會被使用

  故障轉移可能在方法的任何一個層次發生,通常只要其他伺服器可用,用戶就不會感受到。任何的socket或memcache伺服器級別的錯誤(比如記憶體溢出)都可能導致故障轉移。而一般的客戶端錯誤比如使用Memcache::add嘗試增加一個已經存在的key則不會導致故障轉移

  參數host表示要連接的memcached服務端監聽的主機位置。這個參數通常指定其他類型的傳輸比如Unix域套接字使用 unix:///path/to/memcached.sock,這種情況下參數port 必須設置為0

  參數port表示要連接的memcached服務端監聽的埠。當使用UNIX域套接字連接時設置為0

  參數persistent用來控制是否使用持久化連接。預設TRUE

  參數weight表示為此伺服器創建的桶的數量,用來控制此伺服器被選中的權重,單個伺服器被選中的概率是相對於所有伺服器weight總和而言的

  參數timeout表示連接持續(超時)時間(單位秒),預設值1秒

  參數retry_interval表示伺服器連接失敗時重試的間隔時間,預設值15秒。如果此參數設置為-1表示不重試。此參數和persistent參數在擴展以dl()函數動態載入的時候無效

  每個失敗的連接結構有自己的超時時間,並且在它失效之前選擇後端服務請求時該結構會被跳過。一旦一個連接失效,它將會被成功重新連接或被標記為失敗連接以在下一個retry_interval秒重連。典型的影響是每個web服務子進程在服務於一個頁面時將會每retry_interval秒嘗試重新連接一次

  參數status用來控制此伺服器是否可以被標記為線上狀態。設置此參數值為FALSE並且retry_interval參數設置為-1時,允許將失敗的伺服器保留在一個池中以免影響key的分配演算法。對於這個伺服器的請求會進行故障轉移或者立即失敗,這受限於memcache.allow_failover參數的設置。該參數預設TRUE,表明允許進行故障轉移

  參數failure_callback表示允許用戶指定一個運行時發生錯誤後的回調函數。回調函數會在故障轉移之前運行。回調函數會接受到兩個參數,分別是失敗主機的主機名和埠號

<?php
//實例化對象
$memcache = new Memcache;
//連接memcache伺服器
$memcache->addServer('localhost');
$memcache->addServer('192.168.1.2');
$memcache->addServer('192.168.1.3');

for($i=0;$i<100;$i++){
    $memcache->set('key'.$i,md5($i),0,60);
}
//關閉連接
$memcache->close();
?>

 

狀態

Memcache::getStats

  Memcache::getStats — 獲取伺服器統計信息

array Memcache::getStats ([ string $type [, int $slabid [, int $limit = 100 ]]] )

  Memcache::getStats()返回一個關聯數據的伺服器統計信息。數組key是統計信息名, 值就是統計信息的值。同樣可以使用函數memcache_get_stats()

  參數type表示期望抓取的統計信息類型,可以使用的值有{reset, malloc, maps, cachedump, slabs, items, sizes}。通過memcached協議指定這些附加參數是為了方便memcache開發者(檢查其中的變動)

  參數slabid用於與參數type聯合從指定slab分塊拷貝數據,cachedump命令會完全占用伺服器通常用於比較嚴格的調試。

  參數limit用於和參數type聯合來設置cachedump時從服務端獲取的實體條數

<?php
//實例化對象
$memcache = new Memcache;
//連接memcache伺服器
$memcache->addServer('localhost');
/*
Array ( [pid] => 6268 [uptime] => 3054571472 [time] => 240596173 [version] => 1.4.4-14-g9c660c0 [pointer_size] => 64 [curr_connections] => 13 [total_connections] => 24 [connection_structures] => 14 [cmd_get] => 37 [cmd_set] => 526 [cmd_flush] => 1 [get_hits] => 31 [get_misses] => 6 [delete_misses] => 0 [delete_hits] => 2 [incr_misses] => 0 [incr_hits] => 0 [decr_misses] => 0 [decr_hits] => 0 [cas_misses] => 0 [cas_hits] => 0 [cas_badval] => 0 [auth_cmds] => 0 [auth_errors] => 0 [bytes_read] => 27834 [bytes_written] => 9658 [limit_maxbytes] => 67108864 [accepting_conns] => 1 [listen_disabled_num] => 0 [threads] => 4 [conn_yields] => 0 [bytes] => 10655 [curr_items] => 105 [total_items] => 518 [evictions] => 0 )
 */
print_r($memcache->getStats()); 
//關閉連接
$memcache->close();
?>

Memcache::getExtendedStats

  Memcache::getExtendedStats — 緩存伺服器池中所有伺服器統計信息

array Memcache::getExtendedStats ([ string $type [, int $slabid [, int $limit = 100 ]]] )

  Memcache::getExtendedStats() 返回一個二維關聯數據的伺服器統計信息。數組的key由host:port方式組成,無效的伺服器返回的統計信息被設置為false,同樣的,可以使用函數memcache_get_extended_stats()

  參數type表示期望抓取的統計信息類型,可以使用的值有{reset, malloc, maps, cachedump, slabs, items, sizes}。 通過memcached協議指定這些附加參數是為了方便memcache開發者(檢查其中的變動)

  參數slabid用於與參數type聯合從指定slab分塊拷貝數據,cachedump命令會完全占用伺服器通常用於 比較嚴格的調試

  參數limit用於和參數type聯合來設置cachedump時從服務端獲取的實體條數

<?php
    $memcache_obj = new Memcache;
    $memcache_obj->addServer('memcache_host', 11211);
    $memcache_obj->addServer('failed_host', 11211);
    
    $stats = $memcache_obj->getExtendedStats();
/*
Array
(
    [memcache_host:11211] => Array
        (
            [pid] => 3756
            [uptime] => 603011
            [time] => 1133810435
            [version] => 1.1.12
            [rusage_user] => 0.451931
            [rusage_system] => 0.634903
            [curr_items] => 2483
            [total_items] => 3079
            [bytes] => 2718136
            [curr_connections] => 2
            [total_connections] => 807
            [connection_structures] => 13
            [cmd_get] => 9748
            [cmd_set] => 3096
            [get_hits] => 5976
            [get_misses] => 3772
            [bytes_read] => 3448968
            [bytes_written] => 2318883
            [limit_maxbytes] => 33554432
        )

    [failed_host:11211] => false
)
 */    
    print_r($stats);
?>

 

安全

  訪問MYSQL資料庫伺服器時必須通過用戶驗證後才能進入,而訪問memcached伺服器則是通過客戶端連接後直接操作,沒有任何的驗證過程。這樣如果伺服器直接暴露在互聯網上的話是比較危險,輕則數據泄露被其他無關人員查看,重則伺服器被入侵。為了安全起見,有以下兩點安全措施

  1、內網訪問

  最好把兩台伺服器之間的訪問設置為內網形態的,一般在Web伺服器跟Memcache伺服器之間。普遍的伺服器都是有兩塊網卡,一塊指向互聯網,一塊指向內網,那麼就讓Web伺服器通過內網的網卡來訪問Memcache伺服器,Memcache伺服器啟動時,就監聽內網的IP地址和埠,內網間的訪問能夠有效阻止其他非法的訪問

# memcached -d -m 1024 -u root -l 192.168.0.200 -p 11211 -c 1024 start

  Memcache伺服器端設置監聽通過內網的192.168.0.200的ip的11211埠,占用1024MB記憶體,並且允許最大1024個併發連接

  2、設置防火牆

  防火牆是簡單有效的方式,如果memcache和web Server在同一臺機器上,或通過外網IP來訪問Memcache的話,就需要使用防火牆或者代理程式來過濾非法訪問

  一般在Linux下可以使用iptables或者FreeBSD下的ipfw來指定一些規則防止一些非法的訪問,比如可以設置只允許Web伺服器來訪問Memcache伺服器,同時阻止其他的訪問

# iptables -F
# 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

  上面的iptables規則是只允許192.168.0.2這台Web伺服器對Memcache伺服器的訪問,能夠有效的阻止一些非法訪問,相應的也可以增加一些其他的規則來加強安全性,這個可以根據自己的需要來做

 

應用

  在項目中最常見的memcache應用,是緩存從資料庫中查詢的數據結果,以及保存會話控制信息。會話控制將在以後介紹。下麵主要介紹如何將資料庫查詢出來的結果使用memcache伺服器進行緩存,以減少頻繁的資料庫連接及大量的查詢對資料庫造成的壓力。設計的原則是只要資料庫中的記錄沒有被改變,就不需要重新連接資料庫並反覆執行重覆的查詢語句,相同的查詢結果都應該從緩存伺服器中獲取

<?php 
function select($sql,MemCache $memcache){
    $key=md5($sql);
    $data = $memcache->get($key);
    if(!$data) {
        try {
            $pdo = new PDO("mysql:host=localhost;dbname=xsphp", "root", "123456");
        }catch(PDOException $e) {
            die("資料庫連接失敗:".$e->getMessage());
        }
        $stmt = $pdo -> prepare($sql);
        $stmt -> execute();
        $data = $stmt -> fetchAll(PDO::FETCH_ASSOC);
        $memcache -> add($key, $data, MEMCACHE_COMPRESSED, 0);
    }
    return $data;
}
$memcache = new Memcache;
$memcache -> connect("localhost", 11211);
$data = select("SELECT * FROM user",$memcache);
var_dump($data);
$memcache -> close();
?>

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • IO流技術 概念:input - output 輸入輸出流: 輸入:將文件讀到記憶體中; 輸出:將文件從記憶體中寫出到其他地方 作用:主要就是解決設備和設備之間的數據傳輸問題。 File :文件類的使用十分重要 (一)file的構造方法 (二)基本方法 (三)判斷方法 (四)獲取方法 (五)對文件的操作 ...
  • 需要的jar包:xmlpull_1_0_5.jar,xstream-1.4.1.jar) 1、工具類XstreamUtil package com.learn.util; import com.thoughtworks.xstream.XStream; import com.thoughtworks ...
  • pc的demo很多,不記。 移動端做支付的時候要先配置可測試功能變數名稱和授權功能變數名稱,一個在公眾平臺里的微信支付里配置 註意,獲取open的Id的方法需要寫到這個配置好的功能變數名稱下。否則會說功能變數名稱未授權。 還有一個是在微信支付里配置的授權功能變數名稱,在最下麵有個網頁授權,裡面配置你的測試功能變數名稱,然後需要下載一個文件,建議放 ...
  • 很長時間都沒有更新了,最近在補充JavaSE的一些細節部分 關於IO流的一些總結 首先要介紹的是File類,File類用於對文件和目錄的一些操作 1.創建文件CreateNewFile() 2.對文件的信息的獲取getName(),getAbsolutePath() 3.判斷是否是文件isFile( ...
  • 調用requonse.getWriter()方法時可實現文本字元串數據輸出,調用response.getOutputStream()方法可現實位元組流數據的輸出。兩種輸出方式threadlocal模式和osiv模式~~~!!!!threadlocal 是一個局部的線程變數 只是一個map 保存的是線程 ...
  • std::function是可調用對象的包裝器,它最重要的功能是實現延時調用: 由上邊代碼定義std::function<int(int)> fr2,那麼fr2就可以代表返回值和參數表相同的一類函數。可以看出fr2保存了指代的函數,可以在之後的程式過程中調用。這種用法在實際編程中是很常見的。 std ...
  • 1.先將map對象轉成set,然後再轉為迭代器 2.先將map轉為set類型的key值集合,然後轉為迭代器 ...
  • import java.io.BufferedReader;import java.io.BufferedWriter;import java.io.File;import java.io.FileReader;import java.io.FileWriter;import java.io.IOE ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...