memcached的安裝以及php下的使用_Web服務(wù)器教程

      編輯Tag賺U幣
      先把php版本從5.2.0升級(jí)到5.2.4(覆蓋原來(lái)的文件就行了)
        1. memcached-win32/">http://jehiah.cz/projects/memcached-win32/ 上下載memcache的windows穩(wěn)定版,解壓放某個(gè)盤下面,比如在c:\memcached  
        2. 在終端(也即cmd命令界面)下輸入 'c:\memcached\memcached.exe -d install' 安裝
        3. 再輸入: 'c:\memcached\memcached.exe -d start' 啟動(dòng)
            NOTE: 以后memcached將作為windows的一個(gè)服務(wù)每次開機(jī)時(shí)自動(dòng)啟動(dòng)。默認(rèn)端口:11211。
        4. 在C:\winnt\php.ini 加入一行 'extension=php_memcache.dll'
        5. 下載pecl的PECL 5.2.4 Win32 binaries模塊包,解壓縮后將其中的memcache.dll考到c:\php\ext 中,(也不一定是c盤的PHP文件夾下,主要是看你當(dāng)初把PHP安裝到哪個(gè)盤符下面了)
            NOTE: php和pecl的版本要一致。
        6.重新啟動(dòng)Apache,然后查看一下phpinfo,如果有memcache,那么就說(shuō)明安裝成功!


      emcache是danga.com的一個(gè)項(xiàng)目,最早是為 LiveJournal 服務(wù)的,目前全世界不少人使用這個(gè)緩存項(xiàng)目來(lái)構(gòu)建自己大負(fù)載的網(wǎng)站,來(lái)分擔(dān)數(shù)據(jù)庫(kù)的壓力。(關(guān)于Memcache的更多信息請(qǐng)Google)
      Memcache官方網(wǎng)站:http://www.danga.com/memcached
      【安裝Memcache服務(wù)器端】
      我目前的平臺(tái),服務(wù)器是Fedora Core 1(內(nèi)核:2.4.22),客戶端是Windows XP SP2,需要安裝的就是服務(wù)器的Memcached的守護(hù)進(jìn)程和客戶端的PHP擴(kuò)展php_memcache兩個(gè)東西。現(xiàn)在我分別來(lái)講。
      服務(wù)器端主要是安裝memcache服務(wù)器端,目前的最新版本是 memcached-1.2.0 。
      下載:http://www.danga.com/memcached/dist/memcached-1.2.0.tar.gz
      另外,Memcache用到了libevent這個(gè)庫(kù)用于Socket的處理,所以還需要安裝libevent,libevent的最新版本是libevent-1.2。(如果你的系統(tǒng)已經(jīng)安裝了libevent,可以不用安裝)
      官網(wǎng):http://www.monkey.org/~provos/libevent/
      下載:http://www.monkey.org/~provos/libevent-1.2.tar.gz
      我分別把兩個(gè)東東下載回來(lái),放到 /tmp 目錄下:
      # cd /tmp
      # wget
      http://www.danga.com/memcached/dist/memcached-1.2.0.tar.gz
      # wget http://www.monkey.org/~provos/libevent-1.2.tar.gz
      先安裝libevent:
      # tar zxvf libevent-1.2.tar.gz
      # cd libevent-1.2
      # ./configure --prefix=/usr
      # make
      # make install

      然后看看我們的libevent是否安裝成功:
      # ls -al /usr/lib | grep libevent
      lrwxrwxrwx    1 root     root          21 11?? 12 17:38 libevent-1.2.so.1 -> libevent-1.2.so.1.0.3
      -rwxr-xr-x       1 root     root          263546 11?? 12 17:38 libevent-1.2.so.1.0.3
      -rw-r--r--        1 root     root          454156 11?? 12 17:38 libevent.a
      -rwxr-xr-x       1 root     root          811 11?? 12 17:38 libevent.la
      lrwxrwxrwx    1 root     root          21 11?? 12 17:38 libevent.so -> libevent-1.2.so.1.0.3

      還不錯(cuò),都安裝上了,再來(lái)安裝memcache,同時(shí)需要安裝中指定libevent的安裝位置:
      # cd /tmp
      # tar zxvf memcached-1.2.0.tar.gz
      # cd memcached-1.2.0
      # ./configure --with-libevent=/usr
      # make
      # make install

      如果中間出現(xiàn)報(bào)錯(cuò),請(qǐng)仔細(xì)檢查錯(cuò)誤信息,按照錯(cuò)誤信息來(lái)配置或者增加相應(yīng)的庫(kù)或者路徑。
      安裝完成后會(huì)把memcached放到 /usr/local/bin/memcached ,我們看以下是否安裝了:
      # ls -al /usr/local/bin/mem*
      -rwxr-xr-x    1 root     root       137986 11?? 12 17:39 /usr/local/bin/memcached
      -rwxr-xr-x    1 root     root       140179 11?? 12 17:39 /usr/local/bin/memcached-debug

      恩,安裝完成了,現(xiàn)在我們看以下memcache的幫助:
      # /usr/local/bin/memecached -h
      memcached 1.2.0
      -p <num>            port number to listen on
      -s <file>               unix socket path to listen on (disables network support)
      -l <ip_addr>        interface to listen on, default is INDRR_ANY
      -d                          run as a daemon
      -r                           maximize core file limit
      -u <username> assume identity of <username> (only when run as root)
      -m <num>          max memory to use for items in megabytes, default is 64 MB
      -M                         return error on memory exhausted (rather than removing items)
      -c <num>            max simultaneous connections, default is 1024
      -k                          lock down all paged memory
      -v                          verbose (print errors/warnings while in event loop)
      -vv                        very verbose (also print client commands/reponses)
      -h                         print this help and exit
      -i                          print memcached and libevent license
      -b                         run a managed instanced (mnemonic: buckets)
      -P <file>             save PID in <file>, only used with -d option
      -f <factor>          chunk size growth factor, default 1.25
      -n <bytes>         minimum space allocated for key+value+flags, default 48

      參數(shù)不算多,我們來(lái)啟動(dòng)一個(gè)Memcache的服務(wù)器端:
      # /usr/local/bin/memcached -d -m 10  -u root -l 192.168.0.200 -p 12000 -c 256 -P /tmp/memcached.pid
      -d選項(xiàng)是啟動(dòng)一個(gè)守護(hù)進(jìn)程,-m是分配給Memcache使用的內(nèi)存數(shù)量,單位是MB,我這里是10MB,-u是運(yùn)行Memcache的用戶,我這里是root,-l是監(jiān)聽的服務(wù)器IP地址,如果有多個(gè)地址的話,我這里指定了服務(wù)器的IP地址192.168.0.200,-p是設(shè)置Memcache監(jiān)聽的端口,我這里設(shè)置了12000,最好是1024以上的端口,-c選項(xiàng)是最大運(yùn)行的并發(fā)連接數(shù),默認(rèn)是1024,我這里設(shè)置了256,按照你服務(wù)器的負(fù)載量來(lái)設(shè)定,-P是設(shè)置保存Memcache的pid文件,我這里是保存在/tmp/memcached.pid,如果要結(jié)束Memcache進(jìn)程,執(zhí)行:
      # kill `cat /tmp/memcached.pid`

      也可以啟動(dòng)多個(gè)守護(hù)進(jìn)程,不過(guò)端口不能重復(fù)。

      【安裝Memcache的PHP擴(kuò)展】
      Memcache就是在服務(wù)器監(jiān)聽端口,通過(guò)一定的協(xié)議交互來(lái)寫入數(shù)據(jù)到服務(wù)器內(nèi)存中,或者獲取一些值。如果你了解Memcache的交互協(xié)議,完全可以自己構(gòu)建Memcache的客戶端,目前網(wǎng)上也有很多構(gòu)建好的Memcache客戶端的PHPClass,可以直接用,不過(guò)我這里為了效率,還是決定使用PECL中Memcache的專用擴(kuò)展,因?yàn)楫吘故怯肅寫的,效率比較高,而且安裝部署比較方便。
      下載PECL中的Memcache,因?yàn)槲业目蛻舳耸荳indows XP,所以需要下載dll版,我的PHP版本是PHP 5.1.4,必須下載PHP 5.1專用的擴(kuò)展。
      PECL官網(wǎng):http://pecl.php.net (For Linux)
                             http://pecl4win.php.net(For Windows)
      擴(kuò)展下載: http://pecl4win.php.net/download.php/ext/5_1/5.1.2/php_memcache.dll
      如果你的PHP是其他版本,請(qǐng)到 http://pecl4win.php.net/ext.php/php_memcache.dll 選擇你相應(yīng)的版本,如果是Linux下的PHP,請(qǐng)到 http://pecl.php.net/package/memcache 選擇相應(yīng)想要下載的版本。
      下載完了以后,我把php_memcache.dll 拷貝到 c:\php5\ext 目錄下,如果你的擴(kuò)展目錄是在是缺省路徑,(就是沒(méi)有修改過(guò)php.ini中的擴(kuò)展路徑) 請(qǐng)拷貝到 c:\windows\ 目錄下,如果是Linux平臺(tái),請(qǐng)自己編譯安裝,可以在程序中使用dl()函數(shù)加載,或者在編譯php的時(shí)候加載進(jìn)去。最后重啟Web服務(wù)器,IIS/Apache。
      我的網(wǎng)站目錄是在:d:\mysite 目錄下,現(xiàn)在建立一個(gè) phpinfo.php 文件在網(wǎng)站根目錄下,代碼是:
      <?phpinfo()?>
      看有沒(méi)有成功加載 php_memcache.dll 擴(kuò)展。如果顯示了 Memcache 選項(xiàng)和相應(yīng)的版本信息,則說(shuō)明成功加載了,否則請(qǐng)仔細(xì)檢查上面的步驟。
      如果一切正確無(wú)誤,那么說(shuō)明安裝成功。

      【Memcache初試】
      [ 接口介紹 ]
      服務(wù)器端和客戶端都安裝配置好了,現(xiàn)在我們就來(lái)測(cè)試以下我們的成果。Memcache客戶端包含兩組接口,一組是面向過(guò)程的接口,一組是面向?qū)ο蟮慕涌冢唧w可以參考PHP手冊(cè) “LXXV. Memcache Functions”這章。我們?yōu)榱撕?jiǎn)單方便,就使用面向?qū)ο蟮姆绞剑脖阌诰S護(hù)和編寫代碼。Memcache面向?qū)ο蟮某S媒涌诎ǎ?br />Memcache::connect -- 打開一個(gè)到Memcache的連接
      Memcache::pconnect -- 打開一個(gè)到Memcache的長(zhǎng)連接
      Memcache::close -- 關(guān)閉一個(gè)Memcache的連接
      Memcache::set -- 保存數(shù)據(jù)到Memcache服務(wù)器上
      Memcache::get -- 提取一個(gè)保存在Memcache服務(wù)器上的數(shù)據(jù)
      Memcache::replace -- 替換一個(gè)已經(jīng)存在Memcache服務(wù)器上的項(xiàng)目(功能類似Memcache::set)
      Memcache::delete -- 從Memcache服務(wù)器上刪除一個(gè)保存的項(xiàng)目
      Memcache::flush -- 刷新所有Memcache服務(wù)器上保存的項(xiàng)目(類似于刪除所有的保存的項(xiàng)目)
      Memcache::getStats -- 獲取當(dāng)前Memcache服務(wù)器運(yùn)行的狀態(tài)

      [ 測(cè)試代碼 ]
      現(xiàn)在我們開始一段測(cè)試代碼:

      <?php
      //連接
      $mem=new Memcache;
      $mem->connect("192.168.0.200",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:


      基本說(shuō)明我們的Memcache安裝成功,我們?cè)賮?lái)分析以下上面的這段程序。

      [ 程序分析 ]
      初始化一個(gè)Memcache的對(duì)象:
      $mem = new Memcache;
      連接到我們的Memcache服務(wù)器端,第一個(gè)參數(shù)是服務(wù)器的IP地址,也可以是主機(jī)名,第二個(gè)參數(shù)是Memcache的開放的端口:
      $mem->connect("192.168.0.200", 12000);
      保存一個(gè)數(shù)據(jù)到Memcache服務(wù)器上,第一個(gè)參數(shù)是數(shù)據(jù)的key,用來(lái)定位一個(gè)數(shù)據(jù),第二個(gè)參數(shù)是需要保存的數(shù)據(jù)內(nèi)容,這里是一個(gè)字符串,第三個(gè)參數(shù)是一個(gè)標(biāo)記,一般設(shè)置為0或者M(jìn)EMCACHE_COMPRESSED就行了,第四個(gè)參數(shù)是數(shù)據(jù)的有效期,就是說(shuō)數(shù)據(jù)在這個(gè)時(shí)間內(nèi)是有效的,如果過(guò)去這個(gè)時(shí)間,那么會(huì)被Memcache服務(wù)器端清除掉這個(gè)數(shù)據(jù),單位是秒,如果設(shè)置為0,則是永遠(yuǎn)有效,我們這里設(shè)置了60,就是一分鐘有效時(shí)間:
      $mem->set('key1', 'This is first value', 0, 60);
      從Memcache服務(wù)器端獲取一條數(shù)據(jù),它只有一個(gè)參數(shù),就是需要獲取數(shù)據(jù)的key,我們這里是上一步設(shè)置的key1,現(xiàn)在獲取這個(gè)數(shù)據(jù)后輸出輸出:
      $val = $mem->get('key1');
      echo "Get key1 value: " .
      $val;
      現(xiàn)在是使用replace方法來(lái)替換掉上面key1的值,replace方法的參數(shù)跟set是一樣的,不過(guò)第一個(gè)參數(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上面保存了一個(gè)數(shù)組,然后獲取回來(lái)并輸出
      $arr = array('aaa', 'bbb', 'ccc', 'ddd');
      $mem->set('key2', $arr, 0, 60);
      $val2 = $mem->get('key2');
      print_r($val2);

      現(xiàn)在刪除一個(gè)數(shù)據(jù),使用delte接口,參數(shù)就是一個(gè)key,然后就能夠把Memcache服務(wù)器這個(gè)key的數(shù)據(jù)刪除,最后輸出的時(shí)候沒(méi)有結(jié)果
      $mem->delete('key1');
      $val = $mem->get('key1');
      echo "Get key1 value: " . $val .
      "<br>";
      最后我們把所有的保存在Memcache服務(wù)器上的數(shù)據(jù)都清除,會(huì)發(fā)現(xiàn)數(shù)據(jù)都沒(méi)有了,最后輸出key2的數(shù)據(jù)為空,最后關(guān)閉連接
      $mem->flush();
      $val2 = $mem->get('key2');
      echo "Get key2 value: ";
      print_r($val2);
      echo
      "<br>";


      【Memcache協(xié)議分析】
      如果你不喜歡 php_memcache.dll 擴(kuò)展或者服務(wù)器器目前不支持這個(gè)擴(kuò)展,那么就可以考慮自己構(gòu)建,需要構(gòu)建Memcahe的客戶端,要先了解Memcache協(xié)議的交互,這樣才能開發(fā)自己的客戶端,我這里就簡(jiǎn)單的分析以下Memcache的協(xié)議。
      (更詳細(xì)的協(xié)議內(nèi)容請(qǐng)?jiān)贛emcache服務(wù)器端的源碼的 doc/protocol.txt 文件中,本文基本來(lái)源于此)
      Memcache既支持TCP協(xié)議,也支持UDP協(xié)議,不過(guò)我們這里是以TCP協(xié)議的協(xié)議作為主要考慮對(duì)象,想了解UDP協(xié)議的過(guò)程,請(qǐng)參考 doc/protocol.txt 文件。
      [ 錯(cuò)誤指令]
      Memcache的協(xié)議的錯(cuò)誤部分主要是三個(gè)錯(cuò)誤提示之提示指令:
      普通錯(cuò)誤信息,比如指令錯(cuò)誤之類的
      ERROR\r\n
      客戶端錯(cuò)誤
      CLIENT_ERROR <錯(cuò)誤信息>\r\n
      服務(wù)器端錯(cuò)誤
      SERVER_ERROR <錯(cuò)誤信息>\r\n
      [ 數(shù)據(jù)保存指令]
      數(shù)據(jù)保存是基本的功能,就是客戶端通過(guò)命令把數(shù)據(jù)返回過(guò)來(lái),服務(wù)器端接收后進(jìn)行處理。
      指令格式:
      <命令> <鍵> <標(biāo)記> <有效期> <數(shù)據(jù)長(zhǎng)度>\r\n
      <命令> - command name
      主要是三個(gè)儲(chǔ)存數(shù)據(jù)的三個(gè)命令, set, add, replace
      set 命令是保存一個(gè)叫做key的數(shù)據(jù)到服務(wù)器上
      add 命令是添加一個(gè)數(shù)據(jù)到服務(wù)器,但是服務(wù)器必須這個(gè)key是不存在的,能夠保證數(shù)據(jù)不會(huì)被覆蓋
      replace 命令是替換一個(gè)已經(jīng)存在的數(shù)據(jù),如果數(shù)據(jù)不存在,就是類似set功能
      <鍵> - key
      就是保存在服務(wù)器上唯一的一個(gè)表示符,必須是跟其他的key不沖突,否則會(huì)覆蓋掉原來(lái)的數(shù)據(jù),這個(gè)key是為了能夠準(zhǔn)確的存取一個(gè)數(shù)據(jù)項(xiàng)目
      <標(biāo)記> - flag
      標(biāo)記是一個(gè)16位的無(wú)符號(hào)整形數(shù)據(jù),用來(lái)設(shè)置服務(wù)器端跟客戶端一些交互的操作
      <有效期> - expiration time
      是數(shù)據(jù)在服務(wù)器上的有效期限,如果是0,則數(shù)據(jù)永遠(yuǎn)有效,單位是秒,Memcache服務(wù)器端會(huì)把一個(gè)數(shù)據(jù)的有效期設(shè)置為當(dāng)前Unix時(shí)間+設(shè)置的有效時(shí)間
      <數(shù)據(jù)長(zhǎng)度> - bytes
      數(shù)據(jù)的長(zhǎng)度,block data 塊數(shù)據(jù)的長(zhǎng)度,一般在這個(gè)個(gè)長(zhǎng)度結(jié)束以后下一行跟著block data數(shù)據(jù)內(nèi)容,發(fā)送完數(shù)據(jù)以后,客戶端一般等待服務(wù)器端的返回,服務(wù)器端的返回:
      數(shù)據(jù)保存成功
      STORED\r\n
      數(shù)據(jù)保存失敗,一般是因?yàn)榉⻊?wù)器端這個(gè)數(shù)據(jù)key已經(jīng)存在了
      NOT_STORED\r\n

      [ 數(shù)據(jù)提取命令]
      從服務(wù)器端提取數(shù)據(jù)主要是使用get指令,格式是:
      get <鍵>*\r\n
      <鍵>* - key
      key是是一個(gè)不為空的字符串組合,發(fā)送這個(gè)指令以后,等待服務(wù)器的返回。如果服務(wù)器端沒(méi)有任何數(shù)據(jù),則是返回:
      END\r\n
      證明沒(méi)有不存在這個(gè)key,沒(méi)有任何數(shù)據(jù),如果存在數(shù)據(jù),則返回指定格式:
      VALUE <> <標(biāo)記> <數(shù)據(jù)長(zhǎng)度>\r\n
      <數(shù)據(jù)塊>\r\n
      返回的數(shù)據(jù)是以VALUE開始的,后面跟著key和flags,以及數(shù)據(jù)長(zhǎng)度,第二行跟著數(shù)據(jù)塊。
      <鍵> -key
      是發(fā)送過(guò)來(lái)指令的key內(nèi)容
      <標(biāo)記> - flags
      是調(diào)用set指令保存數(shù)據(jù)時(shí)候的flags標(biāo)記
      <數(shù)據(jù)長(zhǎng)度> - bytes
      是保存數(shù)據(jù)時(shí)候定位的長(zhǎng)度
      <數(shù)據(jù)塊> - data block
      數(shù)據(jù)長(zhǎng)度下一行就是提取的數(shù)據(jù)塊內(nèi)容

      [ 數(shù)據(jù)刪除指令]
      數(shù)據(jù)刪除指令也是比較簡(jiǎn)單的,使用get指令,格式是:
      delete <鍵> <超時(shí)時(shí)間>\r\n
      <鍵> - key
      key是你希望在服務(wù)器上刪除數(shù)據(jù)的key鍵
      <超時(shí)時(shí)間> - timeout
      按照秒為單位,這個(gè)是個(gè)可選項(xiàng),如果你沒(méi)有指定這個(gè)值,那么服務(wù)器上key數(shù)據(jù)將馬上被刪除,如果設(shè)置了這個(gè)值,那么數(shù)據(jù)將在超時(shí)時(shí)間后把數(shù)據(jù)清除,該項(xiàng)缺省值是0,就是馬上被刪除
      刪除數(shù)據(jù)后,服務(wù)器端會(huì)返回:
      DELETED\r\n
      刪除數(shù)據(jù)成功
      NOT_FOUND\r\n
      這個(gè)key沒(méi)有在服務(wù)器上找到
      如果要?jiǎng)h除所有服務(wù)器上的數(shù)據(jù),可以使用flash_all指令,格式:
      flush_all\r\n

      這個(gè)指令執(zhí)行后,服務(wù)器上所有緩存的數(shù)據(jù)都被刪除,并且返回:
      OK\r\n
      這個(gè)指令一般不要輕易使,除非你卻是想把所有數(shù)據(jù)都干掉,刪除完以后可以無(wú)法恢復(fù)的。

      [其他指令]
      如果想了解當(dāng)前Memcache服務(wù)器的狀態(tài)和版本等信息,可以使用狀態(tài)查詢指令和版本查詢指令。
      如果想了解當(dāng)前所有Memcache服務(wù)器運(yùn)行的狀態(tài)信息,可以使用stats指令,格式
      stats\r\n
      服務(wù)器將返回每行按照 STAT 開始的狀態(tài)信息,包括20行,20項(xiàng)左右的信息,包括守護(hù)進(jìn)程的pid、版本、保存的項(xiàng)目數(shù)量、內(nèi)存占用、最大內(nèi)存限制等等信息。

      如果只是想獲取部分項(xiàng)目的信息,可以指定參數(shù),格式:
      stats <參數(shù)>\r\n
      這個(gè)指令將只返回指定參數(shù)的項(xiàng)目狀態(tài)信息。
      如果只是想單獨(dú)了解當(dāng)前版本信息,可以使用version指令,格式:
      version\r\n
      將返回以 VERSION 開頭的版本信息
      如果想結(jié)束當(dāng)前連接,使用quit指令,格式:
      quit\r\n
      將斷開當(dāng)前連接
      另外還有其他指令,包括incr, decr 等,我也不太了解作用,就不做介紹了,如果感興趣,可以自己去研究。

      【Memcache在中型網(wǎng)站的使用】
      使用Memcache的網(wǎng)站一般流量都是比較大的,為了緩解數(shù)據(jù)庫(kù)的壓力,讓Memcache作為一個(gè)緩存區(qū)域,把部分信息保存在內(nèi)存中,在前端能夠迅速的進(jìn)行存取。那么一般的焦點(diǎn)就是集中在如何分擔(dān)數(shù)據(jù)庫(kù)壓力和進(jìn)行分布式,畢竟單臺(tái)Memcache的內(nèi)存容量的有限的。我這里簡(jiǎn)單提出我的個(gè)人看法,未經(jīng)實(shí)踐,權(quán)當(dāng)參考。
      [ 分布式應(yīng)用]
      Memcache本來(lái)支持分布式,我們客戶端稍加改造,更好的支持。我們的key可以適當(dāng)進(jìn)行有規(guī)律的封裝,比如以u(píng)ser為主的網(wǎng)站來(lái)說(shuō),每個(gè)用戶都有UserID,那么可以按照固定的ID來(lái)進(jìn)行提取和存取,比如1開頭的用戶保存在第一臺(tái)Memcache服務(wù)器上,以2開頭的用戶的數(shù)據(jù)保存在第二胎Mecache服務(wù)器上,存取數(shù)據(jù)都先按照User ID來(lái)進(jìn)行相應(yīng)的轉(zhuǎn)換和存取。
      但是這個(gè)有缺點(diǎn),就是需要對(duì)User ID進(jìn)行判斷,如果業(yè)務(wù)不一致,或者其他類型的應(yīng)用,可能不是那么合適,那么可以根據(jù)自己的實(shí)際業(yè)務(wù)來(lái)進(jìn)行考慮,或者去想更合適的方法。
      [ 減少數(shù)據(jù)庫(kù)壓力]
      這個(gè)算是比較重要的,所有的數(shù)據(jù)基本上都是保存在數(shù)據(jù)庫(kù)當(dāng)中的,每次頻繁的存取數(shù)據(jù)庫(kù),導(dǎo)致數(shù)據(jù)庫(kù)性能極具下降,無(wú)法同時(shí)服務(wù)更多的用戶,比如MySQL,特別頻繁的鎖表,那么讓Memcache來(lái)分擔(dān)數(shù)據(jù)庫(kù)的壓力吧。我們需要一種改動(dòng)比較小,并且能夠不會(huì)大規(guī)模改變前端的方式來(lái)進(jìn)行改變目前的架構(gòu)。
      我考慮的一種簡(jiǎn)單方法:
      后端的數(shù)據(jù)庫(kù)操作模塊,把所有的Select操作提取出來(lái)(update/delete/insert不管),然后把對(duì)應(yīng)的SQL進(jìn)行相應(yīng)的hash算法計(jì)算得出一個(gè)hash數(shù)據(jù)key(比如MD5或者SHA),然后把這個(gè)key去Memcache中查找數(shù)據(jù),如果這個(gè)數(shù)據(jù)不存在,說(shuō)明還沒(méi)寫入到緩存中,那么從數(shù)據(jù)庫(kù)把數(shù)據(jù)提取出來(lái),一個(gè)是數(shù)組類格式,然后把數(shù)據(jù)在set到Memcache中,key就是這個(gè)SQL的hash值,然后相應(yīng)的設(shè)置一個(gè)失效時(shí)間,比如一個(gè)小時(shí),那么一個(gè)小時(shí)中的數(shù)據(jù)都是從緩存中提取的,有效減少數(shù)據(jù)庫(kù)的壓力。
      缺點(diǎn)是數(shù)據(jù)不實(shí)時(shí),當(dāng)數(shù)據(jù)做了修改以后,無(wú)法實(shí)時(shí)到前端顯示,并且還有可能對(duì)內(nèi)存占用比較大,畢竟每次select出來(lái)的數(shù)據(jù)數(shù)量可能比較巨大,這個(gè)是需要考慮的因素。
      上面只是我兩點(diǎn)沒(méi)有經(jīng)過(guò)深思熟慮的簡(jiǎn)單想法,也許有用,那就最好了。

      【Memcache的安全】
      我們上面的Memcache服務(wù)器端都是直接通過(guò)客戶端連接后直接操作,沒(méi)有任何的驗(yàn)證過(guò)程,這樣如果服務(wù)器是直接暴露在互聯(lián)網(wǎng)上的話是比較危險(xiǎn),輕則數(shù)據(jù)泄露被其他無(wú)關(guān)人員查看,重則服務(wù)器被入侵,因?yàn)镸ecache是以root權(quán)限運(yùn)行的,況且里面可能存在一些我們未知的bug或者是緩沖區(qū)溢出的情況,這些都是我們未知的,所以危險(xiǎn)性是可以預(yù)見的。
      為了安全起見,我做兩點(diǎn)建議,能夠稍微的防止黑客的入侵或者數(shù)據(jù)的泄露。
      [ 內(nèi)網(wǎng)訪問(wèn)]
      最好把兩臺(tái)服務(wù)器之間的訪問(wèn)是內(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ù)器通過(guò)內(nèi)網(wǎng)的網(wǎng)卡來(lái)訪問(wèn)Memcache服務(wù)器,我們Memcache的服務(wù)器上啟動(dòng)的時(shí)候就監(jiān)聽內(nèi)網(wǎng)的IP地址和端口,內(nèi)網(wǎng)間的訪問(wèn)能夠有效阻止其他非法的訪問(wèn)。
      # memcached -d -m 1024  -u root -l 192.168.0.200 -p 11211 -c 1024 -P /tmp/memcached.pid
      Memcache服務(wù)器端設(shè)置監(jiān)聽通過(guò)內(nèi)網(wǎng)的192.168.0.200的ip的11211端口,占用1024MB內(nèi)存,并且允許最大1024個(gè)并發(fā)連接

      [ 設(shè)置防火墻]
      防火墻是簡(jiǎn)單有效的方式,如果卻是兩臺(tái)服務(wù)器都是掛在網(wǎng)的,并且需要通過(guò)外網(wǎng)IP來(lái)訪問(wèn)Memcache的話,那么可以考慮使用防火墻或者代理程序來(lái)過(guò)濾非法訪問(wèn)。
      一般我們?cè)贚inux下可以使用iptables或者FreeBSD下的ipfw來(lái)指定一些規(guī)則防止一些非法的訪問(wèn),比如我們可以設(shè)置只允許我們的Web服務(wù)器來(lái)訪問(wèn)我們Memcache服務(wù)器,同時(shí)阻止其他的訪問(wèn)。
      # 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規(guī)則就是只允許192.168.0.2這臺(tái)Web服務(wù)器對(duì)Memcache服務(wù)器的訪問(wèn),能夠有效的阻止一些非法訪問(wèn),相應(yīng)的也可以增加一些其他的規(guī)則來(lái)加強(qiáng)安全性,這個(gè)可以根據(jù)自己的需要來(lái)做。

      【Memcache的擴(kuò)展性】
      Memcache算是比較簡(jiǎn)潔高效的程序,Memcache 1.2.0 的源代碼大小才139K,在Windows平臺(tái)上是不可想象的,但是在開源世界來(lái)說(shuō),這是比較正常合理的。
      Memcache目前都只是比較簡(jiǎn)單的功能,簡(jiǎn)單的數(shù)據(jù)存取功能,我個(gè)人希望如果有識(shí)之士,能夠在下面兩方面進(jìn)行擴(kuò)展。
      1. 日志功能
      目前Memcache沒(méi)有日志功能,只有一些命令在服務(wù)器端進(jìn)行回顯,這樣是很不利于對(duì)一個(gè)服務(wù)器的穩(wěn)定性和負(fù)載等等進(jìn)行監(jiān)控的,最好能夠相應(yīng)的加上日志的等功能,便于監(jiān)控。
      2. 存儲(chǔ)結(jié)構(gòu)
      目前的數(shù)據(jù)形式就是: key => data 的形式,特別單一,只能夠存儲(chǔ)單一的一維數(shù)據(jù),如果能夠擴(kuò)展的話,變成類似數(shù)據(jù)庫(kù)的格式,能夠存儲(chǔ)二維數(shù)據(jù),那樣會(huì)讓可以用性更強(qiáng),使用面更廣,當(dāng)然相應(yīng)的可能代碼效率和存取效率更差一些。
      3. 同步功能
      數(shù)據(jù)同步是個(gè)比較重要的技術(shù),因?yàn)檎l(shuí)都不能保證一臺(tái)服務(wù)器是持久正常的運(yùn)行的,如果能夠具有類似MySQL的 Master/Slave的功能,那么將使得Memcache的數(shù)據(jù)更加穩(wěn)定,那么相應(yīng)的就可以考慮存儲(chǔ)持久一點(diǎn)的數(shù)據(jù),并且不用害怕Memcache的down掉,因?yàn)橛型降膫浞莘⻊?wù)器,這個(gè)問(wèn)題就不是問(wèn)題了。
      以上三點(diǎn)只是個(gè)人拙見,有識(shí)之士和技術(shù)高手可以考慮。

      【結(jié)束語(yǔ)】
      我上面的內(nèi)容都只是自己安裝和使用的一些想法,不能保證絕對(duì)正確,只是給需要的人一個(gè)參考,一個(gè)推廣Memcache的文章,希望更多的人能夠認(rèn)識(shí)和了解這個(gè)技術(shù),并且為自己所用。
      我花費(fèi)了整整一個(gè)晚上的時(shí)間洋洋灑灑的寫了這么長(zhǎng),無(wú)非是對(duì)于這項(xiàng)開源技術(shù)的熱愛(ài),我想開源世界能夠繁榮起來(lái),就是源于大家的熱愛(ài)并且愿意做出貢獻(xiàn),開源世界才這么精彩。
      希望本文能夠給需要的人一些幫助,希望不會(huì)誤導(dǎo)他們,呵呵。


      附加:(我操作Memcache相應(yīng)對(duì)應(yīng)上面文章內(nèi)容的圖片)
      [ 啟動(dòng)Memcache]



      [ Memcache的PHP測(cè)試代碼]



      [測(cè)試代碼執(zhí)行效果]


      [ 通過(guò)Telnet連接到Memcache ]


      [ 基本的Memcache的數(shù)據(jù)存取協(xié)議交互]



      [ Memcache狀態(tài)信息協(xié)議交互]

      來(lái)源:網(wǎng)絡(luò)搜集//所屬分類:Web服務(wù)器教程/更新時(shí)間:2013-04-14
      相關(guān)Web服務(wù)器教程