PHP默認使用文件存儲session,如果并發量大,效率非常低。而Redis對高并發的支持非常好,所以,可以使用redis替代文件存儲session。
### php的session_set_save_handler 函數的作用和使用方法。
該函數定義用戶級session保存函數(如打開、關閉、寫入等)。原型如下:
`bool session_set_save_hanler(callback open,callback close,callback read,callback write,callback destory,callback gc)`
session_set_save_handler 函數各參數作用如下表
open 當session打開時調用此函數。接收兩個參數,第一個參數是保持session的路徑,第二個參數是session的名字
close 當session操作完成時調用此函數。不接收參數。
read 以session ID作為參數。通過session ID從數據存儲方中取得數據,并返回此數據。如果數據為空,可以返回一個空字符串。此函數在調用session_start 前被觸發
write 當數據存儲時調用。有兩個參數,一個是session ID,另外一個是session的數據
destroy 當調用session_destroy 函數時觸發destroy函數。只有一個參數 session ID
gc 當php執行session垃圾回收機制時觸發
在使用該函數前,先把php.ini配置文件的session.save_handler選項設置為user,否則session_set_save_handle 不會生效。
編寫一個session管理類sessionManager.php ,代碼如下:
~~~
<?php
class SessionManager{
private $redis;
private $sessionSavePath;
private $sessionName;
private $sessionExpireTime=30;//redis,session的過期時間為30s
public function __construct(){
$this->redis = new Redis();//創建phpredis實例
$this->redis->connect('127.0.0.1',6379);//連接redis
$this->redis->auth("107lab");//授權
$retval = session_set_save_handler(
array($this,"open"),
array($this,"close"),
array($this,"read"),
array($this,"write"),
array($this,"destroy"),
array($this,"gc")
);
session_start();
}
public function open($path,$name){
return true;
}
public function close(){
return true;
}
public function read($id){
$value = $this->redis->get($id);//獲取redis中的指定記錄
if($value){
return $value;
}else{
return '';
}
}
public function write($id,$data){
if($this->redis->set($id,$data)){//以session ID為鍵,存儲
$this->redis->expire($id,$this->sessionExpireTime);//設置redis中數據的過期時間,即session的過期時間
return true;
}
return false;
}
public function destroy($id){
if($this->redis->delete($id)){//刪除redis中的指定記錄
return true;
}
return false;
}
public function gc($maxlifetime){
return true;
}
public function __destruct(){
session_write_close();
}
}
~~~
SessionManager構造函數主要用來連接Redis服務器,使用session_set_save_handler函數設置session回調函數,并調用session_start函數開啟session功能。因為本例中open、close和gc回調函數的作用不是很大,所以直接返回true。
在write回調函數中,以session ID 作為key,把session的數據作為value存儲到redis服務器,設置session的過期時間為30秒。在read 回調函中,以session ID 作為key從redis服務器中讀取數據,并返回此數據。而在destroy回調函數重,則以session ID 作為key 從redis服務器中刪除對應的session數據。
使用時,只需包含SessionManager類,然后實例化一個SessionManager對象。下面建立個session_set.php文件。輸入代碼
~~~
<?php
include('SessionManager.php');
new SessionManager();
$_SESSION['username'] = 'captain';
~~~
然后再創建一個session_get.php文件,輸入如下代碼:
~~~
<?php
include('SessionManager.php');
new SessionManager();
echo $_SESSION['username'];
~~~
測試時,首先訪問session_set.php,然后再訪問 session_get.php,輸出結果如下所示:
再查看redis數據庫,如下所示
~~~
127.0.0.1:6379> keys *
1) "oe94eic337slnjv1bvlreoa574"
127.0.0.1:6379> get oe94eic337slnjv1bvlreoa574
"username|s:7:\"captain\";"
~~~
## 安裝redis服務以及php redis擴展
### 一:redis安裝
Download, extract and compile Redis with:
$ wget http://download.redis.io/releases/redis-3.0.4.tar.gz
$ tar xzf redis-3.0.4.tar.gz
$ cd redis-3.0.4
$ make
The binaries that are now compiled are available in the src directory. Run Redis with:
$ src/redis-server
You can interact with Redis using the built-in client:
$ src/redis-cli
redis> set foo bar
OK
redis> get foo
"bar"
More:http://www.redis.io/download
### 二、php擴展:
更多版本:http://pecl.php.net/package/redis
wget http://pecl.php.net/get/redis-2.2.5.tgz
#解壓
tar zxvf redis-2.2.5.tgz
#進入安裝目錄
cd redis-2.2.5
/usr/local/php/bin/phpize
#配置
./configure --with-php-config=/usr/local/php/bin/php-config
#編譯安裝
make && make install
安裝完成之后,出現下面的安裝路徑
/usr/local/php/lib/php/extensions/no-debug-non-zts-20090626/
### 配置php支持
#編輯配置文件php.ini,
vim /usr/local/php/etc/php.ini
在最后一行添加以下內容
extension="redis.so"
這個時候phpinfo()就可以看到redis擴展了。
### redis小例子:
~~~
$redis = new redis();
$test=$redis->connect('127.0.0.1', 6379);
var_dump($test);
$result = $redis->set('test',"111cn.net");
var_dump($result); //結果:bool(true)
$result = $redis->get('test');
var_dump($result); //結果:string(11) "111cn.net"
~~~
### php.ini的設置(所有都用redis存儲session)
~~~
session.save_handler = Redis
session.save_path = “tcp://192.168.5.114:6379”
//Redis不需要密碼驗證
session.save_path = “tcp://192.168.5.114:6379?auth=password”
//Redis 需要密碼驗證
~~~
修改后重啟php-fpm,phpinfo()可以查看到session存儲在redis中。
### 代碼設置(可以靈活使用)
~~~
//不修改php.ini,也可以在代碼中加入:
ini_set("session.save_handler","redis");
ini_set("session.save_path","tcp://127.0.0.1:6379");
ini_set("session.save_path",'tcp://192.168.5.114:6379?auth=password')
~~~
如果配置文件 /etc/redis.conf 里設置了連接密碼 requirepass,保存 session 的時候會報錯,save_path 這樣寫 tcp://127.0.0.1:6379?auth=authpwd 即可
如果選擇redis數據庫
<span style="color:red">session.save_path = "tcp://xx.xx.xx.xx:6789?database=11"</span>
,諸如此類。
### 查看redis存儲session的值
~~~
<?php
session_start();
$_SESSION['sessionid'] = 'www.111cn.net';
$redis = new redis();
$redis->connect('127.0.0.1', 6379);
//redis用 PHPREDIS_SESSION: 加 session_id 作為key,并且是以string的形式存儲
echo $redis->get('PHPREDIS_SESSION:' . session_id()); //輸出www.111cn.net
?>
~~~
在網上看到有人質疑redis存儲session值會有并發一致性的問題(文件存儲session有文件鎖來處理),這塊沒有研究。
- 簡介
- Cookie
- HTML5 LocalStorage
- session
- 當瀏覽器關閉后,Session就銷毀了嗎?
- mysql數據庫保存session
- HTTP協議的由來
- fsockopen異步請求
- http防盜鏈
- Apache偽靜態知識補充
- 大并發量解決方案
- 大型網站是怎樣解決多用戶高并發訪問
- 網站高并發 大流量訪問的處理及解決方法
- 并發數與在線客戶數?注冊用戶數的關系
- 即時聊天程序
- 反向Ajax實現
- ob緩存作用
- 淺聊并發之戰
- php擴展安裝
- php安裝redis擴展
- SQLMap自動化實施SQL注入共計
- 命名空間namespace
- 集群和分布式之【session共享】
- php Redis存儲Session 【1】
- php Redis存儲Session 【2】
- php mysql存儲session【1】
- php緩存
- 文件緩存
- memcache和redis的比較
- 原生session與session in redis對比
- XSS攻擊【1】
- XSS攻擊【2】
- PHP消息隊列
- php+mysql 模擬發送郵件隊列
- php+mysql 模擬訂單處理隊列
- php+redis 模擬秒殺隊列
- RabbitMQ 消息隊列系統
- beanstalkd
- PHP構建即時通訊
- WebSocket協議
- workerman
- PHP變量的作用域
- PHP傳值和傳引用的區別
- PHP匿名函數
- PHP遞歸函數&應用
- PHP單例模式
- PHP性能優化
- RESTful
- 集群
- 增加pgsql擴展
- php.ini路徑查找
- Swoole Compiler
- mysql 主從
- 主從
- mysql-proxy
- window docker環境