iOS 10 的适配问题

wangllw 5年前
   <p>随着iOS10发布的临近,大家的App都需要适配iOS10,下面是我总结的一些关于iOS10适配方面的问题,如果有错误,欢迎指出.</p>    <h3><strong>1.系统判断方法失效:</strong></h3>    <p>在你的项目中,当需要判断系统版本的话,不要使用下面的方法:</p>    <pre>  <code class="language-objectivec">#define isiOS10 ([[[[UIDevice currentDevice] systemVersion] substringToIndex:1] intValue]>=10)  </code></pre>    <p>它会永远返回 NO , substringToIndex:1 在iOS 10 会被检测成 iOS 1了,</p>    <p>应该使用下面的这些方法:</p>    <p>Objective-C 中这样写:</p>    <pre>  <code class="language-objectivec">#define SYSTEM_VERSION_EQUAL_TO(v) ([[[UIDevice currentDevice] systemVersion] compare:v options:NSNumericSearch] == NSOrderedSame)  #define SYSTEM_VERSION_GREATER_THAN(v) ([[[UIDevice currentDevice] systemVersion] compare:v options:NSNumericSearch] == NSOrderedDescending)  #define SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(v) ([[[UIDevice currentDevice] systemVersion] compare:v options:NSNumericSearch] != NSOrderedAscending)  #define SYSTEM_VERSION_LESS_THAN(v) ([[[UIDevice currentDevice] systemVersion] compare:v options:NSNumericSearch] == NSOrderedAscending)  #define SYSTEM_VERSION_LESS_THAN_OR_EQUAL_TO(v) ([[[UIDevice currentDevice] systemVersion] compare:v options:NSNumericSearch] != NSOrderedDescending)  </code></pre>    <p>或者使用:</p>    <pre>  <code class="language-objectivec">if ([[NSProcessInfo processInfo]isOperatingSystemAtLeastVersion:(NSOperatingSystemVersion){.majorVersion = 9, .minorVersion = 1, .patchVersion = 0}]) { NSLog(@"Hello from > iOS 9.1");}  if ([NSProcessInfo.processInfoisOperatingSystemAtLeastVersion:(NSOperatingSystemVersion){9,3,0}]) { NSLog(@"Hello from > iOS 9.3");}  </code></pre>    <p>或者使用:</p>    <pre>  <code class="language-objectivec">if (NSFoundationVersionNumber > NSFoundationVersionNumber_iOS_9_0) { // do stuff for iOS 9 and newer} else { // do stuff for older versions than iOS 9}  </code></pre>    <p>有时候会缺少一些常量, NSFoundationVersionNumber 是在 NSObjCRuntime.h 中定义的,作为Xcode7.3.1的一部分,我们设定常熟范围从iPhone OS 2到 #define NSFoundationVersionNumber_iOS_8_4 1144.17 ,在iOS 10(Xcode 8)中,苹果补充了缺少的数字,设置有未来的版本.</p>    <pre>  <code class="language-objectivec">#define NSFoundationVersionNumber_iOS_9_0 1240.1  #define NSFoundationVersionNumber_iOS_9_1 1241.14  #define NSFoundationVersionNumber_iOS_9_2 1242.12  #define NSFoundationVersionNumber_iOS_9_3 1242.12  #define NSFoundationVersionNumber_iOS_9_4 1280.25  #define NSFoundationVersionNumber_iOS_9_x_Max 1299  </code></pre>    <p>Swift中这样写:</p>    <pre>  <code class="language-objectivec">if NSProcessInfo().isOperatingSystemAtLeastVersion(NSOperatingSystemVersion(majorVersion: 10,minorVersion: 0,patchVersion: 0)) {           // 代码块  }  </code></pre>    <p>或者使用</p>    <pre>  <code class="language-objectivec">if #available(iOS 10.0, *) {          // 代码块  } else {           // 代码块  }  </code></pre>    <h3><strong>2.隐私数据访问问题:</strong></h3>    <p>你的项目中访问了隐私数据,比如:相机,相册,联系人等,在Xcode8中打开编译的话,统统会crash,控制台会输出下面这样的日志:</p>    <p><img src="https://simg.open-open.com/show/046148cdcc9e3a869b2bf8b616632bab.png"></p>    <p>Snip20160905_1.png</p>    <p>这是因为iOS对用户的安全和隐私的增强,在申请很多私有权限的时候都需要添加描述,但是,在使用Xcode 8之前的Xcode还是使用系统的权限通知框.</p>    <p>要想解决这个问题,只需要在 info.plist 添加 NSContactsUsageDescription 的key, value自己随意填写就可以,这里列举出对应的key(Source Code模式下):</p>    <pre>  <code class="language-objectivec"><!-- 相册 -->   <key>NSPhotoLibraryUsageDescription</key>   <string>App需要您的同意,才能访问相册</string>   <!-- 相机 -->   <key>NSCameraUsageDescription</key>   <string>App需要您的同意,才能访问相机</string>   <!-- 麦克风 -->   <key>NSMicrophoneUsageDescription</key>   <string>App需要您的同意,才能访问麦克风</string>   <!-- 位置 -->   <key>NSLocationUsageDescription</key>   <string>App需要您的同意,才能访问位置</string>   <!-- 在使用期间访问位置 -->   <key>NSLocationWhenInUseUsageDescription</key>   <string>App需要您的同意,才能在使用期间访问位置</string>   <!-- 始终访问位置 -->   <key>NSLocationAlwaysUsageDescription</key>   <string>App需要您的同意,才能始终访问位置</string>   <!-- 日历 -->   <key>NSCalendarsUsageDescription</key>   <string>App需要您的同意,才能访问日历</string>   <!-- 提醒事项 -->   <key>NSRemindersUsageDescription</key>   <string>App需要您的同意,才能访问提醒事项</string>   <!-- 运动与健身 -->   <key>NSMotionUsageDescription</key> <string>App需要您的同意,才能访问运动与健身</string>   <!-- 健康更新 -->   <key>NSHealthUpdateUsageDescription</key>   <string>App需要您的同意,才能访问健康更新 </string>   <!-- 健康分享 -->   <key>NSHealthShareUsageDescription</key>   <string>App需要您的同意,才能访问健康分享</string>   <!-- 蓝牙 -->   <key>NSBluetoothPeripheralUsageDescription</key>   <string>App需要您的同意,才能访问蓝牙</string>   <!-- 媒体资料库 -->   <key>NSAppleMusicUsageDescription</key>   <string>App需要您的同意,才能访问媒体资料库</string  </code></pre>    <p>如果不起作用,可以请求后台权限,类似于这样:</p>    <pre>  <code class="language-objectivec"><key>UIBackgroundModes</key>  <array>   <!-- 在这里写上你在后台模式下要使用权限对应的key -->   <string>location</string>  ...  </array>  </code></pre>    <p>或者在Xcode里选中当前的 target ,选择 Capabilities ,找到 Background Modes ,打开它,在里面选择对应权限</p>    <p style="text-align: center;"><img src="https://simg.open-open.com/show/2d4add9e468823fe1c0406b0f689c1c5.png"></p>    <p>后台模式的操作.png</p>    <h3><strong>3.UIColor的问题</strong></h3>    <p>官方文档中说:大多数 core 开头的图形框架和 AVFoundation 都提高了对扩展像素和宽色域色彩空间的支持.通过图形堆栈扩展这种方式比以往支持广色域的显示设备更加容易。现在对UIKit扩展可以在sRGB的色彩空间下工作,性能更好,也可以在更广泛的色域来搭配sRGB颜色.如果你的项目中是通过低级别的api自己实现图形处理的,建议使用sRGB,也就是说在项目中使用了RGB转化颜色的建议转换为使用sRGB,在 UIColor 类中新增了两个api:</p>    <pre>  <code class="language-objectivec">- (UIColor *)initWithDisplayP3Red:(CGFloat)displayP3Redgreen:(CGFloat)greenblue:(CGFloat)bluealpha:(CGFloat)alphaNS_AVAILABLE_IOS(10_0);  + (UIColor *)colorWithDisplayP3Red:(CGFloat)displayP3Redgreen:(CGFloat)greenblue:(CGFloat)bluealpha:(CGFloat)alphaNS_AVAILABLE_IOS(10_0);  </code></pre>    <h3><strong>4.真彩色的显示</strong></h3>    <p>真彩色的显示会根据光感应器来自动的调节达到特定环境下显示与性能的平衡效果,如果需要这个功能的话,可以在 info.plist 里配置(在Source Code模式下):</p>    <pre>  <code class="language-objectivec">UIWhitePointAdaptivityStyle  </code></pre>    <p>它有五种取值,分别是:</p>    <pre>  <code class="language-objectivec">UIWhitePointAdaptivityStyleStandard // 标准模式  UIWhitePointAdaptivityStyleReading // 阅读模式  UIWhitePointAdaptivityStylePhoto // 图片模式  UIWhitePointAdaptivityStyleVideo // 视频模式  UIWhitePointAdaptivityStyleStandard // 游戏模式  </code></pre>    <p>也就是说如果你的项目是阅读类的,就选择 UIWhitePointAdaptivityStyleReading 这个模式,五种模式的显示效果是从上往下递减,也就是说如果你的项目是图片处理类的,你选择的是阅读模式,给选择太好的效果会影响性能.</p>    <h3><strong>5.ATS的问题</strong></h3>    <p>1.在iOS 9的时候,默认非HTTS的网络是被禁止的,我们可以在 info.plist 文件中添加 NSAppTransportSecurity 字典,将 NSAllowsArbitraryLoads 设置为 YES 来禁用ATS;</p>    <p>2.从2017年1月1日起,,所有新提交的 app 默认不允许使用 NSAllowsArbitraryLoads 来绕过ATS的限制,默认情况下你的 app 可以访问加密足够强的(TLS V1.2以上)HTTPS内容;</p>    <p>3.可以选择使用 NSExceptionDomains 设置白名单的方式对特定的域名开放HTTP内容来通过审核,比如说你的应用集成了第三方的登录分享SDK,可以通过这种方式来做,下面以新浪SDK作为示范(Source Code 模式下):</p>    <pre>  <code class="language-objectivec"> NSAppTransportSecurityNSExceptionDomainssina.cnNSThirdPartyExceptionMinimumTLSVersionTLSv1.0NSThirdPartyExceptionRequiresForwardSecrecyNSIncludesSubdomainsweibo.cnNSThirdPartyExceptionMinimumTLSVersionTLSv1.0NSThirdPartyExceptionRequiresForwardSecrecyNSIncludesSubdomainsweibo. comNSThirdPartyExceptionMinimumTLSVersionTLSv1.0NSThirdPartyExceptionRequiresForwardSecrecyNSIncludesSubdomainssinaimg.cnNSThirdPartyExceptionMinimumTLSVersionTLSv1.0NSThirdPartyExceptionRequiresForwardSecrecyNSIncludesSubdomainssinajs.cnNSThirdPartyExceptionMinimumTLSVersionTLSv1.0NSThirdPartyExceptionRequiresForwardSecrecyNSIncludesSubdomainssina.com.cnNSThirdPartyExceptionMinimumTLSVersionTLSv1.0NSThirdPartyExceptionRequiresForwardSecrecyNSIncludesSubdomains  </code></pre>    <p>4.在iOS 10 中 info.plist 文件新加入了 NSAllowsArbitraryLoadsInWebContent 键,允许任意web页面加载,同时苹果会用 ATS 来保护你的app;</p>    <p>5.安全传输不再支持 SSLv3 , 建议尽快停用 SHA1 和 3DES 算法;</p>    <h3><strong>6.UIStatusBar的问题:</strong></h3>    <p>在iOS10中,如果还使用以前设置UIStatusBar类型或者控制隐藏还是显示的方法,会报警告,方法过期,如下图:</p>    <p><img src="https://simg.open-open.com/show/abbac3415ec25bad26784843241e6796.png"></p>    <p>UIStatusBar的警告.png</p>    <p>上面方法到 iOS 10 不能使用了,要想修改UIStatusBar的样式或者状态使用下图中所示的属性或方法:</p>    <pre>  <code class="language-objectivec">@property(nonatomic, readonly) UIStatusBarStylepreferredStatusBarStyleNS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED; // Defaults to UIStatusBarStyleDefault  @property(nonatomic, readonly) BOOL prefersStatusBarHiddenNS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED; // Defaults to NO  - (UIStatusBarStyle)preferredStatusBarStyleNS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED; // Defaults to UIStatusBarStyleDefault  - (BOOL)prefersStatusBarHiddenNS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED; // Defaults to NO  // Override to return the type of animation that should be used for status bar changes for this view controller. This currently only affects changes to prefersStatusBarHidden.  - (UIStatusBarAnimation)preferredStatusBarUpdateAnimationNS_AVAILABLE_IOS(7_0) __TVOS_PROHIBITED; // Defaults to UIStatusBarAnimationFade  </code></pre>    <h3><strong>7.UITextField</strong></h3>    <p>在iOS 10 中, UITextField 新增了 textContentType 字段,是 UITextContentType 类型,它是一个枚举,作用是可以指定输入框的类型,以便系统可以分析出用户的语义.是电话类型就建议一些电话,是地址类型就建议一些地址.可以在 #import 文件中,查看 textContentType 字段,有以下可以选择的类型:</p>    <pre>  <code class="language-objectivec">UIKIT_EXTERNUITextContentTypeconst UITextContentTypeName                      NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeNamePrefix                NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeGivenName                NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeMiddleName                NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeFamilyName                NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeNameSuffix                NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeNickname                  NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeJobTitle                  NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeOrganizationName          NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeLocation                  NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeFullStreetAddress        NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeStreetAddressLine1        NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeStreetAddressLine2        NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeAddressCity              NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeAddressState              NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeAddressCityAndState      NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeSublocality              NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeCountryName              NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypePostalCode                NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeTelephoneNumber          NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeEmailAddress              NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeURL                      NS_AVAILABLE_IOS(10_0);  UIKIT_EXTERNUITextContentTypeconst UITextContentTypeCreditCardNumber          NS_AVAILABLE_IOS(10_0);  </code></pre>    <h3><strong>8.UserNotifications(用户通知)</strong></h3>    <p>iOS 10 中将通知相关的 API 都统一了,在此基础上很多用户定义的通知,并且可以捕捉到各个通知状态的回调.以前通知的概念是:大家想接受的提前做好准备,然后一下全两分发,没收到也不管了,也不关心发送者,现在的用户通知做成了类似于网络请求,先发一个 request 得到 response 的流程,还封装了 error ,可以在各个状态的方法中做一些额外的操作,并且能获得一些字段,比如发送者之类的.这个功能的头文件是: #import<br> 主要有以下文件:</p>    <pre>  <code class="language-objectivec">#import <UserNotifications/NSString+UserNotifications.h>  #import <UserNotifications/UNError.h>  #import <UserNotifications/UNNotification.h>  #import <UserNotifications/UNNotificationAction.h>  #import <UserNotifications/UNNotificationAttachment.h>  #import <UserNotifications/UNNotificationCategory.h>  #import <UserNotifications/UNNotificationContent.h>  #import <UserNotifications/UNNotificationRequest.h>  #import <UserNotifications/UNNotificationResponse.h>  #import <UserNotifications/UNNotificationSettings.h>  #import <UserNotifications/UNNotificationSound.h>  #import <UserNotifications/UNNotificationTrigger.h>  #import <UserNotifications/UNUserNotificationCenter.h>  #import <UserNotifications/UNNotificationServiceExtension.h>  </code></pre>    <h3><strong>9.UICollectionViewCell的的优化</strong></h3>    <p>在iOS 10 之前,UICollectionView上面如果有大量cell,当用户活动很快的时候,整个UICollectionView的卡顿会很明显,为什么会造成这样的问题,这里涉及到了iOS 系统的重用机制,当cell准备加载进屏幕的时候,整个cell都已经加载完成,等待在屏幕外面了,也就是整整一行cell都已经加载完毕,这就是造成卡顿的主要原因,专业术语叫做:掉帧.</p>    <p>要想让用户感觉不到卡顿,我们的app必须帧率达到60帧/秒,也就是说每帧16毫秒要刷新一次.</p>    <p>iOS 10 之前UICollectionViewCell的生命周期是这样的:</p>    <ul>     <li>1.用户滑动屏幕,屏幕外有一个cell准备加载进来,把cell从reusr队列拿出来,然后调用 prepareForReuse 方法,在这个方法里面,可以重置cell的状态,加载新的数据;</li>     <li>2.继续滑动,就会调用 cellForItemAtIndexPath 方法,在这个方法里面给cell赋值模型,然后返回给系统;</li>     <li>3.当cell马上进去屏幕的时候,就会调用 willDisplayCell 方法,在这个方法里面我们还可以修改cell,为进入屏幕做最后的准备工作;</li>     <li>4.执行完 willDisplayCell 方法后,cell就进去屏幕了.当cell完全离开屏幕以后,会调用 didEndDisplayingCell 方法.</li>    </ul>    <p>iOS 10 UICollectionViewCell的生命周期是这样的:</p>    <ul>     <li>1.用户滑动屏幕,屏幕外有一个cell准备加载进来,把cell从reusr队列拿出来,然后调用 prepareForReuse 方法,在这里当cell还没有进去屏幕的时候,就已经提前调用这个方法了,对比之前的区别是之前是cell的上边缘马上进去屏幕的时候就会调用该方法,而iOS 10 提前到cell还在屏幕外面的时候就调用;</li>     <li>2.在 cellForItemAtIndexPath 中创建cell,填充数据,刷新状态等操作,相比于之前也提前了;</li>     <li>3.用户继续滑动的话,当cell马上就需要显示的时候我们再调用 willDisplayCell 方法,原则就是:何时需要显示,何时再去调用 willDisplayCell 方法;</li>     <li>4.当cell完全离开屏幕以后,会调用 didEndDisplayingCell 方法,跟之前一样,cell会进入重用队列.<br> 在iOS 10 之前,cell只能从重用队列里面取出,再走一遍生命周期,并调用 cellForItemAtIndexPath 创建或者生成一个cell.<br> 在iOS 10 中,系统会cell保存一段时间,也就是说当用户把cell滑出屏幕以后,如果又滑动回来,cell不用再走一遍生命周期了,只需要调用 willDisplayCell 方法就可以重新出现在屏幕中了.<br> iOS 10 中,系统是一个一个加载cell的,二以前是一行一行加载的,这样就可以提升很多性能; <p>iOS 10 新增加的Pre-Fetching预加载</p> <p>这个是为了降低UICollectionViewCell在加载的时候所花费的时间,在 iOS 10 中,除了数据源协议和代理协议外,新增加了一个 UICollectionViewDataSourcePrefetching 协议,这个协议里面定义了两个方法:</p> </li>    </ul>    <p> </p>    <p>来自:http://ios.jobbole.com/90717/</p>    <p> </p>