概念相關
1.什么是遠程推送通知?
- 顧名思義,就是從遠程服務器推送給客戶端的通知(需要聯網)
- 遠程推送服務,又稱為APNs(Apple Push Notification Services)
2.為什么需要遠程推送通知?
傳統獲取數據的局限性:只要用戶關閉了app,就無法跟app的服務器溝通,無法從服務器上獲得最新的數據內容
遠程推送通知可以解決以上問題:不管用戶打開還是關閉app,只要聯網了,都能接收到服務器推送的遠程通知
3.所有的蘋果設備,在聯網狀態下,都會與蘋果的服務器建立長連接
什么是長連接?
只要聯網了,就一直建立連接
長連接的作用:
時間校準
系統升級
查找我的iPhone
長連接的好處:
數據傳輸速度快
數據保持最新狀態
4.遠程推送原理
客戶端發送設備的UDID和程序的bundle ID請求蘋果服務器(SSL安全),客戶端獲得Token號存儲起來,客戶端再將Token號和用戶信息等(如QQ號等)綁定發送給公司服務器,公司服務器保存token號和賬戶的關聯信息,在適當時候,公司根據token號再通知蘋果服務器進行消息推送
準備工作
開發iOS程序的推送功能, iOS端需要做的事
-
請求蘋果獲得deviceToken
UDID : 目的是將來可以找到手機
Bundle ID : 目的是將來可以找到手機中的程序 得到蘋果返回的deviceToken
發送deviceToken給公司的服務器
監聽用戶對通知的點擊
調試iOS的遠程推送功能必備條件:真機、付費開發者賬號
調試推送需要的證書文件
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 : 某臺電腦就能發布某個程序
記得推送證書要給后臺?。?!
-
如何創建推送證書?
創建調試用的推送證書流程跟創建普通調試證書一樣, 多了一個選擇BundlD ID的過程 (如果之前配置的是通配符Bundld ID ,則無法使用Push功能)
1.選擇推送證書
選擇推送證書
2.打開Bundle ID設置,確保push選項是enabled狀態,不是可點擊edit進行編輯
保證是enabled狀態
3.或點擊編輯
打勾后進行配置
4.配置成功后鑰匙串中多了一個證書,一個調試,一個push
代碼實現:(前提:確保bundleID和網站的配置一樣)
要注意,由于iOS8 以后推送需要用戶授權,所以AppDelegate中要分別適配不同版本
- 注冊推送,注冊后就會向蘋果服務器發送Token號
iOS8和iOS7注冊通知對比: - 多了一個授權的方法UIUserNotificationSettings
-
以前的方法中Remove換成了User
注冊推送 - 注冊遠程推送完成后調用,該方法返回Token,一般在這個方法中將Token發給公司服務器作保存
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken;
注意:安裝程序之后,無論運行多少次,Token都不應該發送改變!
但是在Xcode7中這個選項打開和關閉的Token值不一樣!打開的時候才是真正的Token值!
注意在Xcode 7 之后必須打開推送的選項,否則可能會導致請求的token不一樣!
代碼優化:
上面方法每次都需要請求Token,Token號只有第一次才需要請求,所以可以進行判斷第一次才需要請求Token
可以用一個字典包裝Token號,并存起來,下次讀取如果字典里有值就不需要再請求了
注意:如果客戶端更換了用戶信息,就需要重新請求Token,刪除本地信息重新請求,并刪除公司服務器端Token信息(也可不刪除添加一個),保證推送到新登錄的賬戶上
3.模擬服務器測試推送:
Easy APNs Provider 工具
PushMeBaby 工具
PushMeBaby 使用方法
(1)導入推送調試證書文件
(2)更改 ApplicationDelegate 中init方法中的對應值
(3)運行,點擊推送
(4)推送成功程序右上角就會有一個1的角標
4.接收到通知后程序回調的代理方法
注意:要考慮三種情況,后臺、前臺、退出程序。遠程推送和本地推送一樣,都需要在兩個地方做代碼的處理:
- 接受到通知時調用的代理方法中(前臺和后臺)
- 啟動時的
didFinishLaunchingWithOptions
方法中(退出狀態),用launchOptions [UIApplicationLaunchOptionsRemoteNotificationKey]
獲取遠程通知對象
(1)前臺和后臺的推送回調這個代理方法(退出的設置在didFinishLaunchingWithOptions方法中)
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo;
(2)前臺、后臺和退出的推送都會調用這個代理方法(iOS 7之后可用)
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult result))completionHandler;
注意:
(1)如果實現了該方法會導致上面方法失效
(2)會有如下警告:在方法中調用下handler就好了,注意該handler需要一個參數
completionHandler(UIBackgroundFetchResultNewData);
(3)還有警告,需要添加一個值在info.plist中,可用到界面把后臺模式更改一下
AppDelegate中的全部代碼
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// 設置要注冊的通知類型,iOS8以后配置授權
if ([UIDevice currentDevice].systemVersion.floatValue >= 8.0) {
UIUserNotificationType type = UIUserNotificationTypeBadge | UIUserNotificationTypeSound | UIUserNotificationTypeAlert;
UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:type categories:nil];
[[UIApplication sharedApplication] registerUserNotificationSettings:settings];
[[UIApplication sharedApplication] registerForRemoteNotifications];
} else {
// 如果是iOS8以前,設置要注冊的通知類型
UIRemoteNotificationType type = UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound | UIRemoteNotificationTypeAlert;
[[UIApplication sharedApplication] registerForRemoteNotificationTypes:type];
}
// iOS7之前,如退出程序后接收到推送,想要處理獲取通知后的事件要在下面代碼中
if (launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey]) {
// 獲取到通知對象, 進行處理
NSDictionary *userInfo = launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey][@"userInfo"];
// 退出程序測試方法,真正接收到了通知就在界面上創建一個紅色的View(控制臺無法打印)
UILabel *label = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, 375, 200)];
label.text = userInfo.description;
label.numberOfLines = 0;
[self.window.rootViewController.view addSubview:label];
}
return YES;
}
#pragma mark 遠程推送注冊完畢, 服務器返回Token時, 調用此方法
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken {
// 將來要在這里將Token 發送給自己的服務器做保存
NSLog(@"deviceToken: %@", deviceToken);
}
#pragma mark 接收到遠程推送的消息時調用此方法(后臺和前臺時可用)
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo {
// 記得要判斷用戶是否在前臺
if (application.applicationState == UIApplicationStateActive) {
NSLog(@"不要自動跳轉/ 給用戶提示");
return ;
}
// 將來需要取消角標的數字, 是根據用戶是否做了某些操作, 來更改數字角標的值
// 獲取推送的值
NSInteger count = [userInfo[@"aps"][@"badge"] intValue];
// 設置相關的屬性
application.applicationIconBadgeNumber = count;
}
#pragma mark 接收到遠程推送的消息時調用此方法(前、后、退出都可用,iOS7以后可用)
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler {
// 測試添加一個label表示接收到通知
UILabel *label = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, 375, 200)];
label.text = userInfo.description;
label.numberOfLines = 0;
[self.window.rootViewController.view addSubview:label];
// 不調用該block會報錯
completionHandler(UIBackgroundFetchResultNewData);
}