iOS开发之轻量级数据库SQLite3(增删改查)

很多时候我们在做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.png
sqlite Query.png

我们可以使用sqlite工具打开这个student.sqlite数据库文件查看表结构:


SQLite_table.png

表中的数据:


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

推荐阅读更多精彩内容