iOS14 隐私适配及部分解决方案

2024-03-17 04:08

本文主要是介绍iOS14 隐私适配及部分解决方案,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在刚刚结束的线上 WWDC 2020 发布会上苹果向我们展示了新的 iOS14 系统。iOS14 的适配,很重要的一环就集中在用户隐私和安全方面。

在 iOS13 及以前,当用户首次访问应用程序时,会被要求开放大量权限,比如相册、定位、联系人,实际上该应用可能仅仅需要一个选择图片功能,却被要求开放整个照片库的权限,这确实是不合理的。对于相册,在 iOS14 中引入了 “LimitedPhotos Library” 的概念,用户可以授予应用访问其一部分的照片,对于应用来说,仅能读取到用户选择让应用来读取的照片,让我们看到了 Apple 对于用户隐私的尊重。这仅仅是一部分,在iOS14 中,可以看到诸多类似的保护用户隐私的措施,也需要我们升级适配。

最近在调研 iOS14的适配方案,本文主要分享一下 iOS14 上对于隐私授权的变更和部分适配方案,欢迎补充指正。

适配点


 相册

✎  iOS14 新增了“Limited Photo Library Access” 模式,在授权弹窗中增加了 Select Photo 选项。用户可以在 App 请求调用相册时选择部分照片让 App 读取。从 App 的视⻆来看,你的相册里就只有这几张照片,App 无法得知其它照片的存在。


✎  iOS14 中当用户选择

“PHAuthorizationStatusLimited” 时,如果未进行适配,有可能会在每次触发相册功能时都进行弹窗询问用户是否需要修改照片权限。


✎  对于这种情况可通过在 Info.plist 中设置 

“PHPhotoLibraryPreventAutomaticLimitedAccessAlert”的值为 YES 来阻止该弹窗反复弹出,并且可通过下面这个 API 来主动控制何时弹出PHPickerViewController 进行照片选择。

[[PHPhotoLibrary sharedPhotoLibrary] presentLimitedLibraryPickerFromViewController:self];

✎  在 iOS14 中官方推荐使用  PHPicker 来替代原 API 进行图片选择。PHPicker 为独立进程,会在视图最顶层进行展示,应用内无法对其进行截图也无法直接访问到其内的数据。

  • UIImagePickerController  -> PHPickerViewController, UIImagePickerViewController 功能受限,每次只能选择一张图片,将逐渐被废弃。 

  • PHPicker 支持多选,支持搜索,支持按 image,video,livePhotos 等进行选择。 

✎  新API及迁移demo:
  
@interface ViewController () <PHPickerViewControllerDelegate>    @property (weak, nonatomic) IBOutlet UIImageView *imageView;@property (nonatomic, strong) NSArray<NSItemProvider *> *itemProviders;  @end  @implementation ViewController    - (void)viewDidLoad {      [super viewDidLoad];      // Do any additional setup after loading the view.}    - (IBAction)button:(id)sender {      // 以下 API 仅为 iOS14 only      PHPickerConfiguration *configuration = [[PHPickerConfiguration alloc] init];      configuration.filter = [PHPickerFilter videosFilter]; // 可配置查询用户相册中文件的类型,支持三种    configuration.selectionLimit = 0; // 默认为1,为0时表示可多选。        PHPickerViewController *picker = [[PHPickerViewController alloc] initWithConfiguration:configuration];      picker.delegate = self;      // picker vc,在选完图片后需要在回调中手动 dismiss    [self presentViewController:picker animated:YES completion:^{        }];  }  #pragma mark - Delegate    - (void)picker:(PHPickerViewController *)picker didFinishPicking:(NSArray<PHPickerResult *> *)results {      [picker dismissViewControllerAnimated:YES completion:nil];      if (!results || !results.count) {          return;      }      NSItemProvider *itemProvider = results.firstObject.itemProvider;      if ([itemProvider canLoadObjectOfClass:UIImage.class]) {          __weak typeof(self) weakSelf = self;          [itemProvider loadObjectOfClass:UIImage.class completionHandler:^(__kindof id<NSItemProviderReading>  _Nullable object, NSError * _Nullable error) {              if ([object isKindOfClass:UIImage.class]) {                  __strong typeof(self) strongSelf = weakSelf;                  dispatch_async(dispatch_get_main_queue(), ^{                      strongSelf.imageView.image = (UIImage *)object;                  });              }          }];       }    }
✎  需要注意的是,在 limit Photo 模式下,AssetsLibrary 访问相册会失败;在 writeOnly 模式下,AssetLibrary 也会有显示问题。建议还在使用 AssetsLibrary 的同学尽快迁移到新 API。
✎  授权相关:旧 API 废弃,增加 PHAccessLevel 参数。如果再使用以前的API来获取权限状态,
PHAuthorizationStatusLimited 状态下也会返回 
PHAuthorizationStatusAuthorized

typedef NS_ENUM(NSInteger, PHAccessLevel) {  PHAccessLevelAddOnly = 1, // 仅允许添加照片  PHAccessLevelReadWrite = 2, // 允许访问照片,limitedLevel 必须为 readWrite} API_AVAILABLE(macos(10.16), ios(14), tvos(14));
// 查询权限PHAccessLevel level = PHAccessLevelReadWrite;PHAuthorizationStatus status = [PHPhotoLibrary authorizationStatusForAccessLevel:level];  switch (status) {      case PHAuthorizationStatusLimited:          NSLog(@"limited");          break;      case PHAuthorizationStatusDenied:          NSLog(@"denied");          break;      case PHAuthorizationStatusAuthorized:          NSLog(@"authorized");          break;      default:          break;}
// 请求权限,需注意 limited 权限尽在 accessLevel 为 readAndWrite 时生效[PHPhotoLibrary requestAuthorizationForAccessLevel:level handler:^(PHAuthorizationStatus status) {  switch (status) {      case PHAuthorizationStatusLimited:          NSLog(@"limited");          break;      case PHAuthorizationStatusDenied:          NSLog(@"denied");          break;      case PHAuthorizationStatusAuthorized:          NSLog(@"authorized");          break;      default:          break;   }}];
 定位

✎  在 iOS13 及以前,App 请求用户定位授权时为如下形态:一旦用户同意应用获取定位信息,当前应用就可以获取到用户的精确定位。 

 
✎  iOS14 新增用户大致位置选项可供用户选择,原因是大多数 App 实际上并不需要获取用户到用户最准确的定位信息。iOS14 授权弹窗新增的 Precise的开关默认会选中精确位置。用户通过这个开关可以进行更改,当把这个值设为 On 时,地图上会显示精确位置;切换为Off时,将显示用户的大致位置。 

✎  对于对用户位置敏感度不高的 App 来说,这个似乎无影响,但是对于强依赖精确位置的 App 适配工作就显得非常重要了。可以通过用户在 “隐私设置” 中设置来开启精确定位,但是可能用户宁可放弃使用这个应用也不愿意开启。这个时候,iOS14 在 CLLocationManager 新增两个方法可用于向用户申请临时开启一次精确位置权限。
 

使用方式也很简单,需要首先在 Info.plist 中配置“NSLocationTemporaryUsageDescriptionDictionary”字典中需要配置 key 和 value 表明使用位置的原因,以及具体的描述。 
 

在本例中,key 即为获取用户权限时传的 "purposeKey",最终呈现给用户的就是左图,右图为当App主动关闭精确定位权限申请。

✎  对于地理位置不敏感的App 来说,iOS14 也可以通过直接在 info.plist 中添加 NSLocationDefaultAccuracyReduced  为 true 默认请求大概位置。
✎  这样设置之后,即使用户想要为该 App 开启精确定位权限,也无法开启。

✎  也可以直接通过API来根据不同的需求设置不同的定位精确度。

✎  需要注意的是,当 App 在 Background 模式下,如果并未获得精确位置授权,那么 Beacon 及其他位置敏感功能都将受到限制。
▐  Local Network

✎  iOS14 当 App 要使用 Bonjour 服务时或者访问本地局域网,使用 mDNS 服务等,都需要授权,开发者需要在 Info.plist 中详细描述使用的为哪种服务以及用途。下图为需要无需申请权限与需要授权的服务:
 
✎  在 "隐私设置" 中也可以查看和修改具体有哪些 App 正在使用 LocalNetwork
    

✎  如果应用中需要使用 LocalNetwork 需要在 Info.plist 中配置两个选项,详细描述为什么需要使用该权限,以及需要列出具体使用 LocalNetwork 的服务列表。 
 
✎  对于使用了下列包含 Bonjour 的 framework,都需要更新描述.
 Wi-Fi Address

✎  iOS8 - iOS13 ,用户在不同的网络间切换和接入时,mac 地址都不会改变,这也就使得网络运营商还是可以通过 mac 地址对用户进行匹配和用户信息收集,生成完整的用户信息。iOS14 提供 Wifi 加密服务,每次接入不同的 WiFi 使用的 mac 地址都不同。每过 24 小时,mac 地址还会更新一次。需要关注是否有使用用户网络 mac 地址的服务。 

✎  下图为 iOS13 及之前用户接入网络时 mac 地址并不会进行改变
 
✎  下图为 iOS14 用户接入 Wi-Fi 时 mac 地址的变化情况
  

✎  并且用户也可以自行选择是否开启 private Wi-Fi address
 剪切板

✎  在 iOS14 中,读取用户剪切板的数据会弹出提示。

✎  弹出提示的原因是使用 UIPasteboard 访问用户数据,访问以下数据都会弹出 toast 提示。


✎  兼容方案:如果应用访问剪切板仅仅用于判断是否为URL格式,则 iOS14 新增了两个 API 可以用于规避该提示。如果应用想直接访问剪切板的数据,暂时可能无法做到规避该提示。iOS14 新增两种 
UIPasteboardDetectionPattern。

 
✎  上面的两个 API 可用于规避提示,但只能用于判断剪切板中是否有 URL,并不是真正的访问剪贴板数据,也拿不到剪切板的真实数据。下面两个 API 可以获得具体的 URL 信息,但是会触发剪切板提示。并且实测当用户剪切板中包含多个 URL 时只会返回第一个。


✎  使用示例
NSSet *patterns = [[NSSet alloc] initWithObjects:UIPasteboardDetectionPatternProbableWebURL, nil];[[UIPasteboard generalPasteboard] detectPatternsForPatterns:patterns completionHandler:^(NSSet<UIPasteboardDetectionPattern> * _Nullable result, NSError * _Nullable error) {    if (result && result.count) {            // 当前剪切板中存在 URL    }}];
 相机和麦克风

✎  iOS14 中 App 使用相机和麦克风时会有图标提示以及绿点和黄点提示,并且会显示当前是哪个 App 在使用此功能。我们无法控制是否显示该提示。
 

✎  会触发录音小黄点的代码示例: 
AVAudioRecorder *recorder = [[AVAudioRecorder alloc] initWithURL:recorderPath settings:nil error:nil];[recorder record];
✎  触发相机小绿点的代码示例:
AVCaptureDeviceInput *videoInput = [[AVCaptureDeviceInput alloc] initWithDevice:videoCaptureDevice error:nil];AVCaptureSession *session = [[AVCaptureSession alloc] init];if ([session canAddInput:videoInput]) {    [session addInput:videoInput];}[session startRunning];
 IDFA

✎  IDFA 全称为 Identity for Advertisers ,即广告标识符。用来标记用户,目前最广泛的用途是用于投放广告、个性化推荐等。
✎  在 iOS13 及以前,系统会默认为用户开启允许追踪设置,我们可以简单的通过代码来获取到用户的 IDFA 标识符。
if ([[ASIdentifierManager sharedManager] isAdvertisingTrackingEnabled]) {    NSString *idfaString = [[ASIdentifierManager sharedManager] advertisingIdentifier].UUIDString;    NSLog(@"%@", idfaString);}

✎  但是在 iOS14 中,这个判断用户是否允许被追踪的方法已经废弃。

✎  iOS14 中,系统会默认为用户关闭广告追踪权限。

✎  对于这种情况,我们需要去请求用户权限。首先需要在 Info.plist 中配置" NSUserTrackingUsageDescription " 及描述文案,接着使用 AppTrackingTransparency 框架中的 ATTrackingManager 中的 requestTrackingAuthorizationWithCompletionHandler 请求用户权限,在用户授权后再去访问 IDFA 才能够获取到正确信息。
#import <AppTrackingTransparency/AppTrackingTransparency.h>#import <AdSupport/AdSupport.h>
- (void)testIDFA {    if (@available(iOS 14, *)) {        [ATTrackingManager requestTrackingAuthorizationWithCompletionHandler:^(ATTrackingManagerAuthorizationStatus status) {            if (status == ATTrackingManagerAuthorizationStatusAuthorized) {                NSString *idfaString = [[ASIdentifierManager sharedManager] advertisingIdentifier].UUIDString;            }        }];    } else {        // 使用原方式访问 IDFA    }}

对于用户拒绝授权 UserTracking 的情况,可以考虑接入苹果的 SKAdNetwork 框架进行广告分析。感兴趣的同学可进一步了解:https://developer.apple.com/documentation/storekit/skadnetwork

 上传 AppStore

更加严格的隐私审核,可以让用户在下载 App 之前就知道此 App 将会需要哪些权限。目前苹果商店要求所有应用在上架时都必须提供一份隐私政策。如果引入了第三方收集用户信息等SDK,都需要向苹果说明是这些信息的用途。 


总结



对于这次 iOS14 的隐私权限大升级和新尝试,体现了苹果对于用户隐私的尊重。


从用户角度来说,近年来越来越精准的广告投放让我们越来越感觉自己被”监视“着,此次升级后,我们有了更多保护自己隐私的方式以及避免广告骚扰的方法,苹果此举无疑会加大我们对其的好感度和信任感。但从另一个角度来说,对于 IDFA 的限制,可能会导致之前许多依靠广告投放收入的免费 App 难以继续维持生计,也可能也会导致免费 App 的数量有所降低。从开发者的角度来说,除了对 iOS14 隐私升级的积极适配外,也让我们感受到了 iOS14 中对于用户隐私的重视无疑会提高获取用户行为信息的成本。


冲击最大的应该就是广告行业,对于目前的推荐算法和用户拉新都会受到影响,如何在充分尊重用户隐私的前提下进行广告的精准投放对于开发者和广告商来说都是一个不小的机遇和挑战。


下一期,我们聊聊 【Metal 新特性详解|带来的技术启发和思考】,敬请持续关注~



参考资料

  • WWDC 2020 Apple Developer

  • Developer Documentation

手淘客户端—小程序与跨平台技术部

我们是支撑小程序、小游戏、Flutter 等跨平台技术的核心团队,有技术广度和也有技术深度,我们需要 iOS、Android、C++、Flutter、Canvas、游戏引擎、WebGL 等各方面的人才。如果你善于学习,这是一个很好的接触跨领域知识的机会。

如果你是个对技术有追求对小伙伴,请别犹豫,立刻联系我!

????:lanya.sly@alibaba-inc.com

✿  拓展阅读

作者|盛兰雅(岚遥)

编辑|橙子君

出品|阿里巴巴新零售淘系技术


这篇关于iOS14 隐私适配及部分解决方案的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/817754

相关文章

Redis连接失败:客户端IP不在白名单中的问题分析与解决方案

《Redis连接失败:客户端IP不在白名单中的问题分析与解决方案》在现代分布式系统中,Redis作为一种高性能的内存数据库,被广泛应用于缓存、消息队列、会话存储等场景,然而,在实际使用过程中,我们可能... 目录一、问题背景二、错误分析1. 错误信息解读2. 根本原因三、解决方案1. 将客户端IP添加到Re

python 字典d[k]中key不存在的解决方案

《python字典d[k]中key不存在的解决方案》本文主要介绍了在Python中处理字典键不存在时获取默认值的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录defaultdict:处理找不到的键的一个选择特殊方法__missing__有时候为了方便起见,

Linux限制ip访问的解决方案

《Linux限制ip访问的解决方案》为了修复安全扫描中发现的漏洞,我们需要对某些服务设置访问限制,具体来说,就是要确保只有指定的内部IP地址能够访问这些服务,所以本文给大家介绍了Linux限制ip访问... 目录背景:解决方案:使用Firewalld防火墙规则验证方法深度了解防火墙逻辑应用场景与扩展背景:

SpringBoot嵌套事务详解及失效解决方案

《SpringBoot嵌套事务详解及失效解决方案》在复杂的业务场景中,嵌套事务可以帮助我们更加精细地控制数据的一致性,然而,在SpringBoot中,如果嵌套事务的配置不当,可能会导致事务不生效的问题... 目录什么是嵌套事务?嵌套事务失效的原因核心问题:嵌套事务的解决方案方案一:将嵌套事务方法提取到独立类

Spring Boot实现多数据源连接和切换的解决方案

《SpringBoot实现多数据源连接和切换的解决方案》文章介绍了在SpringBoot中实现多数据源连接和切换的几种方案,并详细描述了一个使用AbstractRoutingDataSource的实... 目录前言一、多数据源配置与切换方案二、实现步骤总结前言在 Spring Boot 中实现多数据源连接

MySQL的索引失效的原因实例及解决方案

《MySQL的索引失效的原因实例及解决方案》这篇文章主要讨论了MySQL索引失效的常见原因及其解决方案,它涵盖了数据类型不匹配、隐式转换、函数或表达式、范围查询、LIKE查询、OR条件、全表扫描、索引... 目录1. 数据类型不匹配2. 隐式转换3. 函数或表达式4. 范围查询之后的列5. like 查询6

使用Vue.js报错:ReferenceError: “Vue is not defined“ 的原因与解决方案

《使用Vue.js报错:ReferenceError:“Vueisnotdefined“的原因与解决方案》在前端开发中,ReferenceError:Vueisnotdefined是一个常见... 目录一、错误描述二、错误成因分析三、解决方案1. 检查 vue.js 的引入方式2. 验证 npm 安装3.

PHP执行php.exe -v命令报错的解决方案

《PHP执行php.exe-v命令报错的解决方案》:本文主要介绍PHP执行php.exe-v命令报错的解决方案,文中通过图文讲解的非常详细,对大家的学习或工作有一定的帮助,需要的朋友可以参考下... 目录执行phpandroid.exe -v命令报错解决方案执行php.exe -v命令报错-PHP War

鸿蒙开发搭建flutter适配的开发环境

《鸿蒙开发搭建flutter适配的开发环境》文章详细介绍了在Windows系统上如何创建和运行鸿蒙Flutter项目,包括使用flutterdoctor检测环境、创建项目、编译HAP包以及在真机上运... 目录环境搭建创建运行项目打包项目总结环境搭建1.安装 DevEco Studio NEXT IDE

poj 2976 分数规划二分贪心(部分对总体的贡献度) poj 3111

poj 2976: 题意: 在n场考试中,每场考试共有b题,答对的题目有a题。 允许去掉k场考试,求能达到的最高正确率是多少。 解析: 假设已知准确率为x,则每场考试对于准确率的贡献值为: a - b * x,将贡献值大的排序排在前面舍弃掉后k个。 然后二分x就行了。 代码: #include <iostream>#include <cstdio>#incl