php redis實現(xiàn)對200w用戶的即時推送
怎么實現(xiàn)對200w用戶的即時推送,這個推送可以理解為調(diào)用第三方的接口,push,sms之類的東西。
當(dāng)時先寫了一個demo 直接讀取DB然后單個推送,結(jié)果。??上攵?br />
于是設(shè)計一套基于redis+php多進程的方案,用著還不錯而去擴展性蠻高的,故分享之。
=============================================
具體的邏輯如下:(無視我的字體)

其實這里還可以優(yōu)化的,我的設(shè)想是如果用戶數(shù)據(jù)再多一些的話,可以在redis里對數(shù)據(jù)進行分割采取多List,每一個List對應(yīng)多個php進程這樣會更快。
下面是我實現(xiàn)的具體代碼:
主管理腳本:應(yīng)用時啟動這個即可。
<?php //push推送配置 注:使用前請確認log文件為空 2016-04-12
include_once(dirname (__FILE__)."/../../config.inc.php");
//if(exec('ps aux | grep redis_push.php | grep -v grep | wc -l') != 0) goto check;
import('push.class.php');
import('Redis.class.php');
$time =time();
$data = array("apikey"=>'xxxx',"secret"=>'xxxx');
$push = new Channel($data);
$redis = new RedisCache($Credis['host'],$Credis['port']);
if(exec('ps aux | grep redis_push.php | grep -v grep | wc -l') != 0) goto check;//如果有推送任務(wù) 直接執(zhí)行監(jiān)控代碼
/*PUSH配置項*/
$config = array(
"file"=>"test.txt",
"Title"=>"sssss",
"Content"=>"ssssssssssssssss",
"OpenType"=>"0", //1是 0否 是否跳轉(zhuǎn)鏈接
"Url"=>"", //鏈接地址
"num"=>"500", //每次推送條數(shù)
"s"=>"1" //睡眠時間 (單位:秒)
);
$num = 15; //啟動進程數(shù)量
$a = $config['OpenType']==1 ? "是" : "否";
$c = json_encode($config);
$info = <<<monkey
************ 請確認信息是否有誤*10秒后啟動push任務(wù)! *************
* 文件名稱 : {$config['file']};
* 推送標(biāo)題 : {$config['Title']};
* 推送內(nèi)容 : {$config['Content']};
* 是否跳轉(zhuǎn) : {$config['OpenType']};
* 進程數(shù)量 : $num;(如果為單進程無視此項)
* 睡眠時間 : {$config['s']};
* 日志目錄 : /log;
***************************************************************\n
monkey;
echo $info;
sleep(3);
$n = 1;
while($n<=10){
echo (10-$n++),"秒\n";
sleep(1);
}
echo "------------------------- 任務(wù)已啟動 -------------------------\n";
if($redis->Scount('push_getchannel_success')){
echo "隊列有未完成任務(wù)\n";
}else{
$res = exec("php redis_getchannel.php {$config['file']}");//寫入redis腳本
echo $res;
}
smtp_mail('xxxx@qq.com','推送任務(wù)已開啟','請實時監(jiān)測,5秒后您的手機將接收到測試推送!');//推送監(jiān)控 實現(xiàn)定時全自動推送
echo "\n---------------- 5秒后 test 將收到測試推送消息 ----------------\n";
sleep(5);
$re = $push->BaiduPush('xxxx','xxxxx',$config['Content'],$config['Title'],'1',$config['OpenType'],$config['Url'],'xxxxx',$push);
sleep(1);
echo "\n---------------- 測試推送已發(fā)出!如未收到,請及時終止程序! 10秒后正式推送!!! ----------------\n";
$m = 1;
while($m<=10){
echo (10-$m++),"秒\n";
sleep(1);
}
echo "\n---------------- 推送任務(wù)已經(jīng)開始!請耐心等待! ----------------\n";
//下面設(shè)置是否多進程
for($i=1;$i<=$num;$i++){
exec("php redis_push.php '{$c}' > /dev/null 2>&1 &");
}
check:
while(1){
if(exec('ps aux | grep redis_push.php | grep -v grep | wc -l') == 0){
echo "push 發(fā)送完成 用時",time()-$time,"秒";
die();
}
echo "當(dāng)前進程數(shù):",exec('ps aux | grep redis_push.php | grep -v grep | wc -l'),"個","\n";
echo "當(dāng)前剩余推送數(shù)量:".$redis->Scount('push_getchannel_success')."\n";
sleep(10);
}
至于寫入redis和具體的推送腳本這個靠自己的想象里就好了 我就不發(fā)了 嘿嘿
我的做法是具體的推送腳本在推送一定數(shù)量后會自動終止并調(diào)用自己本身。
因為在實際應(yīng)用中發(fā)現(xiàn)php腳本在長時間運行之后會發(fā)生假死(可能是因為上下文切換的問題),所以我都是避免讓php腳本長時間運行。
還有就是用戶肯定不是固定的200w用戶 每天都會有一個增量,我的方案是通過定時腳本每天把增量的用戶整理進我自己設(shè)計的一個用戶表自己管理。
ps:我把所有的腳本弄到了一個我自己整理的小的php原生框架統(tǒng)一管理,過段時間我發(fā)出來。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
CentOS 安裝 PHP5.5+Redis+XDebug+Nginx+MySQL全紀錄
這篇文章主要介紹了在CentOS系統(tǒng)環(huán)境下安裝 PHP5.5+Redis+XDebug+Nginx+MySQL開發(fā)環(huán)境的全過程,非常的細致詳盡,推薦給有需要的小伙伴們參考下吧。2015-03-03
Zend Framework動作助手Redirector用法實例詳解
這篇文章主要介紹了Zend Framework動作助手Redirector用法,結(jié)合實例形式詳細分析了轉(zhuǎn)向器Redirector的功能,使用方法與相關(guān)注意事項,需要的朋友可以參考下2016-03-03
php實現(xiàn)mysql數(shù)據(jù)庫備份類
備份程序一共只有兩個類:數(shù)據(jù)庫備份器(DbBak)和表備份器(TableBak): 這個程序僅僅備份和恢復(fù)數(shù)據(jù),使用方法很簡單,實例化DbBak,然后調(diào)用bakupDb和restoreDb方法:2008-03-03
php根據(jù)數(shù)據(jù)id自動生成編號的實現(xiàn)方法
下面小編就為大家?guī)硪黄猵hp根據(jù)數(shù)據(jù)id自動生成編號的實現(xiàn)方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-10-10
解析php中mysql_connect與mysql_pconncet的區(qū)別詳解
本篇文章是對php中mysql_connect與mysql_pconncet的區(qū)別進行了分析介紹,需要的朋友參考下2013-05-05
Codeigniter上傳圖片出現(xiàn)“You did not select a file to upload”錯誤解決辦法
這篇文章主要介紹了Codeigniter上傳圖片出現(xiàn)“You did not select a file to upload”的解決辦法,需要的朋友可以參考下2014-06-06

