您的位置:首页 > 移动开发 > IOS开发

iOS开发 - ANPs推送通知

2015-05-03 14:12 295 查看

推送通知

注意:这里说的推送通知跟NSNotification有所区别

NSNotification是抽象的,不可见的

推送通知是可见的(能用肉眼看到)

iOS中提供了2种推送通知

本地推送通知(Local Notification)

远程推送通知(Remote Notification)

推送通知的呈现效果总结

总结一下,推送通知有5种不同的呈现效果

在屏幕顶部显示一块横幅(显示具体内容)

在屏幕中间弹出一个UIAlertView(显示具体内容)

在锁屏界面显示一块横幅(锁屏状态下,显示具体内容)

更新app图标的数字(说明新内容的数量)

播放音效(提醒作用)

发出推送通知时,如果当前程序正运行在前台,那么推送通知就不会被呈现出来

点击推送通知后,默认会自动打开发出推送通知的app

不管app打开还是关闭,推送通知都能如期发出

本地推送通知

什么是本地推送通知

顾名思义,就是不需要联网就能发出的推送通知(不需要服务器的支持)

本地推送通知的使用场景

常用来定时提醒用户完成一些任务,比如

清理垃圾、记账、买衣服、看电影、玩游戏

如何发出本地推送通知

//创建本地推送通知对象
UILocalNotification *ln = [[UILocalNotification alloc] init];

//设置本地推送通知属性
//推送通知的触发时间(何时发出推送通知)
@property(nonatomic,copy) NSDate *fireDate;
//推送通知的具体内容
@property(nonatomic,copy) NSString *alertBody;
//在锁屏时显示的动作标题(完整标题:“滑动来” + alertAction)
@property(nonatomic,copy) NSString *alertAction;
//音效文件名
@property(nonatomic,copy) NSString *soundName;
//app图标数字
@property(nonatomic) NSInteger applicationIconBadgeNumber;


//调度本地推送通知(调度完毕后,推送通知会在特地时间fireDate发出)
[[UIApplication sharedApplication] scheduleLocalNotification:ln];

//获得被调度(定制)的所有本地推送通知
@property(nonatomic,copy) NSArray *scheduledLocalNotifications;
(已经发出且过期的推送通知就算调度结束,会自动从这个数组中移除)

//取消调度本地推送通知
- (void)cancelLocalNotification:(UILocalNotification *)notification;
- (void)cancelAllLocalNotifications;

//立即发出本地推送通知
- (void)presentLocalNotificationNow:(UILocalNotification *)notification;


本地推送通知的其他属性

//每隔多久重复发一次推送通知
@property(nonatomic) NSCalendarUnit repeatInterval;

//点击推送通知打开app时显示的启动图片
@property(nonatomic,copy) NSString *alertLaunchImage;

//附加的额外信息
@property(nonatomic,copy) NSDictionary *userInfo;

//时区
@property(nonatomic,copy) NSTimeZone *timeZone;
(一般设置为[NSTimeZone defaultTimeZone] ,跟随手机的时区)


点击本地推送通知

//当用户点击本地推送通知,会自动打开app,这里有2种情况
//app并没有关闭,一直隐藏在后台
//让app进入前台,并会调用AppDelegate的下面方法(并非重新启动app)
- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification;

//app已经被关闭(进程已死)
//启动app,启动完毕会调用AppDelegate的下面方法
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions;
launchOptions参数通过UIApplicationLaunchOptionsLocalNotificationKey取出本地推送通知对象


远程推送通知

什么是远程推送通知

顾名思义,就是从远程服务器推送给客户端的通知(需要联网)

远程推送服务,又称为APNs(Apple Push Notification Services)

为什么需要远程推送通知?

传统获取数据的局限性

只要用户关闭了app,就无法跟app的服务器沟通,无法从服务器上获得最新的数据内容

远程推送通知可以解决以上问题

不管用户打开还是关闭app,只要联网了,都能接收到服务器推送的远程通知

远程推送通知使用须知

所有的苹果设备,在联网状态下,都会与苹果的服务器建立长连接

什么是长连接

只要联网了,就一直建立连接

长连接的作用

时间校准

系统升级

查找我的iPhone

.. …

长连接的好处

数据传输速度快

数据保持最新状态

一.开发iOS程序的推送功能, iOS端需要做的事

1.请求苹果获得deviceToken

2.得到苹果返回的deviceToken

3.发送deviceToken给公司的服务器

4.监听用户对通知的点击

二.调试iOS的远程推送功能, 必备条件:

1.真机

2.调试推送需要的证书文件

1> aps_development.cer : 某台电脑就能调试某个app的推送服务

2> ios_development.cer : 让电脑具备真机调试的能力(调试设备)

3> iphone5_qq.mobileprovision : 某台电脑就能利用某台设备调试某个程序

三.发布具有推送服务的app

1> aps_production.cer : 如果发布的程序中包含了推送服务,就必须安装这个证书

2> ios_distribution.cer : 让电脑具备发布程序的能力

3> qq.mobileprovision : 某台电脑就能发布某个程序

注册远程通知

//客户端如果想接收APNs的远程推送通知,必须先注册(得到用户的授权)
//一般在App启动完毕后就马上注册
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
    // 注册远程通知
       UIRemoteNotificationType type = UIRemoteNotificationTypeAlert | UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound;
    [application registerForRemoteNotificationTypes:type];
    return YES;
}


//注册成功后会调用AppDelegate的下面方法,得到设备的deviceToken
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
    NSLog(@"%@", deviceToken);
}


//当用户点击远程推送通知,会自动打开app,这里有2种情况
//app并没有关闭,一直隐藏在后台
//让app进入前台,并会调用AppDelegate的下面方法(并非重新启动app)
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo;

//app已经被关闭(进程已死)
//启动app,启动完毕会调用AppDelegate的下面方法
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions;
launchOptions参数通过UIApplicationLaunchOptionsRemoteNotificationKey取出服务器返回的字典内容


PushMeBaby

PushMeBaby是一款用来测试ANPs的开源Mac项目

它充当了服务器的作用,用法非常简单

它负责将内容提交给苹果的APNs服务器,苹果的APNs服务器再将内容推送给用户的设备

PushMeBaby的主页

https://github.com/stefanhafeneger/PushMeBaby

远程推送获取DeviceToken实例

@interface AppDelegate ()

@end

@implementation AppDelegate

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.

    if ([UIDevice currentDevice].systemVersion.doubleValue <= 8.0) {
        // 不是iOS8
        UIRemoteNotificationType type = UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound | UIRemoteNotificationTypeAlert;
        // 当用户第一次启动程序时就获取deviceToke
        // 该方法在iOS8以及过期了
        // 只要调用该方法, 系统就会自动发送UDID和当前程序的Bunle ID到苹果的APNs服务器
        [application registerForRemoteNotificationTypes:type];
    }else
    {
        // iOS8
        UIUserNotificationType type = UIUserNotificationTypeBadge | UIUserNotificationTypeAlert | UIUserNotificationTypeSound;

        UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:type categories:nil];
        // 注册通知类型
        [application registerUserNotificationSettings:settings];

        // 申请试用通知
        [application registerForRemoteNotifications];
    }

    // 1.取出数据
    NSDictionary *userInfo = launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey];

    if (userInfo) {
        static int count = 0;
        count++;
        UILabel *label = [[UILabel alloc] init];
        label.frame = CGRectMake(0, 40, 200, 200);
        label.numberOfLines = 0;
        label.textColor = [UIColor whiteColor];
        label.font = [UIFont systemFontOfSize:11];
        label.backgroundColor = [UIColor orangeColor];
        label.text = [NSString stringWithFormat:@" %@ \n %d", userInfo, count];
        [self.window.rootViewController.view addSubview:label];
    }

    return YES;
}

/**
 *  获取到用户对应当前应用程序的deviceToken时就会调用
 */
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
    NSLog(@"%@", deviceToken);
    // <47e58207 31340f18 ed83ba54 f999641a 3d68bc7b f3e2db29 953188ec 7d0cecfb>
    // <286c3bde 0bd3b122 68be655f 25ed2702 38e31cec 9d54da9f 1c62325a 93be801e>
}

/*
 ios7以前苹果支持多任务, iOS7以前的多任务是假的多任务
 而iOS7开始苹果才真正的推出了多任务
 */
// 接收到远程服务器推送过来的内容就会调用
// 注意: 只有应用程序是打开状态(前台/后台), 才会调用该方法
/// 如果应用程序是关闭状态会调用didFinishLaunchingWithOptions
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo
{
    /*
     如果应用程序在后台 , 只有用户点击了通知之后才会调用
     如果应用程序在前台, 会直接调用该方法
     即便应用程序关闭也可以接收到远程通知
     */
    NSLog(@"%@", userInfo);

//    static int count = 0;
//    count++;
//    UILabel *label = [[UILabel alloc] init];
//    label.frame = CGRectMake(0, 250, 200, 200);
//    label.numberOfLines = 0;
//    label.textColor = [UIColor whiteColor];
//    label.text = [NSString stringWithFormat:@"%@ \n %d", userInfo, count];
//    label.font = [UIFont systemFontOfSize:11];
//    label.backgroundColor = [UIColor grayColor];
//    [self.window.rootViewController.view addSubview:label];
}

//接收到远程服务器推送过来的内容就会调用
// ios7以后用这个处理后台任务接收到得远程通知
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
{
    /*
     UIBackgroundFetchResultNewData, 成功接收到数据
     UIBackgroundFetchResultNoData, 没有;接收到数据
     UIBackgroundFetchResultFailed 接收失败

     */
//    NSLog(@"%s",__func__);
//    NSLog(@"%@", userInfo);

    NSNumber *contentid =  userInfo[@"content-id"];
    if (contentid) {
        UILabel *label = [[UILabel alloc] init];
        label.frame = CGRectMake(0, 250, 200, 200);
        label.numberOfLines = 0;
        label.textColor = [UIColor whiteColor];
        label.text = [NSString stringWithFormat:@"%@", contentid];
        label.font = [UIFont systemFontOfSize:30];
        label.backgroundColor = [UIColor grayColor];
        [self.window.rootViewController.view addSubview:label];
        //注意: 在此方法中一定要调用这个调用block, 告诉系统是否处理成功.
        // 以便于系统在后台更新UI等操作
        completionHandler(UIBackgroundFetchResultNewData);
    }else
    {
        completionHandler(UIBackgroundFetchResultFailed);
    }

}
@end
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: