我正在使用NSUserDefaults存储应用程序EULA和PP是否已被接受(除其他外)这一般工作正常.我可以开始,退出然后返回到应用程序,它读取值很好.我可以杀死应用程序并重新启动 – 读取默认值.我可以重新启动手机,然后重新启动应用程序,它会读取默认设置.

但是当手机从平板电池重新启动时,我打开应用程序并提示再次接受我的EULA PP.这只发生在IOS7上的iPhone5上.我在IOS6上有一个3GS,它没有表现出相同的行为.

我怀疑它可能是解决了here问题的类似问题,但这指的是钥匙串中的许可问题.相同的权限问题是否适用于NSUserDefaults?

有没有人在使用NSUserDefaults的IOS7上遇到过类似的问题?

解决方法

所以经过实验和谷歌搜索我得出的结论如下:

显着的更新更改,实际上任何在锁定屏幕后启动应用程序的进程都是此处的问题. [NSUserDefaults defaultUser]加载的文件.plist受到保护(NSFileProtectionCompleteUntilFirstUserAuthentication),因此在应用程序启动后第一次解锁后才能访问它.因此,如果进程在后台启动您的应用程序并且您的应用程序尝试访问defaultUser用户默认值,则无法加载该文件,因此会为您提供一组新的用户默认值.

在我的案例中发生的事情是,应用程序然后进入等待EULA和PP被接受的状态,因为它从默认值(无法读取)中读取它们尚未被接受.解锁手机并重新打开应用程序 – 请注意已经“启动” – 有一些进程写入NSUserDefaults,一些在我的应用程序中,一些在我的应用程序使用的库中.在大多数情况下,我在默认情况下调用同步,因此爆破了无法读取的旧默认值.我想这可能是很多人的情况.

有几种不同的解决方案.

首先,我编写了一个等同于NSUserDefaults的类,它包装了一个NSMutableDictionary,并将字典保存到库/应用程序支持中的.plist.我将文件上的保护更改为NSFileProtectionNone.请注意,如果将敏感信息存储在此文件中,则不建议这样做.另请注意,每次编写文件时都必须设置文件的权限.就像是:

NSError *error;   
BOOL saved = [defaultsDic writetoURL:defaultsFileUrl atomically:YES];
[[NSFileManager defaultManager] setAttributes:[NSDictionary dictionaryWithObject:NSFileProtectionNone forKey:NSFileProtectionKey]ofItemAtPath:[defaultsFileUrl path] error:&error];

这种方法工作正常,但事实证明我有另一个问题,我正在从钥匙串读取和写入数据.请参阅上面我的问题中的链接,它是同样的问题.在应用程序启动后第一次解锁之前,钥匙串值具有相同的保护.我不想删除对钥匙串的保护,我实际上也不太习惯从我的自定义用户默认设置中删除保护.

所以下一个解决方案是实际解决问题.如果应用程序在锁定屏幕后面启动,请不要尝试访问受保护的数据!这意味着我必须检测到应用程序是在锁定屏幕后面启动的,然后等到应用程序解锁后再继续读取我的用户默认值和钥匙串值.

第一个要求是检查应用程序启动是否有受保护的数据,因此可能在applicationDidLaunch或其他合适的地方.

[[UIApplication sharedApplication]isProtectedDataAvailable]

如果启动应用程序时不是这样,那么您就在锁定屏幕后面.您应该暂停此操作并避免访问NSUserDefaults或Keychain(或任何受保护的文件!)的任何操作.然后,您需要等待受保护数据可用的信号.当用户解锁锁定屏幕时,appDelegate会收到以下信息:

-(void)applicationProtectedDataDidBecomeAvailable:(UIApplication *)application

收到后,您可以继续执行您的申请.

在我的情况下,我控制单例类中的所有内容.创建该类时(仅在应用程序启动时发生)我检查受保护数据是否可用,并订阅NSNotificationCenter以获取相同的通知:

[[NSNotificationCenter defaultCenter]addobserver:self selector:@selector(applicationProtectedDataDidBecomeAvailable) name:UIApplicationProtectedDataDidBecomeAvailable object:nil];

因此,使用第二种方法,问题得以解决,数据仍然受到保护,每个人都很高兴.

为什么在扁平电池IOS7之后无法读取NSUserDefaults的更多相关文章

  1. html5 canvas手势解锁源码分享

    这篇文章主要介绍了html5 canvas手势解锁源码分享,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下

  2. h5使用canvas画布实现手势解锁

    这篇文章主要介绍了h5使用canvas画布实现手势解锁的相关资料,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  3. ios – NSUserDefaults与NSCoding

    从我的应用程序中保存少量数据时,最好使用NSUserDefaults还是NSCoding?

  4. iOS Swift在哪里存储用户登录数据或OAuth令牌?

    事情并不像在用户手机上存储登录数据的最佳做法那样清晰.有人建议将userID=123和loggedIn=true类型数据等数据存储在NSUSerDefaults数据中.然而根据我的理解,根据这篇文章https://www.andyibanez.com/nsuserdefaults-not-for-sensitive-data/,这些数据可以很容易地被操作所以问题是:当用户浏览各种屏幕时,持久登录数

  5. ios – 在iPhone应用程序中保存状态的位置

    我想知道正确的地方可以保存应用程序的运行状态,因为iCloud和iOS5已经改变了各种选项的工作方式.NSUserDefaults的?是不是与iCloud同步?我没有特别希望上传应用程序的状态.这是不必要的.缓存?那现在已经被清除了.这不是一个保存东西的安全点.那么这些天拯救国家的官方推荐地点是什么?解决方法在我看来,iOS开发人员倾向于使用NSUserDefaults,即使它指的是用户设置.例如,我最近将流行的Appirater库添加到我的应用程序中,并发现它使用NSUserDefaults来存储其状态

  6. ios – 我正在将数据从我的Swift应用程序上传到Amazon S3,它就像其他任何东西一样耗尽电池.如何避免这种情况?

    在我的’Swift‘应用程序中,我有一个将照片上传到我的AmazonS3存储桶的功能.当用户连接到WiFi或LTE时,没有问题,但是当连接速度稍慢(例如3G)时,上传需要花费很多时间(最多一分钟),iphone可能会损失15-20%电池!我将照片调整到大约200-300kb,这应该不是问题.我用的代码是:有什么事情让你想到我在这里做错了什么?如何避免这种巨大的电池消耗?解决方法以下答案的灵感来自h

  7. ios – NSUserDefaults没有保存

    我在我的精灵工具包应用程序中遇到问题,我的NSUserDefaults变量无效.在createSceneContents(我知道被调用)然后当您单击SKLabelNode以更改难度时,将调用此代码但是当我停止程序并再次运行时,它总是说难度:容易.有什么建议?解决方法正如rmaddy的回答所解释的那样,NSUserDefaults不会立即将数据写入长期存储.当你调用setobject:forKey:

  8. ios – Swift – 如何设置NSUserDefaults的初始值

    我有一个名为soundSwitch的开关,我使用用户默认值保存按钮的状态:当用户首次启动应用程序时,实际的默认值最初为false.如果用户首次启动应用程序并且尚未配置它,则如何实现默认值为true.我在Objective-C中看到过方法,但在Swift中没有.从我看到的,你可以在应用程序中做一些如何,或在plist文件中.我该如何做这些?解决方法Swift3语法示例注册一个布尔默认值:并获得价值:Sidenote:虽然上述代码将返回true,但请注意,在设置之前,该值实际上并不写入磁盘:

  9. 显示在视图控制器之间保留的iOS应用程序的计时器

    我一直试图通过使用NSTimer在我的应用程序的左下角显示一个计时器,并将“经过时间”显示为左下角的UILabel,但它并没有为我工作.计时器实际上工作,但我不能让它由按钮触发.我正在尝试让计时器继续运行,而不是在进入下一个storyboard/xib文件时重新启动.解决方法要在按下按钮时实现计时器操作,您需要在IBAction方法上编写它,如:要存储以前的值,可以使用NSUserDefaults或sqlite数据库.为此,我建议NSUserDefaults.更改aTime方法,如:

  10. NSUserDefaults不在iOS 8中的应用程序和自定义键盘之间存储值

    我正在尝试在我的应用程序和自定义键盘扩展之间共享数据.我在主应用程序目标和自定义键盘目标中打开了AppGroups.在我的主应用程序中,我添加了一个具有以下内容的对象:在控制台中打印出[userDefaultsdictionaryRepresentation]会显示该对象已被保存,调用[userDefaultsobjectForKey:@“KEY”]也是如此.但是,当我尝试在自定义键盘扩展中访问此

随机推荐

  1. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  2. iOS – genstrings:无法连接到输出目录en.lproj

    使用我桌面上的项目文件夹,我启动终端输入:cd然后将我的项目文件夹拖到终端,它给了我路径.然后我将这行代码粘贴到终端中找.-name*.m|xargsgenstrings-oen.lproj我在终端中收到此错误消息:genstrings:无法连接到输出目录en.lproj它多次打印这行,然后说我的项目是一个目录的路径?没有.strings文件.对我做错了什么的想法?

  3. iOS 7 UIButtonBarItem图像没有色调

    如何确保按钮图标采用全局色调?解决方法只是想将其转换为根注释,以便为“回答”复选标记提供更好的上下文,并提供更好的格式.我能想出这个!

  4. ios – 在自定义相机层的AVFoundation中自动对焦和自动曝光

    为AVFoundation定制图层相机创建精确的自动对焦和曝光的最佳方法是什么?

  5. ios – Xcode找不到Alamofire,错误:没有这样的模块’Alamofire’

    我正在尝试按照github(https://github.com/Alamofire/Alamofire#cocoapods)指令将Alamofire包含在我的Swift项目中.我创建了一个新项目,导航到项目目录并运行此命令sudogeminstallcocoapods.然后我面临以下错误:搜索后我设法通过运行此命令安装cocoapodssudogeminstall-n/usr/local/bin

  6. ios – 在没有iPhone6s或更新的情况下测试ARKit

    我在决定下载Xcode9之前.我想玩新的框架–ARKit.我知道要用ARKit运行app我需要一个带有A9芯片或更新版本的设备.不幸的是我有一个较旧的.我的问题是已经下载了新Xcode的人.在我的情况下有可能运行ARKit应用程序吗?那个或其他任何模拟器?任何想法或我将不得不购买新设备?解决方法任何iOS11设备都可以使用ARKit,但是具有高质量AR体验的全球跟踪功能需要使用A9或更高版本处理器的设备.使用iOS11测试版更新您的设备是必要的.

  7. 将iOS应用移植到Android

    我们制作了一个具有2000个目标c类的退出大型iOS应用程序.我想知道有一个最佳实践指南将其移植到Android?此外,由于我们的应用程序大量使用UINavigation和UIView控制器,我想知道在Android上有类似的模型和实现.谢谢到目前为止,guenter解决方法老实说,我认为你正在计划的只是制作难以维护的糟糕代码.我意识到这听起来像很多工作,但从长远来看它会更容易,我只是将应用程序的概念“移植”到android并从头开始编写.

  8. ios – 在Swift中覆盖Objective C类方法

    我是Swift的初学者,我正在尝试在Swift项目中使用JSONModel.我想从JSONModel覆盖方法keyMapper,但我没有找到如何覆盖模型类中的Objective-C类方法.该方法的签名是:我怎样才能做到这一点?解决方法您可以像覆盖实例方法一样执行此操作,但使用class关键字除外:

  9. ios – 在WKWebView中获取链接URL

    我想在WKWebView中获取tapped链接的url.链接采用自定义格式,可触发应用中的某些操作.例如HTTP://我的网站/帮助#深层链接对讲.我这样使用KVO:这在第一次点击链接时效果很好.但是,如果我连续两次点击相同的链接,它将不报告链接点击.是否有解决方法来解决这个问题,以便我可以检测每个点击并获取链接?任何关于这个的指针都会很棒!解决方法像这样更改addobserver在observeValue函数中,您可以获得两个值

  10. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部