国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 語言 > PHP > 正文

PHP使用Redis長連接的方法詳解

2024-05-05 00:02:21
字體:
來源:轉載
供稿:網友

本文實例講述了PHP使用Redis長連接的方法。分享給大家供大家參考,具體如下:

php-redis在github上的項目地址:https://github.com/phpredis/phpredis

pconnect函數聲明

其中time_out表示客戶端閑置多少秒后,就斷開連接。函數連接成功返回true,失敗返回false:

pconnect(host, port, time_out, persistent_id, retry_interval)    host: string. can be a host, or the path to a unix domain socket    port: int, optional    timeout: float, value in seconds (optional, default is 0 meaning unlimited)    persistent_id: string. identity for the requested persistent connection    retry_interval: int, value in milliseconds (optional)

下面的例子詳細介紹了pconnect連接的重用情況。

$redis->pconnect('127.0.0.1', 6379);$redis->pconnect('127.0.0.1'); // 默認端口6379,跟上面的例子使用相同的連接。$redis->pconnect('127.0.0.1', 6379, 2.5); // 設置了2.5秒的過期時間。將是不同于上面的新連接$redis->pconnect('127.0.0.1', 6379, 2.5, 'x'); //設置了持久連接的id,將是不同于上面的新連接$redis->pconnect('/tmp/redis.sock'); // unix domain socket - would be another connection than the four before.

pconnect使用介紹

對pconnect方法簡單描述

使用該方法創建連接,連接不會在調用close方法之后關閉,只有在進程結束之后該連接才會被關閉。

[待驗證]如果使用的是長連接,Redis配置文件中的timeout配置項需要設置為0,否則連接池中的連接會因為超時而失效

針對PHP-FPM來說明一下pconnect

長連接只會在PHP-FPM進程結束之后結束,連接的生命周期就是PHP-FPM進程的生命周期。
相比較短連接而言,在每一個PHP-FPM調用過程中都會產生一個redis的連接,在服務器上的表性形式就是過多的time_out連接狀態。
而長連接相反,PHP-FPM調用的所有CGI都只會共用一個長連接,所以也就是只會產生固定數量的time_out。

關閉長連接

可以調用close和unset方法,但兩則差異很大:

- close的作用僅僅是使當前PHP進程不能再進行redis請求,但無法真正關閉redis長連接,連接在后續請求中仍然會被重用,直FPM進程生命周期結束。所以close 并不會銷毀redis對象,只是斷開連接而已。

- unset 變量才會銷毀。也需要注意并不是使用了 pconnect 就不要 close 了,如果當前腳本執行時間很長 那么也會一直占用一個連接的。

如何判斷當前Redis是否處于連接狀態

等效的問題是,在單例模式中,判斷當前實例是否有效。

習慣上調用echo,判斷是否正常返回字符串本身,或者調用ping,查看返回值是否為 +PONG。

但是需要特別小心的是,在redis斷開連接之后,調用echo以及ping(返回'+POMG')時,均會拋出異常。所以要通過異常捕獲機制來處理。

代碼分析pconnect連接重用的問題

情況一:非單例模式。

說明:a實例和b實例共用了一條連接,b實例將a實例的連接修改了:
所以下面的例子導致最終$a實例得到的值變成了2,需要特別注意。

$a = pconnect(host, port, time_out);select(3);$a -> setex(id, 3);echo $a -> get(id);//之后執行下面的連接$b = pconnect(host, port, time_out);select(2);$b->set(id,2)echo $a->get(id);  //這個id操作的db變成了2,不再是之前的3了。因為這兩個連接共用了一個連接通道。

情況二:單例模式。

將上述的代碼修改,a和b都通過getInstance來生成。生成的前提是判斷當前實例是否存在。單例模式的混淆點在于:

$a生成了一個實例,這時候生成$b, $b使用了$a的實例,然后修改了$a的連接,之后調用$a肯定是調用的$b修改之后的實例。跟情況二一致。
單例模式的代碼如下:

public static function getInstance($db = 0){  if (!isset(self::$_instance)) {    self::$_instance = new Redis();  }  self::_connect();  self::$_instance->select($db);  return self::$_instance;}

兩種情況都說明了連接重用的問題。如何修復這個bug?兩點:

1.為每一個db生成一個單例。
2.避免連接重用問題。

所以代碼可以做調整為返回一個單例數組:

public static function getInstance($db = 0){  try{    if (isset(self::$_instance[$db]) && self::$_instance[$db]->Ping() == 'Pong') {      return self::$_instance[$db];    }  } catch (Exception $e) {  }  self::$_instance[$db] = new Redis();  self::_connect($db);  return self::$_instance[$db];}

需要注意的地方

避免在Task類成員變量中使用redis對象。

在redis的單例模式中,聲明了time_out的過期時間。如果redis處理的場合是一個任務,而任務調用redis間隔時間又比較長。當間隔大于time_out時候,redis就會斷開連接,這時候所有對redis的操作都會失效。解決的辦法就是避免這種調用方式,通過在調用的地方動態聲明redis類來執行。這種問題對于長連接和短鏈接是沒有區分,屬于調用的方式錯誤。

希望本文所述對大家PHP程序設計有所幫助。


注:相關教程知識閱讀請移步到PHP教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

主站蜘蛛池模板: 德保县| 驻马店市| 垣曲县| 丽水市| 镇康县| 漳州市| 稻城县| 濮阳县| 佳木斯市| 武乡县| 遵义县| 汉源县| 军事| 安顺市| 惠州市| 拉萨市| 扎鲁特旗| 长岭县| 上饶县| 红河县| 伊吾县| 视频| 都兰县| 中西区| 平湖市| 昌宁县| 卢湾区| 沈丘县| 弋阳县| 饶河县| 赤壁市| 资中县| 五河县| 修武县| 丽水市| 民权县| 定陶县| 郯城县| 长汀县| 界首市| 白银市|