iOS 9 適配中一定會遇到的大坑合集
本文主要是說一些iOS9適配中出現(xiàn)的坑,如果只是要單純的了解iOS9新特性可以看瞄神的開發(fā)者所需要知道的 iOS 9 SDK 新特性。9月17日凌晨,蘋果給用戶推送了iOS9正式版,隨著有用戶陸續(xù)升級iOS9,也就逐漸的衍生出了一系列的問題,筆者也在趕忙為自己維護的App做適配,本文寫的一些坑基本都是親身體驗了。
一、NSAppTransportSecurity
iOS9讓所有的HTTP默認使用了HTTPS,原來的HTTP協(xié)議傳輸都改成TLS1.2協(xié)議進行傳輸。直接造成的情況就是App發(fā)請求的時候彈出網(wǎng)絡(luò)無法連接。解決辦法就是在項目的info.plist 文件里加上如下節(jié)點:
NSAppTransportSecurity - NSAllowsArbitraryLoads
這個子節(jié)點的意思是:是否允許任性的加載?! 設(shè)為YES的話就將禁用了AppTransportSecurity轉(zhuǎn)而使用用戶自定義的設(shè)置,這個問題就解決了。
上面說是蘋果限制了HTTP協(xié)議,但是也并不是說所有的HTTPS都能***適配iOS9了。
舉個栗子,從app內(nèi)起webView加載https的網(wǎng)頁。新建個項目寫幾行起網(wǎng)頁的代碼
- - (void)loadView{
- UIWebView *web = [[UIWebView alloc]initWithFrame:[UIScreen mainScreen].bounds];
- self.view = web;
- }
- - (void)viewDidLoad {
- [super viewDidLoad];
- UIWebView *web = (UIWebView *)self.view; //董鉑然
- NSURL *url = [NSURL URLWithString:@"https://github.com/"];
- NSURLRequest *request = [NSURLRequest requestWithURL:url];
- [web loadRequest:request];
- }
中間的url就是我們想要加載的https地址,用https://baidu.com/ 和 https://github.com/ 分別試一下,結(jié)果不同
github的網(wǎng)頁能打開,百度的網(wǎng)頁打不開,下面打印了一行l(wèi)og
- NSURLSession/NSURLConnection HTTP load failed (kCFStreamErrorDomainSSL, -9802)
原因是蘋果的官方資料說首先必須要基于TLS 1.2版本協(xié)議。然后證書的加密的算法還需要達到SHA256或者更高位的RSA密鑰或ECC密鑰,如果不符合,請求將被中斷并返回nil.
在瀏覽器中是可以直接查看這個網(wǎng)站的加密算法的,先點綠鎖再點證書信息。
從右邊兩張圖可以看出,github帶RSA加密的SHA-256符合蘋果的要求,所以才可以展示。
針對百度的情況可以在info.plist中配置如下,如果網(wǎng)站引用的比較多應(yīng)該是需要針對每個網(wǎng)站進行配置。
NSAppTransportSecurity,NSExceptionDomains,NSIncludesSubdomains,NSExceptionRequiresForwardSecrecy,NSExceptionAllowInsecureHTTPLoads 寫在下面便于復(fù)制。
其中的ForwardSecrecy理解為超前的密碼保護算法,在官方資料里有寫,一共是11種。配置完畢百度可以訪問。
二、Bitcode
bitcode的理解應(yīng)該是把程序編譯成的一種過渡代碼,然后蘋果再把這個過渡代碼編譯成可執(zhí)行的程序。bitcode也允許蘋果在后期重新優(yōu)化我們程序的二進制文件,有類似于App瘦身的思想。
用了xcode7的編譯器編譯之前沒問題的項目可能會出現(xiàn)下列報錯。
- XXXX’ does not contain bitcode. You must rebuild it with bitcode enabled (Xcode setting ENABLE_BITCODE), obtain an updated library from the vendor, or disable bitcode for this target. for architecture arm64
問題的原因是:某些第三方庫還不支持bitcode。要不然是等待庫的開發(fā)者升級了此項功能我們更新庫,要不就是把這個bitcode禁用。
禁用的方法就是找到如下配置,選為NO.(iOS中bitcode是默認YES,watchOS中bitcodes是不讓改的必須YES。)
三、設(shè)置信任
這一條只和企業(yè)級應(yīng)用或inhose 有關(guān),和AppStore渠道的應(yīng)用無關(guān)。
在iOS8只是彈出一個窗問你是否需要讓手機信任這個應(yīng)用,但是在iOS9卻直接禁止,如果真的想信任需要自己去手動開啟。類似于Mac系統(tǒng)從未知開發(fā)者處下載的dmg直接打不開,然后要到系統(tǒng)偏好設(shè)置的安全性與隱私手動打開。 下圖展示左邊iOS8,右邊iOS9
用戶需要去 設(shè)置---》通用---》描述文件 里面自行添加信任。
這種問題的處理方法也就兩種:1.提前周知暫時不要升級iOS9 2.大多是公司員工使用的企業(yè)級應(yīng)用,群發(fā)一個指導(dǎo)郵件。
四、字體
iOS8中,字體是Helvetica,中文的字體有點類似于“華文細黑”。只是蘋果手機自帶渲染,所以看上去可能比普通的華文細黑要美觀。iOS9中,中文系統(tǒng)字體變?yōu)榱藢橹袊O(shè)計的“蘋方” 有點類似于一種word字體“幼圓”。字體有輕微的加粗效果,并且最關(guān)鍵的是字體間隙變大了!
所以很多原本寫死了width的label可能會出現(xiàn)“...”的情況。
上面這兩張圖也可以直觀的看出同一個界面,同一個label的變化。
所以為了在界面顯示上不出錯,就算是固定長度的文字也還是建議使用sizetofit 或者ios向上取整 ceilf() 或者提前計算
- CGSize size = [title sizeWithAttributes:@{NSFontAttributeName: [UIFont systemFontOfSize:14.0f]}];
- CGSize adjustedSize = CGSizeMake(ceilf(size.width), ceilf(size.height));
五、URL scheme
URL scheme一般使用的場景是應(yīng)用程序有分享或跳其他平臺授權(quán)的功能,分享或授權(quán)后再跳回來。
在iOS8并沒有做過多限制,但是iOS9需要將你要在外部調(diào)用的URL scheme列為白名單,才可以完成跳轉(zhuǎn)
如果iOS9沒做適配 會報如下錯誤
- canOpenURL: failed for URL : "mqzone://qqapp" - error: "This app is not allowed to query for scheme mqzone"
具體的解決方案也是要在info.plist中設(shè)置 LSApplicationQueriesSchemes 類型為數(shù)組,下面添加所有你用到的scheme
六、statusbar
這個還好只是報一個警告,如果就是不管他,也不會出現(xiàn)問題。
- <Error>: CGContextSaveGState: invalid context 0x0. If you want to see the backtrace, please set CG_CONTEXT_SHOW_BACKTRACE environmental variable.
以前我們?yōu)榱四軌驅(qū)崟r的控制頂部statusbar的樣式,可能會在喜歡使用
- [[UIApplication sharedApplication] setStatusBarStyle:UIStatusBarStyleLightContent]
- [[UIApplication sharedApplication]setStatusBarHidden:YES];
但是這么做之前需要將 info.plist 里面加上View controller-based status bar appearance BOOL值設(shè)為NO,就是把控制器控制狀態(tài)欄的權(quán)限給禁了,用UIApplication來控制。但是這種做法在iOS9不建議使用了,建議我們使用吧那個BOOL值設(shè)為YES,然后用控制器的方法來管理狀態(tài)欄比如。
- - (UIStatusBarStyle)preferredStatusBarStyle
- {
- return UIStatusBarStyleLightContent;
- }
點進頭文件可以驗證剛才說法:
- @property(readwrite, nonatomic,getter=isStatusBarHidden) BOOL statusBarHidden NS_DEPRECATED_IOS(2_0, 9_0, "Use -[UIViewController prefersStatusBarHidden]");
七、didFinishLaunchingWithOptions
如果運行的時候報下列錯誤,那就是你的didFinishLaunchingWithOptions寫的不對了
- ***** Assertion failure in -[UIApplication _runWithMainScene:transitionContext:completion:], /BuildRoot/Library/Caches/com.apple.xbs/Sources/UIKit_Sim/UIKit-3505.16/UIApplication.m:3294**
iOS9不允許在didFinishLaunchingWithOptions結(jié)束了之后還沒有設(shè)置window的rootViewController。 也許是xcode7的編譯器本身就不支持。
解決的方法當(dāng)然就是先初始化個值,之后再賦值替換掉
- UIWindow *window = [[UIWindowalloc] initWithFrame:[UIScreenmainScreen].bounds];
- window.rootViewController = [[UIViewController alloc]init];
八、tableView
雖然現(xiàn)在的iOS9已經(jīng)推送正式版了,但是iOS9使用時還是會感覺到App比以前更加卡頓了,tableView拖動時卡頓顯示的最為明顯。 并且之前遇到一個bug,原本好的項目用xcode7一編譯,tableView刷新出了問題 ,[tableView reloadData]無效 有一行cell明明改變了但是刷新不出來。 感覺可能是這個方法和某種新加的特性沖突了,猜測可能是reloadData的操作被推遲到下一個RunLoop執(zhí)行最終失效。
解決的方法是,注釋[tableView reloadData],改用局部刷新,問題居然就解決了。
- [self.tableView reloadSections:[NSIndexSet indexSetWithIndex:0] withRowAnimation:UITableViewRowAnimationNone];
如果你不是在董鉑然博客園看到本文,請點擊查看原文。
暫時遇到這些問題,感覺iOS9的出現(xiàn)讓所有iOS開發(fā)都是菊花一緊,希望蘋果這種大刀闊斧做改變,特立獨行的風(fēng)格發(fā)展下去以后別和government 產(chǎn)生矛盾,然后公司倒閉 導(dǎo)致開發(fā)人員失業(yè),也許是我想多了。預(yù)祝所有的iOS都能及時的做好適配改完bug,下個版本一上線,所有問題都解決。