iOS本地通知:UILocalNotification

首先是添加一个本地通知到系统中,代码如下:

  1. // 初始化本地通知对象  

  2. UILocalNotification *notification = [[UILocalNotification alloc] init];  

  3. if (notification) {  

  4.     // 设置通知的提醒时间  

  5.     NSDate *currentDate   = [NSDate date];  

  6.     notification.timeZone = [NSTimeZone defaultTimeZone]; // 使用本地时区  

  7.     notification.fireDate = [currentDate dateByAddingTimeInterval:5.0];  

  8.       

  9.     // 设置重复间隔  

  10.     notification.repeatInterval = kCFCalendarUnitDay;  

  11.       

  12.     // 设置提醒的文字内容  

  13.     notification.alertBody   = @"Wake up, man";  

  14.     notification.alertAction = NSLocalizedString(@"起床了", nil);  

  15.       

  16.     // 通知提示音 使用默认的  

  17.     notification.soundName= UILocalNotificationDefaultSoundName;  

  18.       

  19.     // 设置应用程序右上角的提醒个数  

  20.     notification.applicationIconBadgeNumber++;  

  21.       

  22.     // 设定通知的userInfo,用来标识该通知  

  23.     NSMutableDictionary *aUserInfo = [[NSMutableDictionary alloc] init];  

  24.     aUserInfo[kLocalNotificationID] = @"LocalNotificationID";  

  25.     notification.userInfo = aUserInfo;  

  26.       

  27.     // 将通知添加到系统中  

  28.     [[UIApplication sharedApplication] scheduleLocalNotification:notification];  

  29. }  


上面的alertBody是设备收到本地通知时横额或锁屏时的主要文字内容,alertActions是锁屏时显示的slide to后面的文字内容。例如:

iOS本地通知:UILocalNotification_第1张图片iOS本地通知:UILocalNotification_第2张图片


repeatInterval表示通知的重复间隔,在SDK中定义如下:

[objc] view plaincopy在CODE上查看代码片

  1. @property(nonatomic) NSCalendarUnit repeatInterval;      // 0 means don't repeat  


其取值主要有:

[objc] view plaincopy在CODE上查看代码片

  1. NSCalendarUnitEra                = kCFCalendarUnitEra,  

  2. NSCalendarUnitYear               = kCFCalendarUnitYear,  

  3. NSCalendarUnitMonth              = kCFCalendarUnitMonth,  

  4. NSCalendarUnitDay                = kCFCalendarUnitDay,  

  5. NSCalendarUnitHour               = kCFCalendarUnitHour,  

  6. NSCalendarUnitMinute             = kCFCalendarUnitMinute,  

  7. NSCalendarUnitSecond             = kCFCalendarUnitSecond,  

  8. NSCalendarUnitWeekday            = kCFCalendarUnitWeekday,  

  9. NSCalendarUnitWeekdayOrdinal     = kCFCalendarUnitWeekdayOrdinal,  


分别表示一个世纪、一年、一个月等等,0表示不重复。具体可以查看CFCalendar Reference

repeatInterval的下限应该是NSCalendarUnitMinute,即每分钟重复发送一次通知。

如果设置为NSCalendarUnitSecond,那么消息不会重复,每秒发送一次通知,iOS系统当然不会容许这样的存在了。

这里比较不好的一点是该值不能自定义(很遗憾,NSCalendarUnit是个枚举类型),例如你不能塞个10.0给它从而希望它每十秒重复一次。所以如果你想每20分钟发送一次通知,一小时内发送3次,那么只能同时设定三个通知了。



上面的代码运行后,5秒钟之后就可以收到一个本地通知。

在收到通知后,调用程序委托中的下列方法处理:

[objc] view plaincopy在CODE上查看代码片

  1. -(void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification{  

  2.     NSLog(@"Application did receive local notifications");  

  3.       

  4.     UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Hello" message:@"welcome" delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil, nil nil];  

  5.     [alert show];  

  6. }  


注意这个方法只有在程序启动之后才会执行,因此当程序处于后台时,该方法不会执行。


有一点需要注意,如果我们的应用程序给系统发送的本地通知是周期性的,那么即使把程序删了重装,之前的本地通知在重装时依然存在(没有从系统中移除)。例如,我们在viewDidLoad方法中启动添加本地通知的方法,多跑几次,然后把程序在模拟器中删除,再重新跑,并用下列方法输出所有的本地通知:

[objc] view plaincopy在CODE上查看代码片

  1. NSArray *localNotifications = [[UIApplication sharedApplication] scheduledLocalNotifications];  

  2. NSLog(@"%@", localNotifications);  


控制台输出:

[plain] view plaincopy在CODE上查看代码片

  1. 2014-03-14 15:46:37.145 LocalNotificationDemo[4419:60b] (  

  2.     "<UIConcreteLocalNotification: 0xa32ce30>{fire date = Friday, March 14, 2014 at 3:38:16 PM China Standard Time, time zone = Asia/Chongqing (GMT+8) offset 28800, repeat interval = NSDayCalendarUnit, repeat count = UILocalNotificationInfiniteRepeatCount, next fire date = Saturday, March 15, 2014 at 3:38:16 PM China Standard Time, user info = {\n    ClockID = LocalNotificationID;\n}}",  

  3.     "<UIConcreteLocalNotification: 0xa32dfc0>{fire date = Friday, March 14, 2014 at 3:44:45 PM China Standard Time, time zone = Asia/Chongqing (GMT+8) offset 28800, repeat interval = NSDayCalendarUnit, repeat count = UILocalNotificationInfiniteRepeatCount, next fire date = Saturday, March 15, 2014 at 3:44:45 PM China Standard Time, user info = {\n    ClockID = LocalNotificationID;\n}}",  

  4.     "<UIConcreteLocalNotification: 0xa32e470>{fire date = Friday, March 14, 2014 at 3:44:55 PM China Standard Time, time zone = Asia/Chongqing (GMT+8) offset 28800, repeat interval = NSDayCalendarUnit, repeat count = UILocalNotificationInfiniteRepeatCount, next fire date = Saturday, March 15, 2014 at 3:44:55 PM China Standard Time, user info = {\n    ClockID = LocalNotificationID;\n}}",  

  5.     "<UIConcreteLocalNotification: 0xa32e950>{fire date = Friday, March 14, 2014 at 3:45:13 PM China Standard Time, time zone = Asia/Chongqing (GMT+8) offset 28800, repeat interval = NSDayCalendarUnit, repeat count = UILocalNotificationInfiniteRepeatCount, next fire date = Saturday, March 15, 2014 at 3:45:13 PM China Standard Time, user info = {\n    ClockID = LocalNotificationID;\n}}",  

  6.     "<UIConcreteLocalNotification: 0xa32edb0>{fire date = Friday, March 14, 2014 at 3:45:29 PM China Standard Time, time zone = Asia/Chongqing (GMT+8) offset 28800, repeat interval = NSDayCalendarUnit, repeat count = UILocalNotificationInfiniteRepeatCount, next fire date = Saturday, March 15, 2014 at 3:45:29 PM China Standard Time, user info = {\n    ClockID = LocalNotificationID;\n}}",  

  7.     "<UIConcreteLocalNotification: 0xa32e870>{fire date = Friday, March 14, 2014 at 3:46:28 PM China Standard Time, time zone = Asia/Chongqing (GMT+8) offset 28800, repeat interval = NSDayCalendarUnit, repeat count = UILocalNotificationInfiniteRepeatCount, next fire date = Saturday, March 15, 2014 at 3:46:28 PM China Standard Time, user info = {\n    ClockID = LocalNotificationID;\n}}"  

  8. )  


可以看到之前发送的本地通知一直滞留在系统中。

不只是模拟器,在iOS设备上也是这样,博主之前的App在设备上重装时以前的本地通知会继续发送。

因此我们需要取消通知的方法,当然该对象也会在scheduledLocalNotifications数组中移除。

取消方法分为两种。

第一种比较暴力,直接取消所有的本地通知:

[objc] view plaincopy在CODE上查看代码片

  1. [[UIApplication sharedApplication] cancelAllLocalNotifications];  


这个适合在App重装时第一次启动的时候,或还原程序默认设置等场合下使用。

第二种方法是针对某个特定通知的:

[objc] view plaincopy在CODE上查看代码片

  1. - (void)cancelLocalNotification:(UILocalNotification *)notification NS_AVAILABLE_IOS(4_0);  


这时就需要通知有一个标识,这样我们才能定位是哪一个通知。可以在notification的userInfo(一个字典)中指定。

例如:

[objc] view plaincopy在CODE上查看代码片

  1. -(void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification{  

  2.     NSLog(@"Application did receive local notifications");  

  3.       

  4.     // 取消某个特定的本地通知  

  5.     for (UILocalNotification *noti in [[UIApplication sharedApplication] scheduledLocalNotifications]) {  

  6.         NSString *notiID = noti.userInfo[kLocalNotificationID];  

  7.         NSString *receiveNotiID = notification.userInfo[kLocalNotificationID];  

  8.         if ([notiID isEqualToString:receiveNotiID]) {  

  9.             [[UIApplication sharedApplication] cancelLocalNotification:notification];  

  10.             return;  

  11.         }  

  12.     }  

  13.       

  14.     UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Hello" message:@"welcome" delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil, nil nil];  

  15.     [alert show];  

  16. }  



最后建议本地通知不要发得太频繁,不然用户会觉得非常的烦。



参考的文章和资料:

IOS本地通知

CFCalendar Reference

iOS: 枚举类型 enum,NS_ENUM,NS_OPTIONS

你可能感兴趣的:(iOS本地通知:UILocalNotification)