iOS 遠程推送

概念相關

1.什么是遠程推送通知?

  • 顧名思義,就是從遠程服務器推送給客戶端的通知(需要聯網)
  • 遠程推送服務,又稱為APNs(Apple Push Notification Services)

2.為什么需要遠程推送通知?

  • 傳統獲取數據的局限性:只要用戶關閉了app,就無法跟app的服務器溝通,無法從服務器上獲得最新的數據內容

  • 遠程推送通知可以解決以上問題:不管用戶打開還是關閉app,只要聯網了,都能接收到服務器推送的遠程通知

3.所有的蘋果設備,在聯網狀態下,都會與蘋果的服務器建立長連接

什么是長連接?
只要聯網了,就一直建立連接

長連接的作用:
時間校準
系統升級
查找我的iPhone

長連接的好處:
數據傳輸速度快
數據保持最新狀態

4.遠程推送原理
客戶端發送設備的UDID和程序的bundle ID請求蘋果服務器(SSL安全),客戶端獲得Token號存儲起來,客戶端再將Token號和用戶信息等(如QQ號等)綁定發送給公司服務器,公司服務器保存token號和賬戶的關聯信息,在適當時候,公司根據token號再通知蘋果服務器進行消息推送

遠程推送原理

準備工作

  1. 開發iOS程序的推送功能, iOS端需要做的事

  2. 請求蘋果獲得deviceToken

    UDID : 目的是將來可以找到手機
    Bundle ID : 目的是將來可以找到手機中的程序

  3. 得到蘋果返回的deviceToken

  4. 發送deviceToken給公司的服務器

  5. 監聽用戶對通知的點擊

  6. 調試iOS的遠程推送功能必備條件:真機、付費開發者賬號

  7. 調試推送需要的證書文件
    1> aps_development.cer : 推送測試證書,某臺電腦就能調試某個app的推送服務
    2> ios_development.cer : 調試證書,讓電腦具備真機調試的能力(調試設備)
    3> iphone5_qq.mobileprovision :描述文件, 某臺電腦就能利用某臺設備調試某個程序

  8. 發布具有推送服務的app
    1> aps_production.cer : 如果發布的程序中包含了推送服務,就必須安裝這個證書
    2> ios_distribution.cer : 讓電腦具備發布程序的能力
    3> qq.mobileprovision : 某臺電腦就能發布某個程序

記得推送證書要給后臺?。?!
  1. 如何創建推送證書?
    創建調試用的推送證書流程跟創建普通調試證書一樣, 多了一個選擇BundlD ID的過程 (如果之前配置的是通配符Bundld ID ,則無法使用Push功能)
    1.選擇推送證書


    選擇推送證書

    2.打開Bundle ID設置,確保push選項是enabled狀態,不是可點擊edit進行編輯


    保證是enabled狀態

3.或點擊編輯


點擊編輯

打勾后進行配置


打勾后進行配置

4.配置成功后鑰匙串中多了一個證書,一個調試,一個push
鑰匙串

代碼實現:(前提:確保bundleID和網站的配置一樣)

要注意,由于iOS8 以后推送需要用戶授權,所以AppDelegate中要分別適配不同版本

  1. 注冊推送,注冊后就會向蘋果服務器發送Token號
    iOS8和iOS7注冊通知對比:
  2. 多了一個授權的方法UIUserNotificationSettings
  3. 以前的方法中Remove換成了User


    注冊推送
  4. 注冊遠程推送完成后調用,該方法返回Token,一般在這個方法中將Token發給公司服務器作保存
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken;
打印Token號

注意:安裝程序之后,無論運行多少次,Token都不應該發送改變!
但是在Xcode7中這個選項打開和關閉的Token值不一樣!打開的時候才是真正的Token值!

注意在Xcode 7 之后必須打開推送的選項,否則可能會導致請求的token不一樣!
打開推送選項

代碼優化:
上面方法每次都需要請求Token,Token號只有第一次才需要請求,所以可以進行判斷第一次才需要請求Token
可以用一個字典包裝Token號,并存起來,下次讀取如果字典里有值就不需要再請求了
注意:如果客戶端更換了用戶信息,就需要重新請求Token,刪除本地信息重新請求,并刪除公司服務器端Token信息(也可不刪除添加一個),保證推送到新登錄的賬戶上

3.模擬服務器測試推送:

Easy APNs Provider 工具
PushMeBaby 工具

PushMeBaby 使用方法
(1)導入推送調試證書文件

導入推送證書

(2)更改 ApplicationDelegate 中init方法中的對應值


修改Token號

(3)運行,點擊推送


點擊推送

(4)推送成功程序右上角就會有一個1的角標
推送成功!

4.接收到通知后程序回調的代理方法

注意:要考慮三種情況,后臺、前臺、退出程序。遠程推送和本地推送一樣,都需要在兩個地方做代碼的處理:

  1. 接受到通知時調用的代理方法中(前臺和后臺)
  2. 啟動時的 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中,可用到界面把后臺模式更改一下


處理警告

勾選remote notifications

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);
}
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,321評論 6 543
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,559評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 178,442評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,835評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,581評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,922評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,931評論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,096評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,639評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,374評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,591評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,104評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,789評論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,196評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,524評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,322評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,554評論 2 379

推薦閱讀更多精彩內容