Redis內(nèi)存數(shù)據(jù)庫存儲(chǔ)

前言

會(huì)話是臨時(shí)存儲(chǔ)數(shù)據(jù)信息的一種機(jī)制遂赠。但是如果項(xiàng)目需要使用一些高復(fù)用的服務(wù)器架構(gòu)(如下圖)潜圃,就需要實(shí)現(xiàn)session數(shù)據(jù)同步或者統(tǒng)一存儲(chǔ),不然就有可能出現(xiàn)會(huì)話狀態(tài)丟失的問題创千。所以如何共享Session信息就是一個(gè)需要解決的問題间学。下面本文講述的是,如何使用Thinkphp搭配Redis存儲(chǔ)Session信息(單應(yīng)用服務(wù)器嘹屯,單redis版本)攻询,以達(dá)到Session共享的目的。

base lbs

環(huán)境

  • 操作系統(tǒng) Linux ubuntu 16.04
  • PHP環(huán)境 PHP 7.0.28-0ubuntu0.16.04.1 ( NTS )
  • redis版本 4.0.8

介紹

Redis是一個(gè)使用ANSI C編寫的開源州弟、支持網(wǎng)絡(luò)钧栖、基于內(nèi)存、可選持久性的鍵值對(duì)存儲(chǔ)數(shù)據(jù)庫婆翔。根據(jù)月度排行網(wǎng)站DB-Engines.com的數(shù)據(jù)顯示桐经,Redis是最流行的鍵值對(duì)存儲(chǔ)數(shù)據(jù)庫

安裝Redis

源碼包安裝

根據(jù)redis官網(wǎng)的教程
1, Download, extract and compile Redis with:

wget http://download.redis.io/releases/redis-4.0.8.tar.gz
tar xzf redis-4.0.8.tar.gz
cd redis-4.0.8
make

2, The binaries that are now compiled are available in the src directory. Run Redis with:

src/redis-server

3, You can interact with Redis using the built-in client:

src/redis-cli
redis> set foo bar
OK
redis> get foo
"bar"

一切順利的話,源碼包Redis就安裝完畢了浙滤。另外我們也可以使用Docker安裝Redis。

Docker倉庫安裝

這里是Redis的倉庫 气堕,提供了 Redis 3.x ~ 4.x 各個(gè)版本的鏡像纺腊。

1, 查看docker-hub上的遠(yuǎn)程倉庫

docker search redis
image

2, 執(zhí)行下面命令畔咧,大意是首先docker會(huì)本地搜索docker images中理出來的鏡像,如果沒有則從docker hub(國(guó)內(nèi)鏡像)拉取redis的docker鏡像揖膜,然后啟動(dòng)一個(gè)名為some-redis容器誓沸,-p是把本地的9736端口和容器中的6379端口映射起來,-d (daemon)并讓容器運(yùn)行在后臺(tái),--requirepass客戶端鏈接需要認(rèn)證壹粟。

docker run --name my-redis -p 9736:6379 -d redis --requirepass "11110000"
##查看docker已經(jīng)啟動(dòng)的容器
docker ps
##安裝redis客戶端
sudo apt install reids-tools
##嘗試從外部主機(jī)范圍docker中的redis
redis-cli -h 127.0.0.1 -p 9736 -a 11110000
test-redis

Docker倉庫安裝的程序拜隧,拓展遷徙相對(duì)簡(jiǎn)單,和其他系統(tǒng)的耦合性很小趁仙。接下來我們安裝拓展洪添。

安裝php-redis 拓展

拓展我們選擇phpredis,這個(gè)比較通用雀费,更新比較及時(shí)干奢。github地址在這里

git clone https://github.com/phpredis/phpredis.git
##切換到phpredis所在的文件夾
phpize
./configure
make && make install
##重啟php服務(wù)
systemctl restart  php7.0-fpm.service

這時(shí)候就可以去phpinfo頁面看到redis拓展已經(jīng)成功出現(xiàn)了。


phpinfo-redis.png

這時(shí)候盏袄,我們就可以在php程序中操作redis數(shù)據(jù)庫了忿峻。新建一個(gè)文件vim phpredis.php,內(nèi)容如下辕羽。

<?php
 if( !extension_loaded('redis')){
     echo "not support redis";
     exit();
  }
  $redis = new Redis();
  $redis->connect('127.0.0.1',9736);
  $redis->auth("11110000");
  var_dump($redis->info());
##執(zhí)行上面的測(cè)試文件逛尚,如果成功了,就出現(xiàn)類似下面的結(jié)果刁愿。
php phpredis.php
redis-out.png

到這绰寞,php-redis拓展已經(jīng)安裝成功,我們也可以在php程序總操作redis數(shù)據(jù)庫了酌毡。接下來我們就在項(xiàng)目中使用redis存儲(chǔ)session克握。

Redis存儲(chǔ)Session

如果是使用原生的php來實(shí)現(xiàn)這個(gè)功能,就需要調(diào)整配置文件php.ini枷踏,配置phpredis使用文檔編寫代碼菩暗。

session.save_handler = redis
session.save_path = "tcp://host1:6379?weight=1"

如果是使用php框架的話,一般框架已經(jīng)集成了對(duì)應(yīng)的模塊操作驅(qū)動(dòng)旭蠕。我們可以到thinkphp官網(wǎng)找到Session的Redis存儲(chǔ)驅(qū)動(dòng)類停团。我們需要在相對(duì)應(yīng)的位置(./ThinkPHP/Library/Think/Session/Driver/
)新建一個(gè)Redis.class.php文件,內(nèi)容如下掏熬。

<?php
namespace Think\Session\Driver;

/**
 * Redis Session驅(qū)動(dòng)
 */
class Redis {

    /**
     * Redis句柄
     */
    private $handler;
    private $get_result;

    public function __construct(){
        if ( !extension_loaded('redis') ) {
            E(L('_NOT_SUPPERT_').':redis');
        }
        if(empty($options)) {
            $options = array (
            'host'       => C('REDIS_HOST') ? C('REDIS_HOST') : '127.0.0.1',
            'port'       => C('REDIS_PORT') ? C('REDIS_PORT') : 6379,
            'timeout'    => C('REDIS_TIMEOUT') ? C('REDIS_TIMEOUT') : false,
            'persistent' => C('REDIS_PERSISTENT') ? C('REDIS_PERSISTENT'):false,
            'auth'      => C('REDIS_AUTH') ? C('REDIS_AUTH') : false,
            );
        }
        $options['host'] = explode(',', $options['host']);
        $options['port'] = explode(',', $options['port']);
        $options['auth'] = explode(',', $options['auth']);
        foreach ($options['host'] as $key=>$value) {
            if (!isset($options['port'][$key])) {
                $options['port'][$key] = $options['port'][0];
            }
            if (!isset($options['auth'][$key])) {
                $options['auth'][$key] = $options['auth'][0];
            }
        }
        $this->options =  $options;
        $expire = C('SESSION_EXPIRE');
        $this->options['expire'] =  
        isset($expire) ? (int)$expire : (int)ini_get('session.gc_maxlifetime');
        $this->options['prefix'] =  
        isset($options['prefix']) ?  $options['prefix']  :  C('SESSION_PREFIX');
        $this->handler  = new \Redis;
    }

    /**
     * 連接Redis服務(wù)端
     * @access public
     * @param bool $is_master : 是否連接主服務(wù)器
     */
    public function connect($is_master = true) {
        if ($is_master) {
            $i = 0;
        } else {
            $count = count($this->options['host']);
            if ($count == 1) {
                $i = 0;
            } else {
                $i = rand(1, $count - 1);   //多個(gè)從服務(wù)器隨機(jī)選擇
            }
        }
        $func = $this->options['persistent'] ? 'pconnect' : 'connect';
        try {
            if ($this->options['timeout'] === false) {
                $result = $this
                ->handler
                ->$func($this->options['host'][$i], $this->options['port'][$i]);
                if (!$result)
                    throw new \Think\Exception('Redis Error', 100);
            } else {
                $result = $this
                ->handler
                ->$func($this->options['host'][$i], $this->options['port'][$i],
                $this->options['timeout']);
                if (!$result)
                    throw new \Think\Exception('Redis Error', 101);
            }
            if ($this->options['auth'][$i]) {

                $result = $this->handler->auth($this->options['auth'][$i]);
                if (!$result) {
                    throw new \Think\Exception('Redis Error', 102);
                }
            }
        } catch ( \Exception $e ) {
          exit('Error:'.$e->getMessage().'<br>Error Code:'.$e->getCode().'');
        }
    }

    /**
      +----------------------------------------------------------
     * 打開Session
      +----------------------------------------------------------
     * @access public
      +----------------------------------------------------------
     * @param string $savePath
     * @param mixed $sessName
      +----------------------------------------------------------
     */
    public function open($savePath, $sessName) {
        return true;
    }

    /**
      +----------------------------------------------------------
     * 關(guān)閉Session
      +----------------------------------------------------------
     * @access public
      +----------------------------------------------------------
     */
    public function close() {
        if ($this->options['persistent'] == 'pconnect') {
            $this->handler->close();
        }
        return true;
    }

    /**
      +----------------------------------------------------------
     * 讀取Session
      +----------------------------------------------------------
     * @access public
      +----------------------------------------------------------
     * @param string $sessID
      +----------------------------------------------------------
     */
    public function read($sessID) {
      $this->connect(0);
      $this->get_result = $this->handler->get($this->options['prefix'].$sessID);
      //延長(zhǎng)有效期
      $this->handler->expire($this->options['prefix']
      .$sessID,C('SESSION_EXPIRE'));
      return $this->get_result;
    }

    /**
      +----------------------------------------------------------
     * 寫入Session
      +----------------------------------------------------------
     * @access public
      +----------------------------------------------------------
     * @param string $sessID
     * @param String $sessData
      +----------------------------------------------------------
     */
    public function write($sessID, $sessData) {
        if (!$sessData || $sessData == $this->get_result) {
            return true;
        }
        $this->connect(1);
        $expire  =  $this->options['expire'];
        $sessID   =   $this->options['prefix'].$sessID;
        if(is_int($expire) && $expire > 0) {
            $result = $this->handler->setex($sessID, $expire, $sessData);
            $re = $result ? 'true' : 'false';
        }else{
            $result = $this->handler->set($sessID, $sessData);
            $re = $result ? 'true' : 'false';
        }
        return $result;
    }

    /**
      +----------------------------------------------------------
     * 刪除Session
      +----------------------------------------------------------
     * @access public
      +----------------------------------------------------------
     * @param string $sessID
      +----------------------------------------------------------
     */
    public function destroy($sessID) {
        $this->connect(1);
        return $this->handler->delete($this->options['prefix'].$sessID);
    }

    /**
      +----------------------------------------------------------
     * Session 垃圾回收
      +----------------------------------------------------------
     * @access public
      +----------------------------------------------------------
     * @param string $sessMaxLifeTime
      +----------------------------------------------------------
     */
    public function gc($sessMaxLifeTime) {
        return true;
    }

    /**
      +----------------------------------------------------------
     * 打開Session
      +----------------------------------------------------------
     * @access public
      +----------------------------------------------------------
     * @param string $savePath
     * @param mixed $sessName
      +----------------------------------------------------------
     */
    public function execute() {
        session_set_save_handler(
                array(&$this, "open"),
                array(&$this, "close"),
                array(&$this, "read"),
                array(&$this, "write"),
                array(&$this, "destroy"),
                array(&$this, "gc")
        );
    }

    public function __destruct() {
        if ($this->options['persistent'] == 'pconnect') {
            $this->handler->close();
        }
        session_write_close();
    }

}

接著在配置文件中配置如下配置項(xiàng)佑稠,不需要改動(dòng)php.ini配置項(xiàng)了。

//SESSION 配置
'SESSION_AUTO_START'    => true, //是否開啟session
'SESSION_TYPE'          =>  'Redis',    //session 驅(qū)動(dòng)
'SESSION_PREFIX'        =>  '',       //session前綴
'SESSION_EXPIRE'        =>  '7200',        //session有效期(單位:秒) 0表示永久緩存旗芬。
//Redis 配置
'REDIS_HOST'        =>'127.0.0.1', //redis服務(wù)器ip舌胶,多臺(tái)用逗號(hào)隔開;
'REDIS_PORT'        =>'9736',//端口號(hào)
'REDIS_TIMEOUT'     =>'30',//超時(shí)時(shí)間(秒)
'REDIS_PERSISTENT'  =>false,//是否長(zhǎng)連接 false=短連接
'REDIS_AUTH'        =>'11110000',//AUTH認(rèn)證密碼

其他層面的代碼不需要改動(dòng)疮丛,因?yàn)榭蚣芤呀?jīng)將Session類的操作方面封裝起來了幔嫂。業(yè)務(wù)存儲(chǔ)Session時(shí)候辆它,底層會(huì)根據(jù)配置項(xiàng)來實(shí)例化對(duì)應(yīng)的存儲(chǔ)驅(qū)動(dòng)類。從而就實(shí)現(xiàn)了session存儲(chǔ)在對(duì)應(yīng)的redis數(shù)據(jù)庫中了履恩。

如無意外锰茉,至此,我們就完成了這次的小目標(biāo)了切心。

結(jié)語

本文簡(jiǎn)單地實(shí)現(xiàn)了單機(jī)版的Redis存儲(chǔ)php程序的Session數(shù)據(jù)飒筑。下一步,計(jì)劃利用Docker技術(shù)搭建Redis基礎(chǔ)的高可用分布式集群绽昏,就本文的基礎(chǔ)上進(jìn)行項(xiàng)目架構(gòu)拓展协屡。如有興趣,敬請(qǐng)關(guān)注而涉!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末著瓶,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子啼县,更是在濱河造成了極大的恐慌材原,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,657評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件季眷,死亡現(xiàn)場(chǎng)離奇詭異余蟹,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)子刮,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門威酒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人挺峡,你說我怎么就攤上這事葵孤。” “怎么了橱赠?”我有些...
    開封第一講書人閱讀 164,057評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵尤仍,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我狭姨,道長(zhǎng)宰啦,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,509評(píng)論 1 293
  • 正文 為了忘掉前任饼拍,我火速辦了婚禮赡模,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘师抄。我一直安慰自己漓柑,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,562評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著欺缘,像睡著了一般栋豫。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上谚殊,一...
    開封第一講書人閱讀 51,443評(píng)論 1 302
  • 那天,我揣著相機(jī)與錄音蛤铜,去河邊找鬼嫩絮。 笑死,一個(gè)胖子當(dāng)著我的面吹牛围肥,可吹牛的內(nèi)容都是我干的剿干。 我是一名探鬼主播,決...
    沈念sama閱讀 40,251評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼穆刻,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼置尔!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起氢伟,我...
    開封第一講書人閱讀 39,129評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤榜轿,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后朵锣,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體谬盐,經(jīng)...
    沈念sama閱讀 45,561評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,779評(píng)論 3 335
  • 正文 我和宋清朗相戀三年诚些,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了飞傀。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,902評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡诬烹,死狀恐怖砸烦,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情绞吁,我是刑警寧澤幢痘,帶...
    沈念sama閱讀 35,621評(píng)論 5 345
  • 正文 年R本政府宣布,位于F島的核電站掀泳,受9級(jí)特大地震影響雪隧,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜员舵,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,220評(píng)論 3 328
  • 文/蒙蒙 一脑沿、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧马僻,春花似錦庄拇、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽溶弟。三九已至,卻和暖如春瞭郑,著一層夾襖步出監(jiān)牢的瞬間辜御,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工屈张, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留擒权,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,025評(píng)論 2 370
  • 正文 我出身青樓阁谆,卻偏偏與公主長(zhǎng)得像碳抄,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子场绿,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,843評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • Docker — 云時(shí)代的程序分發(fā)方式 要說最近一年云計(jì)算業(yè)界有什么大事件剖效?Google Compute Engi...
    ahohoho閱讀 15,532評(píng)論 15 147
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)焰盗,斷路器璧尸,智...
    卡卡羅2017閱讀 134,656評(píng)論 18 139
  • 為什么一定要把想不開的事放下《耗現(xiàn)在多好啊,每一場(chǎng)夢(mèng)里梦湘,你都能來瞎颗。我突然覺得不那么難受了,唯一的小小心愿捌议,是你一定一...
    87的腿兒閱讀 200評(píng)論 0 0
  • 在進(jìn)行數(shù)據(jù)連接時(shí)候需要進(jìn)行網(wǎng)絡(luò)狀態(tài)判斷 //判斷網(wǎng)絡(luò)連接 + (BOOL) isConnectionAvailab...
    眼睛不笑的人閱讀 378評(píng)論 0 0
  • Weex頁面由<template>, , 三個(gè)部分構(gòu)成哼拔。1.<template>: 必須的, 使用類HTML的語法...
    阿凡提說AI閱讀 347評(píng)論 0 0