很多时候我们在做App的时候,需要对一些数据做缓存,在没网络支持的情况下给用户展示一些效果,这样的需求在iOS开发中是非常常见的,qpple给我们提供了一个非常好用的数据库:SQLite,它是一个轻量级关系数据库,最初的设计目标是用于嵌入式系统,它占用资源非常少.在iOS中,只需要加入li’blibsqlite3.0依赖以及引入sqlite3.h头文件即可.
SQLite是无类型的数据库,可以保存任何类型的数据,对于SQLite来说对字段不指定类型是完全有效的,学过数据库的人都知道,存储在数据库中的每一条数据都必须要有PrimaryKey,这里设计到一个非常重要的概念:主键
数据表中每一条记录都有一个主键,这就像我们的身份证号码一样,反过来说每一个主键对应着一条数据记录,所以,主键必须是唯一的,一般情况主键同时也是一个索引,所以通过主键查找记录速度比较快,在关系类型库中,一个表的主键可以作为另外一个表的外键,这样,这两个表之间就通过这个键建立了关系,主键一般是整数或者字符串,只要保证唯一就行,在SQLite中,主键如果是整型类型,该列的值可以自动增长,
所有的操作都是通过sql语句来进行操作的,包括建表、数据的增删差改、表关联等等,至于sql语句在这里就不做介绍了,这里我们简单的实现一下存储数据并进行增删查改操作,废话不多说,看代码:
- (void)sqliteAction {
NSString *documentPath = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) firstObject];
NSString *dbPath = [documentPath stringByAppendingPathComponent:@"student.sqlite"];
int result =sqlite3_open([dbPath UTF8String], &_dataBase);
if (result == SQLITE_OK) {
NSLog(@"数据库打开成功!");
//创建表
char *ERROR = NULL;
//建表
NSString *sqlStr = @"create table if not exists 'student' ('number' integer primary key autoincrement not null,'name' text,'sex' text,'age' integer)";
if (sqlite3_exec(_dataBase, [sqlStr UTF8String], NULL, NULL, &ERROR) == SQLITE_OK) {
NSLog(@"创建表成功!");
//插入
for (int i = 0; i< 100; i ++) {
NSString *sqlStr =[NSString stringWithFormat:@"INSERT OR REPLACE INTO student(name,sex,age) VALUES('%@','%@','%d')",[NSString stringWithFormat:@"Chan%d",i],
i %2 == 0 ? @"男":@"女",
100 +i];
if (sqlite3_exec(_dataBase, [sqlStr UTF8String], NULL, NULL, &ERROR) == SQLITE_OK) {
NSLog(@"插入成功!");
}
}
//查询
NSString *selectStr = @"SELECT * FROM student";
sqlite3_stmt *stmt;
if (sqlite3_prepare(_dataBase, [selectStr UTF8String], -1, &stmt, nil) == SQLITE_OK) {
while (sqlite3_step(stmt) == SQLITE_ROW ) {
NSString *name = [NSString stringWithUTF8String:(char*) sqlite3_column_text(stmt, 1)];
NSLog(@"name:%@",name);
NSString *sex = [NSString stringWithUTF8String:(char *)sqlite3_column_text(stmt, 2)];
NSLog(@"sex:%@",sex);
NSInteger age = sqlite3_column_int(stmt, 3);
NSLog(@"%zd",age);
/* StudentModel *model = [StudentModel new];
model.name = name;
model.sex = sex;
model.age = age;
[_dataArray addObject: model];*/
}
if (sqlite3_step(stmt) == SQLITE_DONE) {
//释放句柄
sqlite3_finalize(stmt);
}
}
} else {
NSLog(@"创建表失败!");
}
} else {
NSLog(@"数据库打开失败!");
sqlite3_close(_dataBase);
}
}
- (void)openSqlite {
//判断数据库是否为空,如果不为空说明已经打开
if(_dataBase != nil) {
NSLog(@"数据库已经打开");
return;
}
//获取文件路径
NSString *str = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) firstObject];
NSString *strPath = [str stringByAppendingPathComponent:@"my.sqlite"];
NSLog(@"%@",strPath);
//打开数据库
//如果数据库存在就打开,如果不存在就创建一个再打开
int result = sqlite3_open([strPath UTF8String], &_dataBase);
//判断
if (result == SQLITE_OK) {
NSLog(@"数据库打开成功");
} else {
NSLog(@"数据库打开失败");
}
}
- (void)createTable {
NSString *sqlite = [NSString stringWithFormat:@"create table if not exists 'student' ('number' integer primary key autoincrement not null,'name' text,'sex' text,'age'integer)"];
char *error = NULL;//执行sqlite语句失败的时候,会把失败的原因存储到里面
int result = sqlite3_exec(_dataBase, [sqlite UTF8String], nil, nil, &error);
if (result == SQLITE_OK) {
NSLog(@"创建表成功");
} else {
NSLog(@"创建表失败");
}
}
//添加数据
- (void)addStudent:(Student *)stu {
NSString *sqlite = [NSString stringWithFormat:@"insert into student(number,name,age,sex) values ('%ld','%@','%@','%ld')",stu.number,stu.name,stu.sex,stu.age];
char *error = NULL;//执行sqlite语句失败的时候,会把失败的原因存储到里面
int result = sqlite3_exec(_dataBase, [sqlite UTF8String], nil, nil, &error);
if (result == SQLITE_OK) {
NSLog(@"添加数据成功");
} else {
NSLog(@"添加数据失败");
}
}
//删除数据
- (void)delete:(Student*)stu {
//1.准备sqlite语句
NSString *sqlite = [NSString stringWithFormat:@"delete from student where number = '%ld'",stu.number];
//2.执行sqlite语句
char *error = NULL;//执行sqlite语句失败的时候,会把失败的原因存储到里面
int result = sqlite3_exec(_dataBase, [sqlite UTF8String], nil, nil, &error);
if (result == SQLITE_OK) {
NSLog(@"删除数据成功");
} else {
NSLog(@"删除数据失败%s",error);
}
}
//修改数据
- (void)updataWithStu:(Student *)stu {
//1.sqlite语句
NSString *sqlite = [NSString stringWithFormat:@"update student set name = '%@',sex = '%@',age = '%ld' where number = '%ld'",stu.name,stu.sex,stu.age,stu.number];
//2.执行sqlite语句
char *error = NULL;//执行sqlite语句失败的时候,会把失败的原因存储到里面
int result = sqlite3_exec(_dataBase, [sqlite UTF8String], nil, nil, &error);
if (result == SQLITE_OK) {
NSLog(@"修改数据成功");
} else {
NSLog(@"修改数据失败");
}
}
//查询所有数据
- (NSMutableArray*)selectWithStu {
NSMutableArray *array = [[NSMutableArray alloc] init];
NSString *sqlite = [NSString stringWithFormat:@"select * from student"];
sqlite3_stmt *stmt = NULL;
int result = sqlite3_prepare(_dataBase, sqlite.UTF8String, -1, &stmt, NULL);//第4个参数是一次性返回所有的参数,就用-1
if (result == SQLITE_OK) {
NSLog(@"查询成功");
//4.执行n次
while (sqlite3_step(stmt) == SQLITE_ROW) {
Student *stu = [[Student alloc] init];
//从伴随指针获取数据,第0列
stu.number = sqlite3_column_int(stmt, 0);
//从伴随指针获取数据,第1列
stu.name = [NSString stringWithUTF8String:(const char *)sqlite3_column_text(stmt, 1)] ;
//从伴随指针获取数据,第2列
stu.sex = [NSString stringWithUTF8String:(const char *)sqlite3_column_text(stmt, 2)] ;
//从伴随指针获取数据,第3列
stu.age = sqlite3_column_int(stmt, 3);
}
} else {
NSLog(@"查询失败");
}
sqlite3_finalize(stmt);
return array;
}
- (void)closeSqlite {
if (sqlite3_close(_dataBase) == SQLITE_OK) {
NSLog(@"数据库关闭成功");
} else {
NSLog(@"数据库关闭失败");
}
}
运行完成之后,我们可以在沙盒文件系统里面看到创建的数据库文件:
我们可以使用sqlite工具打开这个student.sqlite数据库文件查看表结构:
表中的数据: