CoreData
- CoreData数据持久化框架是Cocoa API的一部分,首次在iOS5版本中出现,它允许按照实体-属性-值模式组织数据,并以XML,二进制文件或者SQLite数据文件的格式持久化数据.
- CoreData主要提供的对象-关系映射(ORM)功能,把OC对象转化为数据保存到文件,也可以数据转化为OC对象
CoreData和SQLite的比较
- SQLite
1 基于C接口,需要使用sql语句,代码繁琐
2 在处理大量数据时,表关系更直观
3 在OC中是不是可视化的 - CoreData
1 可视化,有undo/redo能力
2 可以实现多种文件格式 NSSQLiteStoreType,NSBinaryStoreType,NSInMemoryStore Type,NSXMLStoreType.
3.苹果官方API支持,与iOS结合更紧密
CoreData核心对象
- NSManagedObjectContext 数据管理器器类(负责应用和数据库之间的交互)
- NSManagedObjectModel 数据模型器类(当对于可视化创建model的基类)
- NSPersistentStoreCoordinator 数据协调器类 (添加持久化数据库)
- NSEntityDescription 实体描述类(用来描述实体类)
CoreData数据库框架的核心对象
- NSPersistentStore:持久化存储,一个被封装好的底层类,用于存储数据
- 存储文件:用来存储和管理数据的文件
1 NSSQLiteStoreType
2 NSBinaryStoreType
3 NSInMemoryStoreType
4 NSXMLStoreType. - NSManagedObjectContext:被管理对象上下文CoreData中用于操作和使用数据,负责应用和数据库之间的交互
1 数据的保存需要NSManagedObjectContext进行save操作
2 数据的查询需要NSManagedObjectContext进行executeFetchRequest操作(返回值是数组)
3 CoreData提供的是对象关系映射,NSManagedObjectContext操作的都是NSManagedObject对象 - NSManagedObjectMode:被管理对象对象模型,管理多个对象
- NSManagedObject:被管理对象,CoreData返回的数据模型,被管理的对象是根据实体描述生成的
- NSEntityDescription:实体描述类,根据实体创建被管理对象
- Entity:实体类,实体是对象文件数据的描述,被管理对象表示实体,实体包含名称,属性(字段)和关系,实体的名称通常和被管理对象名一致
- NSFetchRequest:查询请求,NSManagedObjectContext根据NSFetchRequest查询数据,以数组形式返回,数组中包含被管理对象(NSManagedObject)
- NSSortDescriptor:排序操作
CoreData数据库的简单操作
第一步:添加AppDelegate头文件
#import "AppDelegate.h"
第二步:通过单例方法获取协议
AppDelegate *delegate = [UIApplication sharedApplication].delegate;
第三步:创建实体对象
//.创建实体对象(也就是具体对那个表进行操作)
NSEntityDescription *entity = [NSEntityDescription entityForName:@"Person" inManagedObjectContext:delegate.managedObjectContext];
第四步:增删查改
/**增**/
//在创建的时候要使用coreData提供的方法,这样数据管理器就可以管理此数据了
Person *person = [[Person alloc] initWithEntity:entity insertIntoManagedObjectContext:delegate.managedObjectContext];
//同步到数据库
person.name = @"张三";
person.sex = @"nan";
person.age = 12;
[delegate saveContext];
/**删**/
// 创建请求体
NSFetchRequest *request1 = [[NSFetchRequest alloc] init];
// //请求需要的具体实体
request1.entity = entity;
// //创建请求条件
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"age > 1"];
request1.predicate = predicate;
NSArray *array1 = [delegate.managedObjectContext executeFetchRequest:request1 error:nil];
for (Person *person2 in array1) {
[delegate.managedObjectContext deleteObject:person2];
NSLog(@"%@",person2.name);
}
[delegate saveContext];
/**查**/
//创建请求体
NSFetchRequest *request = [[NSFetchRequest alloc] init];
//请求体需要的具体实体
request.entity = entity;
NSArray *array = [delegate.managedObjectContext executeFetchRequest:request error:nil];
for (Person *person1 in array) {
NSLog(@"%@",person1.name);
NSLog(@"%p",person1);
}
/**改**/
//创建请求体
NSFetchRequest *request2 = [[NSFetchRequest alloc] init];
request2.entity = entity;
//创建请求条件
NSPredicate *predicate1 = [NSPredicate predicateWithFormat:@"name = %@",@"张三"];
request2.predicate = predicate1;
NSArray *array2 = [delegate.managedObjectContext executeFetchRequest:request2 error:nil];
for (Person *person3 in array2) {
person3.name = @"李四";
NSLog(@"%@",person3.name);
}
[delegate saveContext];
下面是对CoreData简单封装的代码(使用了FMDB)
#import <Foundation/Foundation.h>
@interface FMDBHelp : NSObject
+ (FMDBHelp*)sharedFMDBHelp;
//给数据库命名
- (void)createDBWithName:(NSString*)dbName;
//无返回结果集的操作
- (BOOL)notResultSetWithSql:(NSString*)sql;
//查询操作
- (NSArray*)qureyWithSql:(NSString*)sql;
@end
#import "FMDBHelp.h"
#import <FMDB/FMDB.h>
@interface FMDBHelp ()
@property(nonatomic,strong)NSString *fileName;//数据库文件的路径
@property(nonatomic,strong)FMDatabase *database; //数据库对象
@end
@implementation FMDBHelp
#pragma mark - 单例
+ (FMDBHelp*)sharedFMDBHelp {
static FMDBHelp *help = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
help = [[FMDBHelp alloc] init];
});
return help;
}
#pragma mark - 让用户来命名数据库的名称
- (void)createDBWithName:(NSString*)dbName {
if (dbName.length == 0) {
//是防止用户直接传值为nil 或者 NULL
self.fileName = @"";
} else {
//判断用户是否为数据库文件添加后缀名
if ([dbName containsString:@".sqlite"]) {
self.fileName = dbName;
} else {
self.fileName = [dbName stringByAppendingString:@".sqlite"];
}
}
}
#pragma amrk - 根据名称创建沙盒路径用来保存数据库文件
- (NSString*)dbPath {
//说明fileName不为空
if (self.fileName.length) {
//获取沙盒主路径
NSString *homePath = NSHomeDirectory();
//完整路径
NSString *savePath = [homePath stringByAppendingPathComponent:[NSString stringWithFormat:@"documents/%@",self.fileName]];
NSLog(@"%@",savePath);
return savePath;
} else {
return @"";
}
}
#pragma mark - 创建数据库对象
//懒加载
- (FMDatabase*)database {
if (!_database) {
_database = [FMDatabase databaseWithPath:[self dbPath]];
}
return _database;
}
#pragma mark - 打开或者创建数据库
- (BOOL)openOrCreateDB {
if ([self.database open]) {
NSLog(@"数据库打开成功");
return YES;
} else {
NSLog(@"数据库打开失败");
return NO;
}
}
#pragma mark - 无返回结果集的操作
- (BOOL)notResultSetWithSql:(NSString*)sql {
//打开数据库
BOOL isOpen = [self openOrCreateDB];
if (isOpen) {
//进行操作
BOOL isSuccess = [self.database executeUpdate:sql];
[self closeDB];
NSLog(@"打开数据库成功");
return isSuccess;
} else {
NSLog(@"打开数据库失败");
return NO;
}
}
#pragma mark - 关闭数据库的方法
- (void)closeDB {
BOOL isClose = [self.database close];
if (isClose) {
NSLog(@"关闭数据库成功");
} else {
NSLog(@"关闭数据库失败");
}
}
#pragma mark - 通用的查询方法
- (NSArray*)qureyWithSql:(NSString*)sql {
//打开数据库
BOOL isOpen = [self openOrCreateDB];
if (isOpen) {
//得到所有记录的结果集
FMResultSet *set = [self.database executeQuery:sql];
//声明一个可变数组,用来存放所有的记录
NSMutableArray *array = [NSMutableArray array];
//遍历结果集,取出每一条记录,将每一条记录转换为字典类型,并且存储到可变数组中
while ([set next]) {
//直接将一条记录转换为字典类型
NSDictionary *dic = [set resultDictionary];
[array addObject:dic];
}
//释放结果集
[set close];
[self closeDB];
return array;
} else {
NSLog(@"打开数据库失败");
return nil;
}
}