一聚教程网:一个值得你收藏的教程网站

最新下载

热门教程

php实现推送ios消息大数据量的实例

时间:2015-07-16 编辑:简简单单 来源:一聚教程网

公司要求app周五要推送push消息,deviceToken已经保存了近16w的数据,有点多,所以采用php进程执行推送。

涉及到进程,那么放到队列里面是最合适的,这里用的是Redis的list类型进行存储推送消息内容。

运营人员在后台编辑完推送内容之后,php会将所有deviceToken在mysql查询出来foreach存入Redis中(包括推送内容),然后编写一个php从Redis中取数据并发送的脚本(每次都建立苹果连接),php脚本每次开启运行5分钟、5分钟以后自动退出。然后编辑个shell脚本,循环执行php脚本20次(次数根据deviceToken多少进行增加),最后将shell脚本编辑到crontab中运行、至于运行时间设置为php脚本执行时间最好、这样就相当于不间断取数据推送。这样做的好处就是预防php超时

代码如下
php后台编辑内容入redis的代码我就不发了,发一下php脚本代码

ps'] = array(
                'alert' => $message,
                'sound' => 'default',
                'openType' => $info['iType'],
                'openId'    => strval($info['iPushid']),
                'link'        => $info['iPushid'],
            );
 
 
        $payload = json_encode($body);
        $msg = chr(0) . pack("n",32) . pack('H*', str_replace(' ', '', $info['token'])) . pack("n",strlen($payload)) . $payload;
        fwrite($fp, $msg);
        fclose($fp);
    }
 
    /**
    *获取IOS推送对象
    */
    public static function getIosObj($pass='123456'){
        $ctx = stream_context_create();
        $pem = dirname(__FILE__) .'/'.'push-dev2.pem';
        stream_context_set_option($ctx, 'ssl', 'local_cert', $pem);  
        stream_context_set_option($ctx, 'ssl', 'passphrase', $pass);
        $fp = stream_socket_client('ssl://gateway.push.apple.com:2195',$err,$errstr, 60, STREAM_CLIENT_CONNECT, $ctx);
        if(!$fp)
            return false;
        else
            return $fp;
    }
}
$pushKey = 'ios_push_user'; //Redis Key
$time    = 300;              //执行时间
 
$obj = new push;
MQTaskHelper::consume($pushKey,array($obj,'sendPush'),$time,['host'=>'127.0.0.1']);
    
MQTaskHelper.php 类
listPush($strKey,$mixMessage);
    }
 
    public static function consume($strKey,$strCallback,$iRunTime = 60,$arrOptions=array()){
 
        $oRedis = self::_getRedisObj($arrOptions);
 
        //开始时间
        $iStartTime = time();
 
        while(true){
 
            $iCurrentTime = time();
            if($iCurrentTime - $iStartTime >= $iRunTime){
                print_r("[".date('Y-m-d H:i:s')."] I worked from["
                    .date('Y-m-d H:i:s',$iStartTime)."] to ["
                    .date('Y-m-d H:i:s',$iCurrentTime)."]. Quiting!\n");
                exit;
            }
 
            $mixMessage = $oRedis->lPop($strKey);
 
            if(!$mixMessage){
                continue;
            }else{
 
                try{
                    call_user_func($strCallback, $mixMessage);
 
                }catch(Exception $ex){
                    print_r("mq_consume_exception:".$ex->getMessage()."\n");
                }
 
            }
        }
    }
 
 
    private static function _getRedisObj($arrOptions)
    {
        if(empty(self::$redisObj)){
            self::$redisObj = new Redis();
            self::$redisObj->connect($arrOptions['host'],6379);
        }
        return self::$redisObj;
    }
}




服务器push脚本设置

*/5 * * * * /bin/bash   /data/xxx/push/push.sh

push.sh

for i in {1..10}
do
/data/xxx/php/bin/php -f /data/xxx/push/push.php &
done



基本推送了15w条的时间在1小时左右吧、如果觉得慢的话,多开几个进程、或者多弄几台服务器都是可以的。





iOS实现推送消息并用PHP做推送服务端的方法


IOS推送消息是许多IOS应用都具备的功能,最近也在研究这个功能,参考了很多资料终于搞定了,下面就把步骤拿出来分享下:

iOS消息推送的工作机制可以简单的用下图来概括:


\

Provider是指某个iPhone软件的Push服务器,APNS是Apple Push Notification Service的缩写,是苹果的服务器。

上图可以分为三个阶段:

第一阶段:应用程序把要发送的消息、目的iPhone的标识打包,发给APNS。

第二阶段:APNS在自身的已注册Push服务的iPhone列表中,查找有相应标识的iPhone,并把消息发送到iPhone。

第三阶段:iPhone把发来的消息传递给相应的应用程序,并且按照设定弹出Push通知。


\

从上图我们可以看到:

1、应用程序注册消息推送。

2、iOS从APNS Server获取device token,应用程序接收device token。

3、应用程序将device token发送给PUSH服务端程序。

4、服务端程序向APNS服务发送消息。

5、APNS服务将消息发送给iPhone应用程序。

无论是iPhone客户端和APNS,还是Provider和APNS,都需要通过证书进行连接。

下面我介绍一下几种用到的证书。

一、CSR文件

1、生成Certificate Signing Request(CSR)


\

2、填写你的邮箱和常用名称,并选择保存到硬盘。


\

点击继续:


\

这样就在本地生成了一个Push.certSigningRequest文件。

二、p12文件

1、导出密钥。


\


\

2、输入你的密码。


\

这样就生成了一个Push.p12文件。

三、SSL certificate文件

1、用你付过费的帐号登录到iOS Provisioning Portal,并新建一个App ID,这个过程可以参考:iOS应用的真机调试,这样就会生成下面这条记录:


\

2、点击右侧的Configure:


\

3、点击Development Push SSL Certificate一行后的Configure:


\

4、点击Continue:


\

5、选择前面生成好的Push.certSigningRequest文件,点击Generate,出现如下所示的页面:


\

6、点击Continue:


\

7、点击Download,并将文件命名为aps_developer_identity.cer。

8、点击Done,你会发现状态变成了Enabled:


\

到现在为止,我们已经生成了三个文件:

1、Push.certSigningRequest

2、Push.p12

3、aps_developer_identity.cer

双击aps_developer_dientity.cer 注册到你的钥匙串中,这样你的钥匙串中就会有


\

二、准备profile证书,因为推送消息只能再真机上测试,所以要建一个profile证书


\

点击"new profile"为上面新建的APP ID建个profile ,成功之后下载*_Dev_Profile.mobileprovision

双击将其加入到xcode 的Provisioning Profiles 中,这里有一点要注意,再将这个加入xcode之前如果之前已经加入过一定要把之前加入的删掉,如果有多个的话会出错。

三、工程代码

到这里证书已经准备完毕,接下来,我们在xcode中新建一个测试工程,注意设置工程的Bundle Identifier必须与上面建的APP ID 里的相同


\

在didFinishLaunchingWithOptions 中加入一下代码

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
[self.window makeKeyAndVisible];
    [[UIApplication sharedApplication] registerForRemoteNotificationTypes: UIRemoteNotificationTypeBadge |UIRemoteNotificationTypeSound | UIRemoteNotificationTypeAlert];
    return YES;
}
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)pToken {
    NSLog(@"regisger success:%@", pToken);
    //注册成功,将deviceToken保存到应用服务器数据库中
}
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo{
    // 处理推送消息
    UIAlertView *alert=[[UIAlertView alloc]initWithTitle:@"通知" message:@"我的信息" delegate:selfcancelButtonTitle:@"取消" otherButtonTitles:nil, nil];
    [alert show];
    [alert release];
NSLog(@"%@", userInfo);
}
- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error {
    NSLog(@"Regist fail%@",error); 
}



到这里一切顺利的话我们就可以在真机运行了,注册成功我们会得到iphone 的deviceToken,

My token is:

<740f4707 bebcf74f 9b7c25d4 8e335894 5f6aa01d a5ddb387 462c7eaf 61bb78ad>

四、在应用服务器采用php的方式将消息推送给APNS,

1、php连接APNS也是需要证书的,还记得我们上面获得的几个证书吗?打开终端,对上面的证书做如下处理,

cd  进入证书所在目录

把.cer文件转换成.pem文件:

$ openssl x509 -in aps_developer_identity.cer -inform der

-out PushChatCert.pem

把私钥Push.p12文件转换成.pem文件:

$ openssl pkcs12 -nocerts -out PushChatKey.pem -in Push.p12

Enter Import Password:

MAC verified OK

Enter PEM pass phrase:

Verifying ? Enter PEM pass phrase:

你首先需要为.p12文件输入passphrase密码短语,这样OpenSSL可以读它。然后你需要键入一个新的密码短语来加密PEM文件。还是使用”pushchat”来作为PEM的密码短语。你需要选择一些更安全的密码短语。

注意:如果你没有键入一个PEM passphrase,OpenSSL将不会返回一个错误信息,但是产生的.pem文件里面将不会含有私钥。

最后。把私钥和证书整合到一个.pem文件里:

$ cat PushChatCert.pem PushChatKey.pem > ck.pem

为了测试证书是否工作,执行下面的命令:

$ telnet gateway.sandbox.push.apple.com 2195

Trying 17.172.232.226…

Connected to gateway.sandbox.push-apple.com.akadns.net.

Escape character is ‘^]’.

它将尝试发送一个规则的,不加密的连接到APNS服务。如果你看到上面的反馈,那说明你的MAC能够到达APNS。按下Ctrl+C 关闭连接。如果得到一个错误信息,那么你需要确保你的防火墙允许2195端口。

然后再次连接,这次用我们的SSL证书和私钥来设置一个安全的连接:

$ openssl s_client -connect gateway.sandbox.push.apple.com:2195

-cert PushChatCert.pem -key PushChatKey.pem

Enter pass phrase for PushChatKey.pem:

你会看到一个完整的输出,让你明白OpenSSL在后台做什么。如果连接是成功的,你可以键入一些字符。当你按下回车后,服务就会断开连接。如果在建立连接时有问题,OpenSSL将会给你一个错误消息,

ck.pem文件就是我们需要得到php连接APNS 的文件,将ck.pem和push.php放入同一目录上传到服务器,push.php的代码如下:

 $message,
'sound' => 'default'
);
// Encode the payload as JSON
$payload = json_encode($body);
// Build the binary notification
$msg = chr(0) . pack('n', 32) . pack('H*', $deviceToken) . pack('n', strlen($payload)) . $payload;
// Send it to the server
$result = fwrite($fp, $msg, strlen($msg));
if (!$result)
echo 'Message not delivered' . PHP_EOL;
else
echo 'Message successfully delivered' . PHP_EOL;
// Close the connection to the server
fclose($fp);
?>



接下来我们访问http://localhost/push/push.php

iphone就会接收到一条推送消息了,如果有问题的话就检查上面的操作步骤,特别是加红的部分

另外去除标记的方法为,在viewDidApper中加入

int badge = [UIApplication sharedApplication].applicationIconBadgeNumber;

if(badge > 0)
{
    badge--;
    [UIApplication sharedApplication].applicationIconBadgeNumber = badge;
}

热门栏目