最近在使用xmppframwork來實現一個聊天應用,碰到了一個問題,應用進入後台以後,就接收不到訊息了;
怎麼樣才能使應用被切到後台時,應用中的網路連接仍然接收資料並維持存活(像QQ iphone 一樣),下面一步一步來分析。
針對iOS應用的狀態,存在以下三種情況:
1)若iOS應用為活動狀態,剛與伺服器保持一個長串連,用戶端與伺服器通過此串連收發訊息。
2)若iOS應用為退出狀態,長串連被斷開,伺服器向用戶端發訊息則通過APNS推送訊息實現。
3)若iOS應用剛剛切至後台,還沒有關閉,發現長串連並沒有斷開,伺服器通過長串連向用戶端發送訊息還是能發出去,但是iOS應用只有重新切至前台時才能收到訊息。
針對第三種情況,有下面的疑問:
iOS應用剛切至後台時,串連還保持著,此時用戶端在後台是否可以收伺服器來的訊息,然後用本地通知來通知使用者?如果可以的話應該怎麼做?還是說一旦切至後台,就必須通過APNS來通知使用者?
結合蘋果官方文檔App Programming Guide裡有關Background
Execution and Multitasking的章節和satckoverflow對xmpp支援後台socket問題的解答,得出以下解決方案:
1.因為蘋果只支援少數幾咱類型的應用的代碼可以在後台運行(例如:audio、location、voip等),所以需要在應用的(appname)-info.plist檔案裡,增加一個"Required
background modes"的key,值設定為voip,來指定類型。
2.ios xmppframwork裡已經有這方面的支援,只要初始化xmppstream時做下面設定即可:
//允許後台模式(注意ios模擬器上是不支援後台socket的) xmppStream.enableBackgroundingOnSocket = YES;
下面做一下測試,看看效果如何。
先在- (void)xmppStream:(XMPPStream *)sender didReceiveMessage:(XMPPMessage *)message方法裡加入下面代碼:
//程式運行在前台,訊息正常顯示 if ([[UIApplication sharedApplication] applicationState] == UIApplicationStateActive) { }else{//如果程式在後台運行,收到訊息以通知類型來顯示 UILocalNotification *localNotification = [[UILocalNotification alloc] init]; localNotification.alertAction = @"Ok"; localNotification.alertBody = [NSString stringWithFormat:@"From: %@\n\n%@",@"test",@"This is a test message"];//通知主體 localNotification.soundName = @"crunch.wav";//通知聲音 localNotification.applicationIconBadgeNumber = 1;//標記數 [[UIApplication sharedApplication] presentLocalNotificationNow:localNotification];//發送通知 }
真機測試,登入後切換到後台狀態,然後在電腦上用spark發一條訊息到iphone上,狀態列翻出通知訊息,基本成功實現了。
-------------------------------------------------------------------------------------------------------------------------------------------------
補充說明:
網上查資料時,發現一個問題,如果你的應用沒有實現voip,如果按上面這種方法做,有被蘋果reject的危險;那麼替代方法就是當應用進入後台或是已經退出後,伺服器端會根據使用者狀態的變化,發送訊息給使用者,同時使用蘋果apns來推送新的訊息通知給使用者;但我們知道,蘋果的apns在即時性和可靠性方面都是不做保證的,也就是說新訊息的通知傳到蘋果那邊,蘋果不保證能成功幫你推送,也不保證即時推送,所以只能找一下看有沒有第三方的免費推送服務提供者。
Google了一下,找到個叫極光推送的免費推送,正在研究中,有收穫在更新...