当前位置:   article > 正文

IOS 基于APNS消息推送原理与实现(JAVA后台)_java apns

java apns

Push的原理:

Push 的工作机制可以简单的概括为下图



 

图中,Provider是指某个iPhone软件的Push服务器,这篇文章我将使用.net作为Provider。

APNS 是Apple Push Notification Service(Apple Push服务器)的缩写,是苹果的服务器。

上图可以分为三个阶段。

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

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

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

 



 

从上图我们可以看到。

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

2、 IOS跟APNS Server要deviceToken。应用程序接受deviceToken。

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

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

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

无论是iPhone客户端跟APNS,还是Provider和APNS都需要通过证书进行连接的。下面介绍一下所用到证书的制作。

一、CSR文件

 

1、生成Certificate Signing Request(CSR)

  

 

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

 

 

点击继续:



 

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

 

 

二、SSL certificate文件

 

1、用你付过费的帐号登录到iOS Provisioning Portal,并创建Certificates(已创建可省略),如下图:



 

 

 

 点击Submit

 创建Certificate完毕。

2、新建一个App ID

 点击New App ID

输入Description,Bundle Identifier,点击Submit,新建App ID完毕。

找到新建的App ID 点击右侧的Configure:



 Development Push SSL Certificate ,与Production Push SSL Certificate 区别在于一个是用于开发的推送证书,一个是用于发布产品的推送证书。两个证书获取到的终端deviceToken是不一样的,用两个证书生成的P12证书用于JAVA后台连接APNS的服务器地址也是不同的,Development Push SSL Certificate 对应连接的服务器地址是:gateway.sandbox.push.apple.com。Production Push SSL Certificate  对应连接的服务器地址是:gateway.push.apple.com。

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

 点击Continue:




 

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



点击Continue:



 

 点击Download,下载生成的支持推送服务的证书(命名为:aps_development-6.cer)。

 

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

 

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

1、PushTest.certSigningRequest

2、aps_development-6.cer(下载生成的支持推送服务的证书。)

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

 

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

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

双击将其加入到xcode 的Provisioning Profiles 中。

四、生成JAVA后台用于连接APNS的证书:

打开钥匙串

 
选中Apple Development IOS Push Services:com.easecom.zhwgpushtestdesc,右键将其导出。

导出用于JAVA后台连接APNS的P12证书。

 输入p12 证书的密码,本文中我用的是123456。记住这个密码,JAVA后台使用p12证书的时候要用到。

 输入访问钥匙串的密码:系统登陆密码。

导出PushTest.p12证书完毕。

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

1、PushTest.certSigningRequest

2、aps_development-6.cer(下载生成的支持推送服务的证书。)

3、pushtestdescDevprofile.mobileprovision
4、PushTest.p12

至此IOS消息推送(JAVA后台)证书全部制作完毕。

下面开始上代码:

五、IOS端代码:

1、首先在项目的AppDelegate.m中加入以下两个代理方法

  1. - (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken {
  2. NSString *token = [NSString stringWithFormat:@"%@", deviceToken];
  3. //获取终端设备标识,这个标识需要通过接口发送到服务器端,服务器端推送消息到APNS时需要知道终端的标识,APNS通过注册的终端标识找到终端设备。
  4. NSLog(@"My token is:%@", token);
  5. }
  6. - (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error {
  7. NSString *error_str = [NSString stringWithFormat: @"%@", error];
  8. NSLog(@"Failed to get token, error:%@", error_str);
  9. }

2、在AppDelegate.m的(BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions方法中加入注册消息通知推送能力;加入当应用程序处于未启动状态时,判断是否由远程消息通知触发;加入清除消息推送通知标记。

  1. - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
  2. {
  3. //判断是否由远程消息通知触发应用程序启动
  4. if ([launchOptions objectForKey:UIApplicationLaunchOptionsRemoteNotificationKey]!=nil) {
  5. //获取应用程序消息通知标记数(即小红圈中的数字)
  6. int badge = [UIApplication sharedApplication].applicationIconBadgeNumber;
  7. if (badge>0) {
  8. //如果应用程序消息通知标记数(即小红圈中的数字)大于0,清除标记。
  9. badge--;
  10. //清除标记。清除小红圈中数字,小红圈中数字为0,小红圈才会消除。
  11. [UIApplication sharedApplication].applicationIconBadgeNumber = badge;
  12. }
  13. }
  14. //消息推送注册
  15. [[UIApplication sharedApplication] registerForRemoteNotificationTypes:UIRemoteNotificationTypeSound|UIRemoteNotificationTypeAlert|UIRemoteNotificationTypeBadge];
  16. }

3、在项目AppDelegate.m中加入消息接收处理代理方法。

  1. //处理收到的消息推送
  2. - (void)application:(UIApplication *)application
  3. didReceiveRemoteNotification:(NSDictionary *)userInfo
  4. {
  5. //在此处理接收到的消息。
  6. NSLog(@"Receive remote notification : %@",userInfo);
  7. }

六、JAVA后台代码:

  1. import javapns.notification.*;
  2. import org.apache.commons.lang3.StringUtils;
  3. import org.json.JSONException;
  4. import javapns.communication.exceptions.CommunicationException;
  5. import javapns.communication.exceptions.KeystoreException;
  6. import javapns.devices.Device;
  7. import javapns.devices.implementations.basic.BasicDevice;
  8. public class PushPNS {
  9. public static void iosPush(String token,String alert,String sound,String url,String password,Integer badge) {
  10. //参数依次是:deviceToken,alert,sound,p12证书地址,证书密码,badge
  11. try {
  12. //int badge=1;
  13. //String sound="defalt";
  14. //String url ="D:\\证书Development.p12";
  15. //String password = "xxx";
  16. PushNotificationPayload payload=new PushNotificationPayload();
  17. payload.addAlert(alert);
  18. payload.addBadge(badge);
  19. //payload.addCustomDictionary("remote_id", "11");
  20. //payload.addCustomDictionary("type", "video");
  21. if(!StringUtils.isEmpty(sound)) {
  22. payload.addSound(sound);
  23. }
  24. PushNotificationManager pushManager=new PushNotificationManager();
  25. pushManager.initializeConnection(new AppleNotificationServerBasicImpl(url, password, false));
  26. //List<PushedNotification> notifications=new ArrayList<PushedNotification>();
  27. Device device=new BasicDevice();
  28. device.setToken(token);
  29. PushedNotification notification=pushManager.sendNotification(device, payload,true);
  30. if(notification.isSuccessful()) {
  31. ResponsePacket theErrorResponse = notification.getResponse();
  32. if(theErrorResponse != null && theErrorResponse.isErrorResponsePacket()) {
  33. System.out.println("APNS server cant send message to IOS");
  34. pushManager.stopConnection();
  35. }
  36. pushManager.stopConnection();
  37. }else {
  38. pushManager.stopConnection();
  39. System.out.println("cannot send message to APNS server");
  40. }
  41. } catch (JSONException e) {
  42. e.printStackTrace();
  43. } catch (CommunicationException e) {
  44. e.printStackTrace();
  45. } catch (KeystoreException e) {
  46. e.printStackTrace();
  47. }
  48. }
  49. }

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/你好赵伟/article/detail/525909
推荐阅读
相关标签
  

闽ICP备14008679号