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

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

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

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

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型。

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

看一下例子:

创建表:

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提供了多个方法来获取不同类型的数据:

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

数据库多线程操作:

如果应用中使用了多线程操作数据库,那么就需要使用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

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

时间: 2016-12-10

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

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

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

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

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

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

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来使程序连接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 数据库升级数据迁移的实例详解

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

IOS开发之判断两个数组中数据是否相同实例详解

IOS开发之判断两个数组中数据是否相同实例详解 前言: 工作中遇到的问题,这里记录下,也许能帮助到大家 实例代码: NSArray *array1 = [NSArray arrayWithObjects:@"a", @"b", @"c", nil nil]; NSArray *array2 = [NSArray arrayWithObjects:@"b", @"a", @"c", nil

使用Swift代码实现iOS手势解锁、指纹解锁实例详解

一.手势密码 1. 1.1.用UIButton组成手势的节点. 1.2.当手指接触屏幕时,调用重写的 touchesBegan:withEvent方法(在touchesBegan里调用setNeedsDisplay,这样就会自动调用drawRect方法). 1.3.当手指在屏幕上滑动时,调用重写的touchesEnded:withEvent方法. 这两个方法执行的操作是一样的:通过locationInView获取 触摸的坐标,然后用 CGRectContainsPoint 判断手指是否经过UIB

IOS 开发中画扇形图实例详解

IOS 开发中画扇形图实例详解 昨天在做项目中,遇到一个需要显示扇形图的功能,网上搜了一下,发现code4app里面也没有找到我想要的那种类似的效果,没办法了,只能自己学习一下如何画了. 首先我们需要了解一个uiview的方法 -(void)drawRect:(CGRect)rect 我们知道了这个方法,就可以在自定义UIView的子类的- (void)drawRect:(CGRect)rect里面绘图了,关于drawrect的调用周期,网上也是一找一大堆,等下我会整理一下,转载一篇供你们参考.

IOS 中CALayer绘制图片的实例详解

IOS 中CALayer绘制图片的实例详解 CALayer渲染内容图层.与UIImageView相比,不具有事件响应功能,且UIImageView是管理内容. 注意事项:如何使用delegate对象执行代理方法进行绘制,切记需要将delegate设置为nil,否则会导致异常crash. CALayer绘制图片与线条效果图: 代码示例: CGPoint position = CGPointMake(160.0, 200.0); CGRect bounds = CGRectMake(0.0, 0.0

IOS UIView的生命周期的实例详解

IOS UIView的生命周期的实例详解 任何对象的者有一个生命周期,即都存在一个实例化到销毁的过程. UIView对象也不例外,那么UIView从init/new开始后,直到dealloc结束的过程中都经历了哪些过程呢? 首先自定义继承自UIView的对象LifeView #import <UIKit/UIKit.h> @interface LifeView : UIView @end #import "LifeView.h" @interface LifeView ()

mysql数据存储过程参数实例详解

MySQL 存储过程参数有三种类型:in.out.inout.它们各有什么作用和特点呢? 一.MySQL 存储过程参数(in) MySQL 存储过程 "in" 参数:跟 C 语言的函数参数的值传递类似, MySQL 存储过程内部可能会修改此参数,但对 in 类型参数的修改,对调用者(caller)来说是不可见的(not visible). drop procedure if exists pr_param_in; create procedure pr_param_in ( in id

Spring jdbc中数据库操作对象化模型的实例详解

Spring jdbc中数据库操作对象化模型的实例详解 Spring Jdbc数据库操作对象化 使用面向对象方式表示关系数据库的操作,实现一个线程安全可复用的对象模型,其顶级父类接口RdbmsOperation. SqlOperation继承该接口,实现数据库的select, update, call等操作. 1.查询接口:SqlQuery 1) GenericSqlQuery, UpdatableSqlQuery, MappingSqlQueryWithParameter 2) SqlUpda

java与JSON数据的转换实例详解

java与JSON数据的转换实例详解 JSON与JAVA数据的转换(JSON 即 JavaScript Object Natation,它是一种轻量级的数据交换格式,非常适合于服务器与 JavaScript 的交互.) 代码中有这么一句,是后台的封装数据. JSONObject jo = JSONObject.fromObject(map); 常见的java代码转换成json --请注意,这个方法曾经给我造成过困惑.因为,它在对Object转换的时候是按照domain类中的所有getXXX()方

IOS 开发之Swift 元组的实例详解

IOS 开发之Swift 元组的实例详解 元组是多个值组合而成的复合值.元组中的值可以是任意类型,而且每一个元素的类型可以是不同的. 元组声明 //普通声明 var point = (5,2) var httpResponse = (404, "Not Found") //定义类型声明 var point2 : (Int,Int,Int) = (10,5,2) var httpResponse2 : (Int,String) = (200,"ok") 元组解包 va