iOS学习笔记(十六)——详解数据库操作(使用FMDB)

iOS中原生的SQLite API在使用上相当不友好,在使用时,非常不便。于是,就出现了一系列将SQLite API进行封装的库,例如FMDB、PlausibleDatabase、sqlitepersistentobjects等,FMDB (https://github.com/ccgus/fmdb) 是一款简洁、易用的封装库,这一篇文章简单介绍下FMDB的使用。

在FMDB下载文件后,工程中必须导入如下文件,并使用 libsqlite3.dylib 依赖包。

FMDB同时兼容ARC和非ARC工程,会自动根据工程配置来调整相关的内存管理代码。

FMDB常用类:

  1. FMDatabase : 一个单一的SQLite数据库,用于执行SQL语句。
  2. FMResultSet :执行查询一个FMDatabase结果集,这个和Android的Cursor类似。
  3. FMDatabaseQueue :在多个线程来执行查询和更新时会使用这个类。

创建数据库:

db = [FMDatabase databaseWithPath:database_path]; 

1、当数据库文件不存在时,fmdb会自己创建一个。

2、 如果你传入的参数是空串:@"" ,则fmdb会在临时文件目录下创建这个数据库,数据库断开连接时,数据库文件被删除。

3、如果你传入的参数是 NULL,则它会建立一个在内存中的数据库,数据库断开连接时,数据库文件被删除。

打开数据库:

[db open] 

返回BOOL型。

关闭数据库:

[db close] 

数据库增删改等操作:

除了查询操作,FMDB数据库操作都执行executeUpdate方法,这个方法返回BOOL型。

看一下例子:

创建表:

if ([db open]) {
    NSString *sqlCreateTable = [NSString stringWithFormat:@"CREATE TABLE IF NOT EXISTS '%@' ('%@' INTEGER PRIMARY KEY AUTOINCREMENT, '%@' TEXT, '%@' INTEGER, '%@' TEXT)",TABLENAME,ID,NAME,AGE,ADDRESS];
    BOOL res = [db executeUpdate:sqlCreateTable];
    if (!res) {
      NSLog(@"error when creating db table");
    } else {
      NSLog(@"success to creating db table");
    }
    [db close]; 

  } 

添加数据:

if ([db open]) {
    NSString *insertSql1= [NSString stringWithFormat:
               @"INSERT INTO '%@' ('%@', '%@', '%@') VALUES ('%@', '%@', '%@')",
               TABLENAME, NAME, AGE, ADDRESS, @"张三", @"13", @"济南"];
    BOOL res = [db executeUpdate:insertSql1];
    NSString *insertSql2 = [NSString stringWithFormat:
                @"INSERT INTO '%@' ('%@', '%@', '%@') VALUES ('%@', '%@', '%@')",
                TABLENAME, NAME, AGE, ADDRESS, @"李四", @"12", @"济南"];
    BOOL res2 = [db executeUpdate:insertSql2]; 

    if (!res) {
      NSLog(@"error when insert db table");
    } else {
      NSLog(@"success to insert db table");
    }
    [db close]; 

  } 

修改数据:

if ([db open]) {
    NSString *updateSql = [NSString stringWithFormat:
                @"UPDATE '%@' SET '%@' = '%@' WHERE '%@' = '%@'",
                TABLENAME,  AGE, @"15" ,AGE, @"13"];
    BOOL res = [db executeUpdate:updateSql];
    if (!res) {
      NSLog(@"error when update db table");
    } else {
      NSLog(@"success to update db table");
    }
    [db close]; 

  }

删除数据:

if ([db open]) { 

    NSString *deleteSql = [NSString stringWithFormat:
                @"delete from %@ where %@ = '%@'",
                TABLENAME, NAME, @"张三"];
    BOOL res = [db executeUpdate:deleteSql]; 

    if (!res) {
      NSLog(@"error when delete db table");
    } else {
      NSLog(@"success to delete db table");
    }
    [db close]; 

  } 

数据库查询操作:

查询操作使用了executeQuery,并涉及到FMResultSet。

if ([db open]) {
    NSString * sql = [NSString stringWithFormat:
             @"SELECT * FROM %@",TABLENAME];
    FMResultSet * rs = [db executeQuery:sql];
    while ([rs next]) {
      int Id = [rs intForColumn:ID];
      NSString * name = [rs stringForColumn:NAME];
      NSString * age = [rs stringForColumn:AGE];
      NSString * address = [rs stringForColumn:ADDRESS];
      NSLog(@"id = %d, name = %@, age = %@ address = %@", Id, name, age, address);
    }
    [db close];
  } 

FMDB的FMResultSet提供了多个方法来获取不同类型的数据:

数据库多线程操作:

如果应用中使用了多线程操作数据库,那么就需要使用FMDatabaseQueue来保证线程安全了。 应用中不可在多个线程中共同使用一个FMDatabase对象操作数据库,这样会引起数据库数据混乱。 为了多线程操作数据库安全,FMDB使用了FMDatabaseQueue,使用FMDatabaseQueue很简单,首先用一个数据库文件地址来初使化FMDatabaseQueue,然后就可以将一个闭包(block)传入inDatabase方法中。 在闭包中操作数据库,而不直接参与FMDatabase的管理。

FMDatabaseQueue * queue = [FMDatabaseQueue databaseQueueWithPath:database_path];
  dispatch_queue_t q1 = dispatch_queue_create("queue1", NULL);
  dispatch_queue_t q2 = dispatch_queue_create("queue2", NULL); 

  dispatch_async(q1, ^{
    for (int i = 0; i < 50; ++i) {
      [queue inDatabase:^(FMDatabase *db2) { 

        NSString *insertSql1= [NSString stringWithFormat:
                   @"INSERT INTO '%@' ('%@', '%@', '%@') VALUES (?, ?, ?)",
                   TABLENAME, NAME, AGE, ADDRESS]; 

        NSString * name = [NSString stringWithFormat:@"jack %d", i];
        NSString * age = [NSString stringWithFormat:@"%d", 10+i]; 

        BOOL res = [db2 executeUpdate:insertSql1, name, age,@"济南"];
        if (!res) {
          NSLog(@"error to inster data: %@", name);
        } else {
          NSLog(@"succ to inster data: %@", name);
        }
      }];
    }
  }); 

  dispatch_async(q2, ^{
    for (int i = 0; i < 50; ++i) {
      [queue inDatabase:^(FMDatabase *db2) {
        NSString *insertSql2= [NSString stringWithFormat:
                   @"INSERT INTO '%@' ('%@', '%@', '%@') VALUES (?, ?, ?)",
                   TABLENAME, NAME, AGE, ADDRESS]; 

        NSString * name = [NSString stringWithFormat:@"lilei %d", i];
        NSString * age = [NSString stringWithFormat:@"%d", 10+i]; 

        BOOL res = [db2 executeUpdate:insertSql2, name, age,@"北京"];
        if (!res) {
          NSLog(@"error to inster data: %@", name);
        } else {
          NSLog(@"succ to inster data: %@", name);
        }
      }];
    }
  });

源码下载:DEMO

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • iOS开发中使用FMDB来使程序连接SQLite数据库

    一.简单说明 1.什么是FMDB FMDB是iOS平台的SQLite数据库框架 FMDB以OC的方式封装了SQLite的C语言API 2.FMDB的优点 使用起来更加面向对象,省去了很多麻烦.冗余的C语言代码 对比苹果自带的Core Data框架,更加轻量级和灵活 提供了多线程安全的数据库操作方法,有效地防止数据混乱 3.FMDB的github地址 https://github.com/ccgus/fmdb 二.核心类 FMDB有三个主要的类 (1)FMDatabase 一个FMDatabase

  • iOS App项目中引入SQLite数据库的教程

    引入SQLite sqlite是纯C实现的,所以注定了它是一个跨平台利器,在Android与IOS下均能使用,而且完全可以写出通用的代码,方便我们移植.当然Android和IOS下都有封装过的sqlite给开发者使用,不过这样子一个是不方便移植,另一个是封装后的效率咋样我们也不知道,所以还是原生态的最健康.最后一个重要的原因就是原生的使用也是相当简单.我将在接下来的教程中为您一一讲解. 首先最重要的一点是在工程中导入sqlite,苹果的SDK已经给你包含进来了,所以只要导入一个叫 libsqli

  • iOS App使用SQLite之句柄的定义及数据库的基本操作

    句柄 要操纵一个数据库你就得有一个这个数据库的句柄(又碰到这个难以理解的词了,不过确实还没得一个更好的词来替代它).其实你跟本不需要去在乎这个词叫什么,你只要搞清楚他是一个什么玩意儿.就如同鞋子为什么叫鞋子,仔细想想确实也难以理解,不过 清楚他的功能就OK了,不是吗? 句柄在很多地方我们见到过,最常见的就是文件的句柄,我们要操纵一个文件,我们就要取得一个文件的句柄.句柄是个什么东东呢?其实很简单,句柄是一个东东的描述,他被定义为一个结构体,这个结构体可能会包含要描述的东东的具体信息,比如位置.大

  • iOS sqlite对数据库的各种操作(日常整理全)

    在IOS中使用Sqlite来处理数据.如果你已经了解了SQL,那你可以很容易的掌握SQLite数据库的操作.iOS对于数据库的操作:增加.删除.查找.修改具体介绍如下所示: 首先需要创建一个数据库:本程序的数据库是在火狐浏览器里的插件里写的微量型数据库 火狐找查找SQLite Manager的步骤: 第一步:在工具栏找到附加组件,点击进入 第二步:搜索 SQP,找到并下载,安装完成之后需要重启浏览器 第三步:在工具只乐观找到SQLite Manager,点击打开 SQLite Manager界面

  • IOS 数据库升级数据迁移的实例详解

    IOS 数据库升级数据迁移的实例详解 概要: 很久以前就遇到过数据库版本升级的引用场景,当时的做法是简单的删除旧的数据库文件,重建数据库和表结构,这种暴力升级的方式会导致旧的数据的丢失,现在看来这并不不是一个优雅的解决方案,现在一个新的项目中又使用到了数据库,我不得不重新考虑这个问题,我希望用一种比较优雅的方式去解决这个问题,以后我们还会遇到类似的场景,我们都想做的更好不是吗? 理想的情况是:数据库升级,表结构.主键和约束有变化,新的表结构建立之后会自动的从旧的表检索数据,相同的字段进行映射迁移

  • iOS开发中使用SQL语句操作数据库的基本用法指南

    SQL代码应用示例 一.使用代码的方式批量添加(导入)数据到数据库中 1.执行SQL语句在数据库中添加一条信息 插入一条数据的sql语句: 点击run执行语句之后,刷新数据 2.在ios项目中使用代码批量添加多行数据示例 代码示例: 复制代码 代码如下: // //  main.m //  01-为数据库添加多行数据 // //  Created by apple on 14-7-26. //  Copyright (c) 2014年 wendingding. All rights reserv

  • iOS学习笔记(十六)——详解数据库操作(使用FMDB)

    iOS中原生的SQLite API在使用上相当不友好,在使用时,非常不便.于是,就出现了一系列将SQLite API进行封装的库,例如FMDB.PlausibleDatabase.sqlitepersistentobjects等,FMDB (https://github.com/ccgus/fmdb) 是一款简洁.易用的封装库,这一篇文章简单介绍下FMDB的使用. 在FMDB下载文件后,工程中必须导入如下文件,并使用 libsqlite3.dylib 依赖包. FMDB同时兼容ARC和非ARC工

  • Java基础学习笔记之数组详解

    本文实例讲述了Java基础学习笔记之数组.分享给大家供大家参考,具体如下: 数组的定义于使用 1:数组的基本概念 一组相关变量的集合:在Java里面将数组定义为引用数据类型,所以数组的使用一定要牵扯到内存分配:想到了用new 关键字来处理. 2:数组的定义格式 区别: 动态初始化后数组中的每一个元素的内容都是其对应数据类型的默认值,随后可以通过下标进行数组内容的修改: 如果希望数组定义的时候就可以提供内容,则采用静态初始化的方式: a:数组的动态初始化(声明并初始化数组): 数据类型 数组名称

  • C#学习笔记之适配器模式详解

    什么是适配器模式? 适配器模式(Adapter):将一个类的接口转换成客户希望的另外一个接口. Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作. 什么时候运用适配器模式? 在想使用一个已经存在的类时,如果它的接口,也就是它的方法与我们当前的要求不相同时,就需要考虑用到适配器模式了. 一般而言,使用适配器模式是出于无奈之举,一般存在于软件开发后期或者维护期,在软件设计阶段,我们还是尽量避免该模式的使用. 适配器模式的组成   Target:目标接口,也就是客户所期待的接

  • java9学习笔记之模块化详解

    前言 截止到目前JDK的版本已经更新到10了,虽然java9的生命周期才半年,但是我认为这个版本带来的变革是不可磨灭的,它是第一次深层次的针对架构以及依赖上的革新.下面我们就来学习一下. 模块化的功能有几个目的: 让Java的SE程序更加容易轻量级部署 改进组件间的依赖管理,引入比Jar粒度更大的Module 改进性能和安全性 如果用更加简单解释,那就是"解决Classpath地狱问题,改进部署能力".Module的内容比较多,为了由浅入深,我按照一些问题和我的理解来介绍模块化. 一.

  • javascript学习笔记(十六) 系统对话框(alert、confirm、prompt)

    1.警告框alert() 复制代码 代码如下: alert("欢迎光临!"); 2.信息框confirm(),有取消,确定按钮 复制代码 代码如下: if (confirm("你同意吗?")) { alert("同意"); } else { alert("不同意"); } 3.提示框prompt(),用于提示用户输入一些文本 复制代码 代码如下: var result = prompt("您尊姓大名?",&

  • MySQL数据库学习之分组函数详解

    目录 1.分组函数 极值 求和 平均值 列数和 2.分组查询 3.小练习 4.大BOSS 1.分组函数 极值 示例表内容见此篇文章 找出最高工资: mysql> select max(sal) from emp; +----------+ | max(sal) | +----------+ | 5000.00 | +----------+ 1 row in set (0.00 sec) 找出最低工资: mysql> select min(sal) from emp; +----------+

  • Flask框架学习笔记之消息提示与异常处理操作详解

    本文实例讲述了Flask框架学习笔记之消息提示与异常处理操作.分享给大家供大家参考,具体如下: flask通过flash方法来显示提示消息: from flask import Flask, flash, render_template, request, abort app = Flask(__name__) app.secret_key = '520' @app.route('/') def index(): flash("Hello loli") return render_te

  • Python树莓派学习笔记之UDP传输视频帧操作详解

    本文实例讲述了Python树莓派学习笔记之UDP传输视频帧操作.分享给大家供大家参考,具体如下: 因为我在自己笔记本电脑上没能成功安装OpenCV-Contrib模块,因此不能使用人脸识别等高级功能,不过已经在树莓派上安装成功了,所以我想实现把树莓派上采集的视频帧传输到PC的功能,这样可以省去给树莓派配显示屏的麻烦,而且以后可能可以用在远程监控上. 1 UDP还是TCP 首先考虑用哪种传输方式,平常TCP用的非常多,但是像视频帧这种数据用TCP不是太合适,因为视频数据的传输最先要考虑的是速度而不

  • Python OpenCV学习之图像滤波详解

    目录 背景 一.卷积相关概念 二.卷积实战 三.均值滤波 四.高斯滤波 五.中值滤波 六.双边滤波 七.Sobel算子 八.Scharr算子 九.拉普拉斯算子 十.Canny算法 背景 图像滤波的作用简单来说就是将一副图像通过滤波器得到另一幅图像:明确一个概念,滤波器又被称为卷积核,滤波的过程又被称为卷积:实际上深度学习就是训练许多适应任务的滤波器,本质上就是得到最佳的参数:当然在深度学习之前,也有一些常见的滤波器,本篇主要介绍这些常见的滤波器: 一.卷积相关概念 卷积核大小一般为奇数的原因:

  • PHP读书笔记_运算符详解

    什么是运算符 什么是运算符?运算符是告诉PHP做相关运算的标识符号.例如,你需要计算123乘以456等于多少,这时候就需要一个符号,告诉服务器,你需要做乘法运算. PHP中的运算符有哪些?PHP运算符一般分为算术运算符.赋值运算符.比较运算符.三元运算符.逻辑运算符.字符串连接运算符.错误控制运算符. PHP中的算术运算符 算术运算符主要是用于进行算术运算的,例如:加法运算.减法运算.乘法运算.除法运算.在PHP中的常用的算术运算符对应下表: PHP中的赋值运算符 PHP的赋值运算符有两种,分别

随机推荐