XML属性列表-plist
一、应用沙盒
每个iOS应用都有⾃己的应⽤沙盒(应用沙盒就是文件系统目录),与其他文件系统隔离。应⽤必须待在⾃己的沙盒里,其他应用不能访问该沙盒(提示:在IOS8中已经开放访问)

应⽤沙盒的文件系统⽬录,如下图所示(假设应用的名称叫Layer)

2015121192447046.png (348×169)

模拟器应⽤用沙盒的根路径在: (apple是⽤用户名, 7.0是模拟器版本) /Users/apple/Library/Application Support/iPhone Simulator/7.0/Applications

二、应用沙盒结构分析

应⽤程序包:(上图中的Layer)包含了所有的资源文件和可执行文件

Documents:保存应⽤运行时生成的需要持久化的数据,iTunes同步设备时会备份该目录。例如,游戏应用可将游戏存档保存在该目录

tmp:保存应⽤运行时所需的临时数据,使⽤完毕后再将相应的文件从该目录删除。应用没有运行时,系统也可能会清除该目录下的文件。iTunes同步设备时 不会备份该目录

Library/Caches:保存应用运行时⽣成的需要持久化的数据,iTunes同步设备时不会备份该目录。⼀一般存储体积大、不需要备份的非重要数据

Library/Preference:保存应用的所有偏好设置,iOS的Settings(设置) 应⽤会在该⺫录中查找应⽤的设置信息。iTunes同步设备时会备份该目录

三、应用沙盒常见的获取方式

沙盒根目录:NSString *home = NSHomeDirectory();
 Documents:(2种⽅方式)

利用沙盒根目录拼接”Documents”字符串

NSString *home = NSHomeDirectory();

NSString *documents = [home stringByAppendingPathComponent:@"Documents"]; // 不建议采用,因为新版本的操作系统可能会修改目录名

利用NSSearchPathForDirectoriesInDomains函数
// NSUserDomainMask 代表从用户文件夹下找

// YES 代表展开路径中的波浪字符“~”

NSArray *array = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, NO); // 在iOS中,只有一个目录跟传入的参数匹配,所以这个集合里面只有一个元素
NSString *documents = [array objectAtIndex:0];
tmp:NSString *tmp = NSTemporaryDirectory();

Library/Caches:(跟Documents类似的2种⽅方法)

利用沙盒根目录拼接”Caches”字符串

利⽤NSSearchPathForDirectoriesInDomains函数(将函数的第2个参数改 为:NSCachesDirectory即可)

Library/Preference:通过NSUserDefaults类存取该目录下的设置信息

相应的代码:

#import "NJViewController.h"

#import "NJPerson.h"
@interface NJViewController ()

- (IBAction)saveDataBtnClick:(id)sender;

- (IBAction)readDataBtnClick:(id)sender;
@end

@implementation NJViewController

/**

 *   点击保存按钮

 */

- (IBAction)saveDataBtnClick:(id)sender {

    

    // youtube做法

//    NSString *path = @"/Users/apple/Library/Application Support/iPhone Simulator/7.1/Applications/A6D53E11-DDF0-4392-B2D4-FE77A96888A6/Documents/abc.plist";

    

    // 获取应用程序根目录

    NSString *home = NSHomeDirectory();

    

    // 不建议写/

    //NSString *path = [home stringByAppendingString:@"/Documents"];

    // 不建议Documents写死

    //NSString *path = [home stringByAppendingPathComponent:@"Documents"];

    

    // NSUserDomainMask 在用户目录下查找

    // YES 代表用户目录的~

    // NSDocumentDirectory 查找Documents文件夹

    // 建议使用如下方法动态获取

    NSString *doc = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];

    // 拼接文件路径

    NSString *path = [doc stringByAppendingPathComponent:@"abc.plist"];

    NSLog(@"%@", path);

    

    

    //NSArray *arr = @[@"lnj", @"28"];

    //[arr writeToFile:path atomically:YES];

    

    // NSDictionary *dict = @{@"name": @"lnj", @"age":@"28"};

    // 调用writeToFile将数据写入文件

    // [dict writeToFile:path atomically:YES];

    

    /*

     plist只能存储系统自带的一些常规的类, 也就是有writeToFile方法的对象才可以使用plist保存数据

     字符串/字典/数据/NSNumber/NSData ...

     */

    

    // 自定义的对象不能保存到plist中

    NJPerson *p = [[NJPerson alloc] init];

    p.name =@"lnj";

    

    NSDictionary *dict = @{@"person": @"abc"};

    [dict writeToFile:path atomically:YES];

}

/**

 *   点击读取按钮

 */

- (IBAction)readDataBtnClick:(id)sender {

    NSString *doc = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];

    

    NSString *path = [doc stringByAppendingPathComponent:@"abc.plist"]

    ;

    // 读取数据

    NSDictionary *dict = [NSDictionary dictionaryWithContentsOfFile:path];

    NSLog(@"%@", dict);

}

@end

四、属性列表

属性列表是一种XML格式的文件,拓展名为plist

如果对象是NSString、NSDictionary、NSArray、NSData、 NSNumber等类型,就可以使用writeToFile:atomically:⽅法 直接将对象写到属性列表文件中


NSKeydeArchiver归档
一、简单说明

在使用plist进行数据存储和读取,只适用于系统自带的一些常用类型才能用,且必须先获取路径相对麻烦;
偏好设置(将所有的东西都保存在同一个文件夹下面,且主要用于存储应用的设置信息)
归档:因为前两者都有一个致命的缺陷,只能存储常用的类型。归档可以实现把自定义的对象存放在文件中。
二、代码示例

1.文件结构

2015121192529907.png (1392×334)

//

//  YYViewController.m

//  02-归档

//

//  Created by apple on 14-6-7.

//  Copyright (c) 2014年 itcase. All rights reserved.

//
#import "YYViewController.h"

#import "YYPerson.h"
@interface YYViewController ()

- (IBAction)saveBtnOnclick:(id)sender;

- (IBAction)readBtnOnclick:(id)sender;
@end

@implementation YYViewController
- (void)viewDidLoad

{

    [super viewDidLoad];

}


- (IBAction)saveBtnOnclick:(id)sender {

    //1.创建对象

    YYPerson *p=[[YYPerson alloc]init];

    p.name=@"文顶顶";

    p.age=23;

    p.height=1.7;

    

    //2.获取文件路径

    NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)lastObject];

    NSString *path=[docPath stringByAppendingPathComponent:@"person.yangyang"];

    NSLog(@"path=%@",path);

    

    //3.将自定义的对象保存到文件中

    [NSKeyedArchiver archiveRootObject:p toFile:path];

    

}
- (IBAction)readBtnOnclick:(id)sender {

    //1.获取文件路径

    NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)lastObject];

    NSString *path=[docPath stringByAppendingPathComponent:@"person.yangyang"];

    NSLog(@"path=%@",path);

    

    //2.从文件中读取对象

    YYPerson *p=[NSKeyedUnarchiver unarchiveObjectWithFile:path];

    NSLog(@"%@,%d,%.1f",p.name,p.age,p.height);

}

@end

新建一个person类

YYPerson.h文件

//

//  YYPerson.h

//  02-归档

//

//  Created by apple on 14-6-7.

//  Copyright (c) 2014年 itcase. All rights reserved.

//
#import 
// 如果想将一个自定义对象保存到文件中必须实现NSCoding协议

@interface YYPerson : NSObject
//姓名

@property(nonatomic,copy)NSString *name;

//年龄

@property(nonatomic,assign)int age;

//身高

@property(nonatomic,assign)double height;

@end

YYPerson.m文件
//

//  YYPerson.m

//  02-归档

//

//  Created by apple on 14-6-7.

//  Copyright (c) 2014年 itcase. All rights reserved.

//
#import "YYPerson.h"
@implementation YYPerson
// 当将一个自定义对象保存到文件的时候就会调用该方法

// 在该方法中说明如何存储自定义对象的属性

// 也就说在该方法中说清楚存储自定义对象的哪些属性

-(void)encodeWithCoder:(NSCoder *)aCoder

{

    NSLog(@"调用了encodeWithCoder:方法");

    [aCoder encodeObject:self.name forKey:@"name"];

    [aCoder encodeInteger:self.age forKey:@"age"];

    [aCoder encodeDouble:self.height forKey:@"height"];

}
// 当从文件中读取一个对象的时候就会调用该方法

// 在该方法中说明如何读取保存在文件中的对象

// 也就是说在该方法中说清楚怎么读取文件中的对象

-(id)initWithCoder:(NSCoder *)aDecoder

{

    NSLog(@"调用了initWithCoder:方法");

    //注意:在构造方法中需要先初始化父类的方法

    if (self=[super init]) {

        self.name=[aDecoder decodeObjectForKey:@"name"];

        self.age=[aDecoder decodeIntegerForKey:@"age"];

        self.height=[aDecoder decodeDoubleForKey:@"height"];

    }

    return self;

}

@end

3.打印效果和两个重要的错误提示

点击保存按钮和读取按钮,成功打印结果如下:

2015121192550090.png (882×137)

关于不实现两个协议方法的错误提示:

-(void)encodeWithCoder:(NSCoder *)aCoder方法:

2015121192609232.png (429×303)

-(id)initWithCoder:(NSCoder *)aDecoder方法:

2015121192625411.png (427×283)

三、继承类中的使用

新建一个学生类,让这个类继承自Preson这个类,增加一个体重的属性。

YYstudent.h文件

//

//  YYstudent.h

//  02-归档

//

//  Created by apple on 14-6-7.

//  Copyright (c) 2014年 itcase. All rights reserved.

//
#import "YYPerson.h"
@interface YYstudent : YYPerson

//增加一个体重属性

@property(nonatomic,assign) double weight;

@end

YYstudent.m文件
//

//  YYstudent.m

//  02-归档

//

//  Created by apple on 14-6-7.

//  Copyright (c) 2014年 itcase. All rights reserved.

//
#import "YYstudent.h"
@implementation YYstudent
//在子类中重写这两个方法

- (void)encodeWithCoder:(NSCoder *)aCoder

{

    [super encodeWithCoder:aCoder];

    NSLog(@"调用了YYStudent encodeWithCoder");

    [aCoder encodeFloat:self.weight forKey:@"weight"];

}
- (id)initWithCoder:(NSCoder *)aDecoder

{

    if (self = [super initWithCoder:aDecoder]) {

        NSLog(@"调用了YYstudent initWithCoder");

        self.weight = [aDecoder decodeFloatForKey:@"weight"];

    }

    return self;

}

@end

YYViewController.m文件
//

//  YYViewController.m

//  02-归档

//

//  Created by apple on 14-6-7.

//  Copyright (c) 2014年 itcase. All rights reserved.

//
#import "YYViewController.h"

#import "YYPerson.h"

#import "YYstudent.h"
@interface YYViewController ()

- (IBAction)saveBtnOnclick:(id)sender;

- (IBAction)readBtnOnclick:(id)sender;
@end

@implementation YYViewController
- (void)viewDidLoad

{

    [super viewDidLoad];

}


- (IBAction)saveBtnOnclick:(id)sender {

    //1.创建对象

//    YYPerson *p=[[YYPerson alloc]init];

//    p.name=@"文顶顶";

//    p.age=23;

//    p.height=1.7;

    

    YYstudent *s=[[YYstudent alloc]init];

    s.name=@"wendingding";

    s.age=23;

    s.height=1.7;

    s.weight=62;

    //2.获取文件路径

    NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)lastObject];

    NSString *path=[docPath stringByAppendingPathComponent:@"person.yangyang"];

    NSLog(@"path=%@",path);

    

    //3.将自定义的对象保存到文件中

//    [NSKeyedArchiver archiveRootObject:p toFile:path];

     [NSKeyedArchiver archiveRootObject:s toFile:path];

    

}
- (IBAction)readBtnOnclick:(id)sender {

    //1.获取文件路径

    NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)lastObject];

    NSString *path=[docPath stringByAppendingPathComponent:@"person.yangyang"];

    NSLog(@"path=%@",path);

    

    //2.从文件中读取对象

//    YYPerson *p=[NSKeyedUnarchiver unarchiveObjectWithFile:path];

//    NSLog(@"%@,%d,%.1f",p.name,p.age,p.height);

    YYstudent *s=[NSKeyedUnarchiver unarchiveObjectWithFile:path];

    NSLog(@"%@,%d,%.1f,%f",s.name,s.age,s.height,s.weight);

}

@end

点击保存按钮和读取按钮后的打印输出:

2015121192647864.png (879×179)

四、重要说明

1.保存数据过程:

//1.创建对象

    YYstudent *s=[[YYstudent alloc]init];

    s.name=@"wendingding";

    s.age=23;

    s.height=1.7;

    s.weight=62;

    

    //2.获取文件路径

    NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)lastObject];

    NSString *path=[docPath stringByAppendingPathComponent:@"person.yangyang"];

    NSLog(@"path=%@",path);

    

    //3.将自定义的对象保存到文件中

     [NSKeyedArchiver archiveRootObject:s toFile:path];

2.读取数据过程:
//1.获取文件路径

    NSString *docPath=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)lastObject];

    NSString *path=[docPath stringByAppendingPathComponent:@"person.yangyang"];

    //2.从文件中读取对象

    YYstudent *s=[NSKeyedUnarchiver unarchiveObjectWithFile:path];

3.遵守NSCoding协议,并实现该协议中的两个方法。

4.如果是继承,则子类一定要重写那两个方法。因为person的子类在存取的时候,会去子类中去找调用的方法,没找到那么它就去父类中找,所以最后保存和读取的时候新增加的属性会被忽略。需要先调用父类的方法,先初始化父类的,再初始化子类的。

5.保存数据的文件的后缀名可以随意命名。

6.通过plist保存的数据是直接显示的,不安全。通过归档方法保存的数据在文件中打开是乱码的,更安全。

深入讲解iOS开发中应用数据的存储方式的更多相关文章

  1. HTML5在微信内置浏览器下右上角菜单的调整字体导致页面显示错乱的问题

    HTML5在微信内置浏览器下,在右上角菜单的调整字体导致页面显示错乱的问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

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

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

  3. ios – containerURLForSecurityApplicationGroupIdentifier:在iPhone和Watch模拟器上给出不同的结果

    我使用默认的XCode模板创建了一个WatchKit应用程序.我向iOSTarget,WatchkitAppTarget和WatchkitAppExtensionTarget添加了应用程序组权利.(这是应用程序组名称:group.com.lombax.fiveminutes)然后,我尝试使用iOSApp和WatchKitExtension访问共享文件夹URL:延期:iOS应用:但是,测试NSURL

  4. ios – Testflight无法安装应用程序

    我有几个测试人员注册了testflight并连接了他们的设备……他们有不同的ios型号……但是所有这些都有同样的问题.当他们从“safari”或“testflight”应用程序本身单击应用程序的安装按钮时……达到约90%并出现错误消息…

  5. ibm-mobilefirst – 在iOS 7.1上获取“无法安装应用程序,因为证书无效”错误

    当我的客户端将他们的设备更新到iOS7.1,然后尝试从AppCenter更新我们的应用程序时,我收到了上述错误.经过一番搜索,我找到了一个类似问题的帖子here.但是后来因为我在客户端使用AppCenter更新应用程序的环境中,我无法使用USB插件并为他们安装应用程序.在发布支持之前,是否有通过AppCenter进行下载的解决方法?

  6. ios – 视图的简单拖放?

    我正在学习iOS,但我找不到如何向UIView添加拖放行为.我试过了:它说“UIView没有可见的接口声明选择器addTarget”此外,我尝试添加平移手势识别器,但不确定这是否是我需要的它被称为,但不知道如何获得事件的坐标.在iOS中注册移动事件回调/拖放操作的标准简单方法是什么?

  7. ios – 什么控制iTunes中iPhone应用程序支持的语言列表?

    什么控制iPhone应用程序的iTunes页面中支持的语言?

  8. ios – 获得APNs响应BadDeviceToken或Unregistered的可能原因是什么?

    我知道设备令牌在某些时候是有效的.用户如何使其设备令牌变坏?从关于“未注册”的文档:Thedevicetokenisinactiveforthespecifiedtopic.这是否意味着应用程序已被删除?.您应该看到四种分发方法:如果您选择AppStore或Enterprise,您将在后面的对话框中看到Xcode将APNS权利更改为生产:如果选择AdHoc或Development,则aps-environment下的文本将是开发,然后应与后端的配置匹配.

  9. ios – 当我关闭应用程序时,我从调试器获得消息:由于信号15而终止

    我怎么能解决这个问题,我不知道这个链接MypreviousproblemaboutCoredata对我的问题有影响吗?当我cmd应用程序的Q时,将出现此消息.Messagefromdebugger:Terminatedduetosignal15如果谁知道我以前的问题的解决方案,请告诉我.解决方法>来自调试器的消息:每当用户通过CMD-Q(退出)或STOP手动终止应用程序(无论是在iOS模拟器中还是

  10. ios – NSUbiquityIdentityDidChangeNotification和SIGKILL

    当应用程序被发送到后台时,我们会删除观察者吗?我遇到的问题是,当UbiquityToken发生变化时,应用程序终止,因为用户已经更改了iCloud设置.你们如何设法订阅这个通知,如果你不这样做,你会做什么来跟踪当前登录的iCloud用户?

随机推荐

  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中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部