【IOS开发基础系列】数据持久化专题

iOS中可以有四种持久化数据的方式: 属性列表、对象归档、SQLite3和Core Data。

1 NSUserDefault

NSUserDefaults简介,使用 NSUserDefaults 存储自定义对象

http://my.oschina.net/u/1245365/blog/294449


1.1 基于initWithSuiteName的AppGroup应用间数据共享

1.1.1 存数据

通过以下方式向NSUserDefaults中保存数据:

- (void)saveTextByNSUserDefaults

{

    NSUserDefaults *shared = [[NSUserDefaults alloc] initWithSuiteName: @"group.kevin"];

    [shared setObject: _textField.text forKey: @"kevinjun"];

    [shared synchronize];

}


        需要注意的是:

    1. 保存数据的时候必须指明group id;

    2. 而且要注意NSUserDefaults能够处理的数据只能是可plist化的对象,详情见Property List Programming Guide。

    3. 为了防止出现数据同步问题,不要忘记调用[shared synchronize];

1.1.2 读数据

        对应的读取数据方式:

- (NSString *) readDataFromNSUserDefaults

{

    NSUserDefaults *shared = [[NSUserDefaults alloc] initWithSuiteName: @"group.kevin"];

    NSString *value = [shared valueForKey: @"kevinjun"];

    return value;


2 属性列表

        读取Documents目录下文件:

NSArray* myPaths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);

NSString* myDocPath = [myPaths objectAtIndex: 0];

        获取文件的完整路径:

- (NSString*) filePath: (NSString*)fileName {

    NSArray *myPaths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);

    NSString *myDocPath = [myPaths objectAtIndex: 0];

    NSString *filePath = [myDocPath stringByAppendingPathComponent: fileName];

     return filePath;

}

    获取tmp目录

        获取应用程序的tmp目录要比获取Documents目录容易的多。使用函数NSTemporaryDirectory ()可以获得tmp目录路径。

NSString* tempPath = NSTemporaryDirectory();

    获取文件的完整路径

NSString* tempFile = [tempPath stringByAppendingPathComponent: @"properties.plist"];

3 对象归档

3.1 对象归档

        “归档”是值的另一种形式的序列化,对模型对象进行归档的技术可以轻松将复杂的对象写入文件,然后再从中读取它们,只要在类中实现的每个属性都是基本数据类型(如int或float)或都是符合NSCoding协议的某个类的实例,你就可以对你的对象进行完整归档。

3.2 实现NSCoding协议

        NSCoding协议声明了两个方法:

    - (void) encodeWithCoder: (NSCoder *)aCoder,是将对象写入到文件中。

    - (id) initWithCoder: (NSCoder*)aDecoder,是将文件中数据读入到对象中。

3.3 实现NSCopying协议

        NSCopying协议声明了一个方法: 

    - (id)copyWithZone: (NSZone *)zone ,是将对象复制方法。

Student.h

@interfaceStudent : NSObject

    @property (retain, nonatomic) NSString *studentNo;

    @property (retain, nonatomic) NSString *studentName;

    @property (retain, nonatomic) NSString *studentClass;

@end


Student.m

#import "Student.h"


@implementation Student

@synthesizestudentNo = _studentNo;

@synthesizestudentName = _studentName;

@synthesizestudentClass =_studentClass;


#pragma mark NSCopying

- (id) copyWithZone: (NSZone *)zone {

    Student* copy = [[[self class] allocWithZone: zone] init];

    copy.studentNo = [_studentNo copyWithZone: zone];

    copy.studentName = [_studentName copyWithZone: zone];

    copy.studentClass = [_studentClass copyWithZone: zone];

    return copy;

}


#pragma mark NSCoding

- (void) encodeWithCoder: (NSCoder *)aCoder {

    [aCoder encodeObject: _studentNo forKey: @"studentNo"];

    [aCoder encodeObject: _studentName forKey: @"studentName"];

    [aCoder encodeObject: _studentClass forKey: @"studentClass"];

}

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

    _studentNo = [aDecoder decodeObjectForKey: @"studentNo"];

    _studentName = [aDecoder decodeObjectForKey: @"studentName"];

    _studentClass = [aDecoder decodeObjectForKey: @"studentClass"];

    return self;

}

- (NSString *) description {

    return [[[NSString alloc] initWithFormat: @"no:%@ name: %@ class: %@", _studentNo, _studentName, _studentClass] autorelease];

}


- (void) dealloc {

    [_studentName release];

    [_studentClass release];

    [_studentNo release];

    [super dealloc];

}

@end


EncodingViewController.h

详细见上。

EncodingViewController.m

- (IBAction) saveToArchiver: (id)sender {

    NSString *fileName = [self filePath: @"student.archiver"];

    NSMutableData *data = [NSMutableData data];

    NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData: data];

    Student* student = [[Student alloc] init];

    student.studentNo = studentNo.text;

    student.studentName = studentName.text;

    student.studentClass = studentClass.text;

    [archiver encodeObject: student forKey: @"myStudent"];

    [archiver finishEncoding];

    [data writeToFile: fileName atomically: YES];

    [archiver release];

    [student release];

}


//用于包含编码的数据。

NSMutableData *theData = [NSMutableData data];    

//创建NSKeyedArchiver实例,用于将对象归档到此theData实例中。

NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData: theData];

//使用“键-值”对编码来对希望包含在归档中的对象进行归档。

[archiver encodeObject: student forKey: @"mystudent"]; 

//写入数据到归档文件。

[theData writeToFile: filename atomically: YES]; 


EncodingViewController.m

- (IBAction) loadFromArchiver: (id)sender {

    NSString *fileName = [self filePath: @"student.archiver"];

    NSData *data = [NSData dataWithContentsOfFile: fileName];

    if ([data length] > 0) {

        NSKeyedUnarchiver *unArchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData: data];

        Student *student = [unArchiver decodeObjectForKey: @"myStudent"];

        studentNo.text = student.studentNo;

        studentName.text = student.studentName;

        studentClass.text = student.studentClass;

        [unArchiver finishDecoding];

        [unArchiver release];

    }

}


//从归档文件中获得NSData实例。

NSData *theData = [NSData dataWithContentsOfFile: filename]; 

//创建一个NSKeyedUnarchiver实例对数据进行解码。

NSKeyedUnarchiver *archiver = [[NSKeyedUnarchiver alloc] initForReadingWithData: theData];

//使用与归档编码使用相同的键对象进行解码。

Student *student = [archiver decodeObjectForKey: @"mystudent"];

4 数据库

4.1 SQLite数据库

        SQLite是一个开源的嵌入式关系数据库,它在2000年由D. Richard Hipp发布,它的减少应用程序管理数据的开销,SQLite可移植性好,很容易使用,很小,高效而且可靠。

        SQLite嵌入到使用它的应用程序中,它们共用相同的进程空间,而不是单独的一个进程。从外部看,它并不像一个RDBMS,但在进程内部,它却是完整的,自包含的数据库引擎。 嵌入式数据库的一大好处就是在你的程序内部不需要网络配置,也不需要管理。因为客户端和服务器在同一进程空间运行。SQLite 的数据库权限只依赖于文件系统,没有用户帐户的概念。SQLite 有数据库级锁定,没有网络服务器。它需要的内存,其它开销很小,适合用于嵌入式设备。你需要做的仅仅是把它正确的编译到你的程序。

4.2 SQLite数据类型

        SQLite是无类型的,这意味着你可以保存任何类型的数据到你所想要保存的任何表的任何列中, 无论这列声明的数据类型是什么,对于SQLite来说对字段不指定类型是完全有效的,如:

Create Table ex1(a, b, c);

        SQLite允许忽略数据类型,但是仍然建议在你的Create Table语句中指定数据类型, 因为数据类型对于你和其他的程序员交流, 或者你准备换掉你的数据库引擎。 SQLite支持常见的数据类型, 如:

4.3 在iOS中使用SQLite3

        为了能够在iOS中使用SQLite3需要是将libsqlite3.dylib类库添加到Xcode工程中,在工程的Frameworks(框架) 文件夹右键添加存在Frameworks。

        或者导航到/Developer/Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator.sdk/usr/lib目录下面找到libsqlite3.dylib。

        实例:StudentSQLite3

StudentSQLite3ViewController.h

#import "sqlite3.h"


#define DATA_FILE @"data.sqlite3"

#define TABLE_NAME @"student"

#define FIELDS_NAME_SID @"studentId"

#define FIELDS_NAME_SNAME @"studentName"

#define FIELDS_NAME_SCLASS @"studentClass"


@interface ViewController : UIViewController {

    sqlite3 *db;

}

    @property (retain, nonatomic) IBOutlet UITextField *studentId;

    @property (retain, nonatomic) IBOutlet UITextField *studentName;

    @property (retain, nonatomic) IBOutlet UITextField *studentClass;

- (IBAction) saveFromSqlite: (id)sender;

- (IBAction) loadFromSqlite: (id)sender;

- (NSString *) dataFile;

- (IBAction) textFieldDoneEditing: (id)sender;

@end


StudentSQLite3ViewController.m

@synthesize studentId;

@synthesize studentName;

@synthesize studentClass;


- (NSString *) dataFile {

     NSArray *myPaths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);

    NSString *myDocPath = [myPaths objectAtIndex: 0];

    NSString *fileName = [myDocPath stringByAppendingFormat: DATA_FILE];

    return fileName;

}


        无参数SQLite3处理过程:

    1、打开数据库sqlite3_open。

    2、创建数据库表和执行SQL语句sqlite3_exec。

    3、释放资源sqlite3_close。

4.3.1 创建数据库

- (void) viewDidLoad {

    [super viewDidLoad];

    NSString *fileName = [self dataFile];

    NSLog(@"%@", fileName);

    if(sqlite3_open([fileName UTF8String], &db) != SQLITE_OK) {

        sqlite3_close(db);

        NSAssert(NO, @"OPEN SQLITE DATABASE ERROR!");

    }else{

        char *error;

        NSString *createSQL = [NSString stringWithFormat: @"CREATE TABLE IF NOT EXISTS %@(%@ TEXT PRIMARY KEY, %@ TEXT, %@% TEXT);", TABLE_NAME, FIELDS_NAME_SID, FIELDS_NAME_SNAME, FIELDS_NAME_SCLASS];

        if(sqlite3_exec(db, [createSQL UTF8String], NULL, NULL, &error)) {

            sqlite3_close(db);

            NSAssert1(NO, @"CREATE TABLE ERROR", error);

        }else{

            sqlite3_close(db);  

        }

    }

}


    sqlite3_open([[self dataFilePath] UTF8String], &db)!= SQLITE_OK 

        sqlite3_open打开数据库,注意:在sqlite3中的函数都是使用C字符串[self dataFilePath] UTF8String]是将NSString字符串转换为C字符串,&db是sqlite3指针(* db)的地址。

        该函数sqlite3_open返回SQLITE_OK打开成功。

    sqlite3_exec(db, [tablesql UTF8String], NULL, NULL,&err) != SQLITE_OK

        sqlite3_exec是执行任何不带返回值sql语句,第2个参数是要执行的sql语句,第3个参数是要回调函数,第4个参数是要回调函数的参数,第5个参数是执行出错的字符串。

    sqlite3_close(db); 

        是关闭数据库。

        NSAssert是断言函数,当断言失败时候打印信息。

        NSAssert1是带有一个参数的NSAssert函数,此外还有NSAssert2等函数。

        有参数的SQLite3处理过程:

    1、打开数据库sqlite3_open。

    2、预处理SQL语句sqlite3_prepare_v2。

    3、绑定参数sqlite3_bind_text。

    4、执行语句sqlite3_step(statement) 。

    5、释放资源sqlite3_finalize࿨sqlite3_close。

4.3.2 数据保存

- (IBAction) saveFromSqlite: (id)sender {

    NSString *fileName = [self dataFile];

    NSLog(@"%@", fileName);

    if(sqlite3_open([fileName UTF8String], &db)) {

        sqlite3_close(db);

        NSAssert(NO, @"OPEN DATABASE ERROR");

    }else{

        NSString* sqlStr = [NSString stringWithFormat: @"INSERT OR REPLACE INTO %@(%@, %@, %@) VALUES(?, ?, ?)", TABLE_NAME, FIELDS_NAME_SID, FIELDS_NAME_SNAME, FIELDS_NAME_SCLASS];

        sqlite3_stmt *statement;

        //预处理过程

        if (sqlite3_prepare(db, [sqlStr UTF8String], -1,&statement, NULL) == SQLITE_OK) {

            //绑定参数开始

           sqlite3_bind_text(statement, 1, [studentId.text UTF8String], -1, NULL);

           sqlite3_bind_text(statement, 2, [studentName.text UTF8String], -1, NULL);

           sqlite3_bind_text(statement, 3, [studentClass.text UTF8String], -1, NULL);

            //执行插入

            if(sqlite3_step(statement) != SQLITE_DONE) {

                NSAssert(0, @"INSERT DATABASE ERROR!");

            }

        }

        sqlite3_finalize(statement);

        sqlite3_close(db);

    }

}


    sqlite3_prepare_v2(db, [sqlStr UTF8String], -1,&statement, nil) == SQLITE_OK

        sqlite3_prepare_v2执行sql语句,第3个参数-1代表全部sql字符串长度,第4个参数&statement是sqlite3_stmt指针(* statement)的地址,第5个参数是sql语句没有被执行的部分语句。

    sqlite3_bind_text(statement, 1, [studentId.textUTF8String], -1, NULL);

        是绑定参数,第2个参数为序号(从1开始),第3个参数为字符串值,第4个参数为字符串长度。 第5个参数为一个函数指针,SQLITE3执行完操作后回调此函数,通常用于释放字符串占用的内存。

        sqlite3_step(statement) != SQLITE_DONE判断是否执行完成sql语句执行。

        sqlite3_finalize(statement)和sqlite3_close(db)释放资源。

4.3.3 查询数据

- (IBAction) loadFromSqlite: (id)sender {

    NSString *fileName = [self dataFile];

    NSLog(@"%@", fileName);

    if(sqlite3_open([fileName UTF8String], &db) != SQLITE_OK) {

        sqlite3_close(db);

        NSAssert(NO, @"OPEN DATABASE ERROR!");

    }else{

        NSString *sqlStr = [NSString stringWithFormat: @"SELECT %@,%@,%@ FROM %@ WHERE %@=?", FIELDS_NAME_SID, FIELDS_NAME_SNAME, FIELDS_NAME_SCLASS, TABLE_NAME, FIELDS_NAME_SID];

        sqlite3_stmt *statement;

        //预处理过程

        if (sqlite3_prepare_v2(db, [sqlStr UTF8String], -1, &statement, NULL) == SQLITE_OK) {

            //绑定参数开始

           sqlite3_bind_text(statement, 1, "1000", -1, NULL);

            //执行

            while(sqlite3_step(statement) == SQLITE_ROW) {

                char *field1 = (char*)sqlite3_column_text(statement, 0);

                NSString *field1Str= [[NSString alloc] initWithUTF8String: field1];

                studentId.text = field1Str;


                char *field2 = (char*)sqlite3_column_text(statement, 1);

                NSString *field2Str= [[NSString alloc] initWithUTF8String: field2];

                studentName.text = field2Str;


                char *field3 = (char*)sqlite3_column_text(statement, 2);

                NSString *field3Str = [[NSString alloc] initWithUTF8String: field3];

                studentClass.text = field3Str;


                [field1Str release];

                [field2Str release];

                [field3Str release];

            }

        }

        sqlite3_finalize(statement);

        sqlite3_close(db);

    }

}


    while (sqlite3_step(statement) == SQLITE_ROW)

        sqlite3_step(statement) == SQLITE_ROW单步执行并判断sql语句执行的状态。

    char *field1 = (char *) sqlite3_column_text(statement, 0); 

        sqlite3_column_text(statement, 0);取出字段值,第2个参数是列的顺序,序号是从0开始。

NSString *field1Str = [[NSString alloc] initWithUTF8String: field1];构建NSSting字符串。


        其它部分代码

- (IBAction) textFieldDoneEditing: (id)sender {

    [sender resignFirstResponder];

}

- (void) viewDidUnload

{

    [self setStudentId: nil];

    [self setStudentName: nil];

    [self setStudentClass: nil];

    [super viewDidUnload];

}


- (void) dealloc {

    [studentId release];

    [studentName release];

    [studentClass release];

    [super dealloc];

}

5 参考链接

IOS之数据持久化

http://www.cnblogs.com/syxchina/archive/2012/09/17/2689830.html


IOS ---对象归档

http://blog.sina.com.cn/s/blog_45e2b66c0101evyr.html

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 193,968评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,682评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,254评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,074评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,964评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,055评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,484评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,170评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,433评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,512评论 2 308
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,296评论 1 325
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,184评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,545评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,150评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,437评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,630评论 2 335

推荐阅读更多精彩内容