1.简介

本文将介绍IOS的开发过程中如何集成Sqlite的方法,目前Sqlite的版本为3,所以我们称之为Sqlite3.

在本文中我将介绍Sqlite3的开发配置,本地Sqlite3数据库的建立通用的数据库访问类。

2.Sqlite3的开发环境配置

1.在工程中引入SQlite3开发库。

按照下图进行设定:

3.建立Sqlite数据库和表

1.打开终端窗口,输入如下命令:

172-11-253-106:~ apple$ sqlite3 exampleSqlite3_db.sql

SQLite version 3.8.5 2014-08-15 22:37:57

Enter ".help" for usage hints.

sqlite>

如上我们建立了一个名为exampleSqlite3_db的数据库。

2.建表

CREATE TABLE peopleInfo(peopleInfoID integer primary key, firstname text, lastname text, age integer);

表建立完毕后我们可以通过命令查询表是否建立成功:

sqlite> .tables
peopleInfo
sqlite>

随后退出:

sqlite> .quit
172-11-253-106:~ apple$

3. 至此本地所需要的数据库和表初始化完毕,将文件导入当前工程

4.至此本地数据库和数据库连接文件都已经创建完毕。

4.建立通用的数据库访问类

1.建立数据库model

#import <Foundation/Foundation.h>

@interface PeopleInfo : NSObject
@property(nonatomic, assign) NSInteger peopleInfoID;
@property(nonatomic, strong) NSString * firstname;
@property(nonatomic, strong) NSString * lastname;
@property(nonatomic, assign) NSInteger age;

@end

2.建立通用的Sqlite3访问类

.h文件

//
//  DBManager.h
//  how_to_playwith_Sqlite3
//
//  Created by apple on 15/12/14.
//  Copyright © 2015年 xufeng. All rights reserved.
//

#import <Foundation/Foundation.h>

/**
 *  Sqlite3访问类
 */
@interface DBManager : NSObject

/**
 *  sql执行后受影响的行数,如delete、insert和update的时候可以以此判断数据库是否操作正常
 */
@property (nonatomic) int affectedRows;

/**
 *  记录最后一次的操作索引
 */
@property (nonatomic) long long lastInsertedRowID;

/**
 *  加载数据库文件
 *
 *  @param dbFilename 沙盘中的数据库文件
 *
 *  @return 数据库操作handler
 */
-(instancetype)initWithDatabaseFilename:(NSString *)dbFilename;

/**
 *  执行查询操作
 *
 *  @param query 查询语句
 *
 *  @return 查询结果
 */
-(NSArray *)loadDataFromDB:(NSString *)query;

/**
 *  执行增删和更新的语句
 *
 *  @param query 执行语句
 */
-(void)executeQuery:(NSString *)query;
@end

.m文件

//
//  DBManager.m
//  how_to_playwith_Sqlite3
//
//  Created by apple on 15/12/14.
//  Copyright © 2015年 apple. All rights reserved.
//

#import "DBManager.h"
#import <sqlite3.h>

@interface DBManager()

/**
 *  应用的Documents Directory
 */
@property (nonatomic, strong) NSString *documentsDirectory;

/**
 *  数据库文件
 */
@property (nonatomic, strong) NSString *databaseFilename;

/**
 *  结果数组
 */
@property (nonatomic, strong) NSMutableArray *arrResults;

/**
 *  将bundle中的文件拷贝到Documents Directory
 */
-(void)copyDatabaseIntoDocumentsDirectory;

/**
 *  执行SQLite语句
 *
 *  @param query           SQLite语句
 *  @param queryExecutable 查询还是修改
 */
-(void)runQuery:(const char *)query isQueryExecutable:(BOOL)queryExecutable;
@end

@implementation DBManager

-(instancetype)initWithDatabaseFilename:(NSString *)dbFilename{
    self = [super init];
    if (self) {
        // 查找Documents Directory路径
        NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
        self.documentsDirectory = [paths objectAtIndex:];

        // 保存传入的数据库文件
        self.databaseFilename = dbFilename;

        // 将bundle中的文件拷贝到Documents Directory
        [self copyDatabaseIntoDocumentsDirectory];
    }
    return self;
}

-(void)copyDatabaseIntoDocumentsDirectory{
    // 查询数据库文件是否在ocuments Directory已经存在?
    NSString *destinationPath = [self.documentsDirectory stringByAppendingPathComponent:self.databaseFilename];
    if (![[NSFileManager defaultManager] fileExistsAtPath:destinationPath]) {
        // 如果不不存在则从bundle中将数据库文件拷贝到Documents Directory
        NSString *sourcePath = [[[NSBundle mainBundle] resourcePath] stringByAppendingPathComponent:self.databaseFilename];
        NSError *error;
        [[NSFileManager defaultManager] copyItemAtPath:sourcePath toPath:destinationPath error:&error];

        // 查看是够有错误
        if (error != nil) {
            NSLog(@"%@", [error localizedDescription]);
        }
    }
}

-(NSArray *)loadDataFromDB:(NSString *)query{
    // 执行查询
    [self runQuery:[query UTF8String] isQueryExecutable:NO];

    // 返回查询结果
    return (NSArray *)self.arrResults;
}

-(void)executeQuery:(NSString *)query{
    // 执行修改操作
    [self runQuery:[query UTF8String] isQueryExecutable:YES];
}

-(void)runQuery:(const char *)query isQueryExecutable:(BOOL)queryExecutable
{
    // 数据库连接变量
    sqlite3 *sqlite3Database;

    // 设定数据库文件
    NSString *databasePath = [self.documentsDirectory stringByAppendingPathComponent:self.databaseFilename];

    // 重置结果容器
    if (self.arrResults != nil) {
        [self.arrResults removeAllObjects];
        self.arrResults = nil;
    }
    self.arrResults = [[NSMutableArray alloc] init];

    // 打开数据库
    int openDatabaseResult = sqlite3_open([databasePath UTF8String], &sqlite3Database);
    if(openDatabaseResult == SQLITE_OK) {
        // Declare a sqlite3_stmt object in which will be stored the query after having been compiled into a SQLite statement.
        sqlite3_stmt *compiledStatement;

        // Load all data from database to memory.
        , &compiledStatement, NULL);
        if(prepareStatementResult == SQLITE_OK) {
            // Check if the query is non-executable.
            if (!queryExecutable){
                // In this case data must be loaded from the database.

                // Declare an array to keep the data for each fetched row.
                NSMutableArray *arrDataRow;

                // Declare an array to keep the col name for each fetched row.
                NSMutableArray *arrColRow;

                // Loop through the results and add them to the results array row by row.
                while(sqlite3_step(compiledStatement) == SQLITE_ROW) {
                    // Initialize the mutable array that will contain the data of a fetched row.
                    arrDataRow = [[NSMutableArray alloc] init];
                    arrColRow = [[NSMutableArray alloc] init];

                    // Get the total number of columns.
                    int totalColumns = sqlite3_column_count(compiledStatement);

                    // Go through all columns and fetch each column data.
                    ; i<totalColumns; i++){
                        // Convert the column data to text (characters).
                        char *dbDataAsChars = (char *)sqlite3_column_text(compiledStatement, i);
                        char *dbColAsChars = (char *)sqlite3_column_name(compiledStatement, i);

                        // If there are contents in the currenct column (field) then add them to the current row array.
                        if (dbDataAsChars != NULL) {
                            // Convert the characters to string.
                            [arrDataRow addObject:[NSString  stringWithUTF8String:dbDataAsChars]];
                            [arrColRow addObject:[NSString  stringWithUTF8String:dbColAsChars]];
                        }
                    }

                    // Store each fetched data row in the results array, but first check if there is actually data.
                    ) {
                        // first of all, we should put them in dictionary
                        [self.arrResults addObject:[NSDictionary dictionaryWithObjects:arrDataRow forKeys:arrColRow]];
                    }
                }
            }
            else {
                // This is the case of an executable query (insert, update, ...).

                // Execute the query.
                int executeQueryResults = sqlite3_step(compiledStatement);
                if (executeQueryResults == SQLITE_DONE) {
                    // Keep the affected rows.
                    self.affectedRows = sqlite3_changes(sqlite3Database);

                    // Keep the last inserted row ID.
                    self.lastInsertedRowID = sqlite3_last_insert_rowid(sqlite3Database);
                }
                else {
                    // If could not execute the query show the error message on the debugger.
                    NSLog(@"DB Error: %s", sqlite3_errmsg(sqlite3Database));
                }
            }
        }
        else {
            // In the database cannot be opened then show the error message on the debugger.
            NSLog(@"%s", sqlite3_errmsg(sqlite3Database));
        }

        // Release the compiled statement from memory.
        sqlite3_finalize(compiledStatement);
    }

    // Close the database.
    sqlite3_close(sqlite3Database);
}
@end

3. 测试(先插入后查询)

插入如下测试语句:

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.

    // 数据库客户端操作-打开
    DBManager *dbm = [[DBManager alloc] initWithDatabaseFilename:@"exampleSqlite3_db.sql"];

    // 数据库客户端操作-插入s
    [dbm executeQuery:@"insert into peopleInfo values(100,'zhang','san',20)"];

     != dbm.affectedRows) {
        NSLog(@"%@", @"error happend!");
    }
    else{
        // 数据库客户端操作-查询
        NSArray *result = [dbm loadDataFromDB:@"select * from peopleInfo"];

        // 每行既是一个item,将其转化为具体的对象
        PeopleInfo *peopleInfo = nil;
        NSMutableArray *resultArray = [[NSMutableArray alloc] init];
        for (NSDictionary *rowDict in result)
        {
            peopleInfo = [[PeopleInfo alloc] init];
            // 使用方法的时候必须model的属性名称和字典的key要保持完全一致
            [peopleInfo setValuesForKeysWithDictionary:rowDict];
            [resultArray addObject:peopleInfo];
        }

        NSLog(@"%@", resultArray);
    }
}

结果:

至此Sqlite3的集成方法介绍完毕。

[How to]集成SQLite3的更多相关文章

  1. delphi Sqlite

    Delphi中SQLite如何读写二进制字段(Blob类型) 在Delphi中,有大量的组件可以操作SQLite数据库,如UniDAC就是其中一个比较优秀的,当然还有ASQLite3Component ...

  2. 从零开始用electron整个跨平台桌面应用---基础配置篇

    1.安装node.npm node以及npm都需要是最新版本(版本过低有坑) 2.安装淘宝镜像cnpm(建议,下载较快) npm install -g cnpm --registry=https:// ...

  3. electron 集成 nedb / sqlite3

    nedb nedb 无法创建文件 // webpack 构建的前提 externals: process.env.web ? {} : { "nedb": "requir ...

  4. Python开发入门与实战9-基于vs的集成开发环境

    9. 基于visual studio的Python的集成开发环境 上一章我们描述了如何安装使用Java的集成开发环境Eclipse IDE,本章我们来说明另一种集成开发环境,也是笔者多年的开发习惯使用 ...

  5. 第三十三章 metrics(1) - graphite搭建 + whisper存储模式 + 高精度向低精度聚合方式 + 集成StatsD + 集成grafana

    组件介绍: carbon:Carbon实际上是一系列守护进程,组成一个Graphite安装的存储后端.这些守护进程用一个名为Twisted的事件驱动网络引擎监听时间序列数据.Twisted框架让Car ...

  6. 【解决】Django项目废弃SQLite3拥抱MySQL

    SQLite3数据库就一个文件,拷贝着随时带走,调试方便,超级轻量级,有它的好处. 不过,MySQL才是中小项目的主流,最近想把Django里程碑项目部署到SAE上,所以试着把原来的项目数据库替换成M ...

  7. Android adb使用sqlite3对一个数据库进行sql查询

    sqlite是Android下集成的一个轻量级数据库,我们可以通过adb程序进入数据库命令行,对数据进行查询,具体操作如下: ①打开windows的cmd ②输入adb shell.此时进入了该安卓系 ...

  8. Cloud Foundry中 JasperReports service集成

    Cloud Foundry作为业界第一个开源的PaaS解决方案,正越来越多的被业界接受和认可.随着PaaS的发展,Cloud Foundry顺应潮流,充分发挥开源项目的特点,到目前为止,已经支持了大批 ...

  9. apache+php+mysql常见集成环境安装包

    http://www.thinksaas.cn/group/topic/33/ apache+php+mysql是常见php环境,在windows下也称为WAMP,对于初学者自选版本搭建总是会遇到一些 ...

随机推荐

  1. CentOS 文件搜索find

    1.文件搜索,内置的的命令是find 用法: find [查找路径] 寻找条件 操作 默认路径为当前目录:默认表达式为 -print 2.主要参数: -name 匹配名称 -perm 匹配权限(mod ...

  2. Elasticsearch Query DSL备忘(1)(Constant score query和Bool Query)

    Query DSL (Domain Specific Language),基于json的查询方式 1.Constant score query,常量分值查询,目的就是返回指定的score,一般都结合f ...

  3. [LOJ2540] [PKUWC2018] 随机算法

    题目链接 LOJ:https://loj.ac/problem/2540 Solution 写的时候脑子不太清醒码了好长然后时间\(LOJ\)垫底... 反正随便状压\(dp\)一下就好了,设\(f[ ...

  4. 【以前的空间】bzoj 1072 [SCOI2007]排列perm

    又颓废了一个下午,最近撸mc撸到丧失意识了,玩的有点恶心,于是找水题做,瞧不起颓废的自己啊. another水题. 这题题意很明显啦,就是找数字排列后组成的数去mod d=0后有多少种. 普通的搜索的 ...

  5. HDU - 6333 Harvest of Apples

    题意: T次询问,每次给出n,m.求sigma(k:0->m)C(n, k). 题解: 用离线莫队来做. 令S(n,m) = sigma(k:0->m)C(n, k). S(n+1, m) ...

  6. ubuntu简易教程(如何使用noi linux)

    目录 linux环境下的基础操作 命令行操作 编辑器 程序编译 程序调试 gdb的使用 对拍 在提高组的考试中要求使用noi linux,因此了解一下如何在linux环境下编程是很有必要的. linu ...

  7. [bzoj] 3669 NOI2014 魔法森林 || LCT

    原题 copy一篇题解:原链接 将边按照a排序,然后从小到大枚举,加入图中. 在图中用lct维护一棵两点之间b最大值尽量小的生成树. 当加入一条边(u, v)时: 如果(u, v)不连通,则直接加入这 ...

  8. HDOJ(HDU).1045 Fire Net (DFS)

    HDOJ(HDU).1045 Fire Net [从零开始DFS(7)] 点我挑战题目 从零开始DFS HDOJ.1342 Lotto [从零开始DFS(0)] - DFS思想与框架/双重DFS HD ...

  9. 使用javaScript和JQuery制作经典面试题:光棒效果

    使用javaScript与jQuery添加CSS样式的区别和步骤 使用javaScript制作光棒效果 --首先是javaScript <script> $(function () { v ...

  10. JS实现的随机乱撞的彩色圆球特效代码

    <!doctype html> <html lang="en"> <head> <meta charset="UTF-8&quo ...