php redis实现对200w用户的即时推送
发布:smiling 来源: PHP粉丝网 添加日期:2018-08-06 11:23:30 浏览: 评论:0
怎么实现对200w用户的即时推送,这个推送可以理解为调用第三方的接口,push,sms之类的东西。
当时先写了一个demo 直接读取DB然后单个推送,结果。。可想而知,于是设计一套基于redis+php多进程的方案,用着还不错而去扩展性蛮高的,故分享之。
具体的逻辑如下:(无视我的字体)
其实这里还可以优化的,我的设想是如果用户数据再多一些的话,可以在redis里对数据进行分割采取多List,每一个List对应多个php进程这样会更快。
下面是我实现的具体代码:
主管理脚本:应用时启动这个即可。
- <?php //push推送配置 注:使用前请确认log文件为空 2016-04-12
- <a href="/tags.php/include/" target="_blank">include</a>_once(dirname (__FILE__)."/../../config.inc.php");
- //if(exec('<a href="/fw/photo.html" target="_blank">ps</a> 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=newChannel($data);
- $redis=newRedisCache($Credis['host'],$Credis['port']);
- if(exec('ps aux | grep redis_push.php | grep -v grep | wc -l') != 0)gotocheck;//如果有推送任务 直接执行监控代码
- /*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;
- ***************************************************************\n
- monkey;
- 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'],Ƈ',$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 &");
- } //phpfensi.com
- 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原生框架统一管理,过段时间我发出来。
Tags: php redis php即时推送
相关文章
- ·PHP 是什么?(2013-11-12)
- ·Php.ini文件位置在哪里 Php.ini文件找不到(2013-11-12)
- ·PHP 数据类型(2013-11-12)
- ·php 获取当前脚本的url(2013-11-12)
- ·php技术生成静态页面的实现(2013-11-13)
- ·缺陷月项目启动 披露PHP脚本语言漏洞(2013-11-13)
- ·在PHP中全面阻止SQL注入式攻击(2013-11-13)
- ·php生成随机密码的几种方法(2013-11-13)
- ·PHP中使用FCKeditor2.3.2配置(2013-11-13)
- ·如何使用PHP开发高效的WEB系统(2013-11-13)
- ·php:树形结构的算法(2013-11-13)
- ·php4和php5区别(2013-11-13)
- ·php数据库连接(2013-11-13)
- ·如何正确理解PHP的错误信息(2013-11-13)
- ·php页面漏洞分析及相关问题解决(2013-11-13)
- ·当在连接PHP时,抱怨一些数值没有定义参考?(2013-11-27)
推荐文章
热门文章
最新评论文章
- 写给考虑创业的年轻程序员(10)
- PHP新手上路(一)(7)
- 惹恼程序员的十件事(5)
- PHP邮件发送例子,已测试成功(5)
- 致初学者:PHP比ASP优秀的七个理由(4)
- PHP会被淘汰吗?(4)
- PHP新手上路(四)(4)
- 如何去学习PHP?(2)
- 简单入门级php分页代码(2)
- php中邮箱email 电话等格式的验证(2)