怎么实现对200w用户的即时推送,这个推送可以理解为调用第三方的接口,push,sms之类的东西。

当时先写了一个demo 直接读取DB然后单个推送,结果。。。。可想而知

于是设计一套基于redis+php多进程的方案,用着还不错而去扩展性蛮高的,故分享之。

=============================================

具体的逻辑如下:(无视我的字体)

其实这里还可以优化的,我的设想是如果用户数据再多一些的话,可以在redis里对数据进行分割采取多List,每一个List对应多个php进程这样会更快。

下面是我实现的具体代码:

主管理脚本:应用时启动这个即可。

<?php          //push推送配置  注:使用前请确认log文件为空       2016-04-12include_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;//如果有推送任务 直接执行监控代码/*PUSH配置项*/$config = array(  "file"=>"test.txt",  "Title"=>"sssss",  "Content"=>"ssssssssssssssss",  "OpenType"=>"0",    //1是  0否    是否跳转链接  "Url"=>"",         //链接地址  "num"=>"500",      //每次推送条数  "s"=>"1"           //睡眠时间 (单位:秒));$num = 15;            //启动进程数量$a = $config['OpenType']==1 ? "是" : "否";$c = json_encode($config);$info = <<<monkey   ************ 请确认信息是否有误*10秒后启动push任务! *************   * 文件名称   : {$config['file']};   * 推送标题   : {$config['Title']};   * 推送内容   : {$config['Content']};   * 是否跳转   : {$config['OpenType']};   * 进程数量   : $num;(如果为单进程无视此项)   * 睡眠时间   : {$config['s']};   * 日志目录   : /log;   ***************************************************************\nmonkey;echo $info;sleep(3);$n = 1;while($n<=10){  echo (10-$n++),"秒\n";  sleep(1);}echo "------------------------- 任务已启动 -------------------------\n";if($redis->Scount('push_getchannel_success')){  echo "队列有未完成任务\n";}else{  $res = exec("php redis_getchannel.php {$config['file']}");//写入redis脚本  echo $res;}smtp_mail('xxxx@qq.com','推送任务已开启','请实时监测,5秒后您的手机将接收到测试推送!');//推送监控 实现定时全自动推送 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---------------- 测试推送已发出!如未收到,请及时终止程序! 10秒后正式推送!!! ----------------\n";$m = 1;while($m<=10){  echo (10-$m++),"秒\n";  sleep(1);}echo "\n---------------- 推送任务已经开始!请耐心等待! ----------------\n";//下面设置是否多进程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 发送完成 用时",time()-$time,"秒";    die();  }  echo "当前进程数:",exec('ps aux | grep redis_push.php | grep -v grep | wc -l'),"个","\n";  echo "当前剩余推送数量:".$redis->Scount('push_getchannel_success')."\n";  sleep(10);}

至于写入redis和具体的推送脚本这个靠自己的想象里就好了 我就不发了 嘿嘿

我的做法是具体的推送脚本在推送一定数量后会自动终止并调用自己本身。

因为在实际应用中发现php脚本在长时间运行之后会发生假死(可能是因为上下文切换的问题),

所以我都是避免让php脚本长时间运行。

还有就是用户肯定不是固定的200w用户 每天都会有一个增量,我的方案是通过定时脚本每天把增量的用户整理进我自己设计的一个用户表自己管理。

ps:我把所有的脚本弄到了一个我自己整理的小的php原生框架统一管理,过段时间我发出来。

欢迎指正,谢谢。

更多相关文章

  1. PHP模拟supervisor的进程管理
  2. PHP多进程、信号量及孤儿进程和僵尸进程
  3. php脚本的后缀名是什么
  4. PHP传递数组格式参数到shell脚本中
  5. PHP 实现守护进程

随机推荐

  1. Google放弃“不做恶”? 意欲垄断Android
  2. Android零基础入门第33节:Android事件处理
  3. Android与H5互调详细介绍
  4. Windows下获取Android系统源码
  5. Android 扫码盒子全局接收付款码
  6. 安卓大佬力荐,送你一份超详细的Android学
  7. Android(安卓)studio 快捷键,解决返回上次
  8. Android的多任务之路
  9. 谷歌Google仍需提高Android的五个方面
  10. Android采用HTML设计软件界面