iOS FMDB操作数据库

先扯段犊子:

首先关于数据库,工作中我们可能会接触过很多,像公司常用甲骨文公司的Oracle数据库,特点是安全可靠,适合需要存储大量数据的大公司使用,相比Oracle,微软公司的SQL Server适合中小公司使用,同样安全可靠(一般都是VS + SQL Server配合使用),还有MySQL、DB2、甚至Access等等,但以上介绍的大多都适合PC端使用;而对于移动端iOS、Android来说,非你莫属的要属于SQLite,首先SQLite是一款轻型的嵌入式数据库,同时SQLite同Oracle,SQL Server一样同属于关系型数据库,并且它占用的资源非常小,所以非常适合移动客户端和嵌入式设备所需数据库的特点.

进入正文

select、insert、update、delete、from、create、where、desc、order、by、group、table、alter、view、index等等,切记数据库中不可以使用关键字来命名表、字段等操作

  • 字段类型的说明

相比其他数据库,SQLite数据库的数据类型仅仅就四种:
integer : 整型值
real : 浮点值
text : 文本字符串
blob : 二进制数据(比如文件)

  • SQL语句简单的可以分为以下三类
    数据定义语句(DDL:Data Definition Language)
    包括create和drop等操作(通常用来创建表或者删除表)
    数据操作语句(DML:Data Manipulation Language)
    包括insert、update、delete等操作(添加、修改、删除表中的数据)
    数据查询语句(DQL:Data Query Language)
    可以用于查询获得表中的数据 关键字select(通常用来查询表数据)以及where(添加条件),order by(排序),group by(分组)和having(通常是在一个 SQL 句子的最后)
SQLite数据库的基本操作
  • 表的约束
    工作中,为了符合我们日常操作以及流程的规范,常常创建表的时候我们需要考虑多方面的因素,比如有些代表特别意义的字段不能为空(姓名,性别等...),有些字段必须为一(身份证),这个时候就需要我们给字段设置严格的约束,让其从创建数据库表的时候就把一些不规范的数据给排除掉,以保证数据的规范性.

  • 关健字:

    Not Null :规定字段的值不能为NULL
    Unique :规定字段的值必须唯一
    Default :指定字段的默认值

//NAME字段不能为NULL,并且唯一age字段不能为NULL,并且默认为1
CREATE TABLE t_test (ID INTEGER, NAME TEXT NOT NULL UNIQUE, AGE INTEGER NOT NULL DEFAULE 1);```

* 主键约束

 为了更方便的管理数据,保证数据的规范性,数据库设计之初应该多考虑数据的唯一性,因此再创建表的时候,我们有必要给表添加一个主键,来约束表中的数据,当然如果设计的数据逻辑关联性比较复杂的时候我们也可以指定多个字段来充当我们表的主键(联合主键)

 在创表的时候用PRIMARY KEY声明一个主键

//ID作为t_test表的主键,并且设置为自增长
CREATE TABLE t_test (ID INTEGER PRIMARY KEY AUTOINCREMENT,NAME TEXT,AGE INTEGER);```

  • 外键
    主要使用来约束表与表之间的关系,比如某个表中得一个字段是另一个表中得主键,这个时候我们就需要用到外键了

  • 创建表(CREATE)

格式: CREATE TABLE IF NOT EXISTS 表名 (字段名1 字段1类型,字段名2,字段2类型,...); 
示例: CREATE TABLE IF NOT EXISTS t_test (ID INTEGER,NAME TEXT,AGE INTEGER);```

* 删除表(DROP)

格式: DROP TABLE IF EXISTS 表名;
示例: DROP TABLE IF EXISTS t_test;```

  • 插入数据(INSERT)
格式:INSERT INTO 表名 (字段名1,字段名2,...)VALUES (字段1的值,字段2的值,...);
示例:INSERT INTO t_test (ID,NAME,AGE)VALUES (1,'J_mailbox',23);```

* 更新数据(UPDATE)

格式:UPDATE 表名 SET 字段名1=字段1的值,字段名2=字段名2的值,...;
示例:UPDATE t_test SET ID=2,NAME='J_mailbox',AGE=25;```

  • 删除数据(DELETE)
格式:DELETE FROM 表名;
示例:DELETE FROM t_test;```

* 查询语句

格式:SELECT * FROM 表名;SELECT NAME,AGE FROM 表名;
示例:SELECT * FROM t_test;SELECT NAME,AGE FROM t_test;```

  • 条件语句
格式:
//查询表中字段等于某个值的所有记录
 SELECT * FROM 表名 WHERE 字段 = 某个值;

 //查询表中字段不等于某个值的所有记录
 SELECT * FROM 表名 WHERE 字段 != 某个值

 //查询表中字段是某个值的所有记录
 SELECT * FROM 表名 WHERE 字段 IS 某个值

 //查询表中字段不是某个值的所有记录
 SELECT * FROM 表名 WHERE 字段 IS NOT 某个值

 //查询表中字段大于某个值的所有记录
 SELECT * FROM 表名 WHERE 字段 > 某个值

 //查询表中字段小于某个值的所有记录
 SELECT * FROM 表名 WHERE 字段 < 某个值

 //查询表中某个字段依某个值开头的所有记录
 SELECT * FROM 表名 WHERE 字段 LIKE "某个值%"

 //查询表中某个字段包含某个值的所有记录
 SELECT * FROM 表名 WHERE 字段 LIKE "%某个值%"

 //查询表中记录从多少行开始,查询多少行记录
 SELECT * FROM 表名 LIMIT 从多少行开始,查询多少行;

 //查询表中字段名1等于某个值并且字段名2大于某个值的所有记录(两个条件都要满足)
 SELECT * FROM 表名 WHERE 字段名1 = 某个值 AND 字段名2 >某个值

 //查询表中字段名1等于某个值或者字段名2小于某个值的所有记录(两个条件满足一个就可以)
 SELECT * FROM 表名 WHERE 字段名1 = 某个值 OR 字段名2 <某个值

 //更新表中字段等于某个值并且字段大于某个值的所有记录
 UPDATE 表名 SET 字段名1= WHERE 字段名2=某个值 AND 字段名3 >某个值;
  //删除t_test表中NAME等于J_mailbox或者AGE小于18的所有记录
 DELETE FROM 表名 WHERE 字段名1=某个值 OR 字段名2 <某个值;

 示例:
 //查询t_test表中NAME字段等于J_mailbox的所有记录
 SELECT * FROM t_test WHERE NAME ='J_mailbox';

 //查询t_test表中NAME字段不等于J_mailbox的所有记录
 SELECT * FROM t_test WHERE NAME !='J_mailbox';

 //查询t_test表中NAME字段是J_mailbox的所有记录
 SELECT * FROM t_test WHERE NAME IS 'J_mailbox';

 //查询t_test表中NAME字段不是J_mailbox的所有记录
 SELECT * FROM t_test WHERE NAME IS NOT 'J_mailbox';

 //查询t_test表中AGE字段大于20的所有记录
 SELECT * FROM t_test WHERE AGE > 20;

 //查询t_test表中AGE字段大于20的所有记录
 SELECT * FROM t_test WHERE AGE < 20;

 //查询t_test中NAME依J字母开头的所有记录
 SELECT * FROM 表名 WHERE NAME LIKE "J%"

 //查询t_test中NAME包含_mail的所有记录
 SELECT * FROM 表名 WHERE NAME LIKE "%_mail%"

 //查询表中记录从0行开始,查询10行记录
 SELECT * FROM 表名 LIMIT 0,10;

 //查询t_test表中NAME等于J_mailbox并且AGE字段大于20的所有记录(两个条件都要满足)
 SELECT * FROM t_test WHERE NAME ='J_mailbox' AND AGE > 20;

 //查询t_test表中NAME等于J_mailbox或者AGE字段小于20的所有记录(两个条件满足一个就可以)
 SELECT * FROM t_test WHERE NAME ='J_mailbox' OR AGE < 20;

 //更新t_test表中NAME等于J_mailbox并且AGE大于18的所有记录
 UPDATE t_test SET AGE=20 WHERE NAME='J_mailbox' AND AGE >18;

 //删除t_test表中NAME等于J_mailbox或者AGE小于18的所有记录
 DELETE FROM t_test WHERE NAME='J_mailbox' OR AGE <18;```

* 起别名(AS)

格式:SELECT 字段名1 别名1 ,字段名2 别名2,... FROM 表名;SELECT 字段名1 AS 别名1 ,字段名2 AS 别名2,... FROM 表名;
示例:SELECT NAME 姓名 ,AGE 年龄 FROM t_test;SELECT NAME AS 姓名,AGE AS 年龄 FROM t_test;```

  • 排序(ORDER BY, ASC,DESC)
格式:SELECT * FROM 表名 ORDER BY 字段名;SELECT * FROM 表名 ORDER BY 字段名 DESC;SELECT * FROM 表名 ORDER BY 字段名1 ASC,字段名2 DESC;
示例:SELECT * FFROM t_test ORDER BY AGE;SELECT * FFROM t_test ORDER BY AGE DESC;SELECT * FROM t_test ORDER BY 字段名1 ASC,字段名2 DESC;```

* 计算记录的总数量(COUNT)
````格式:SELECT COUNT(*) FROM 表名;SELECT COUNT(字段) FROM 表名;
示例:SELECT COUNT(*) FROM t_test;SELECT COUNT(AGE) FROM t_test;```


#####Objective-C对SQLite数据库的增删改查操作
要使用SQLite,首先我们要导入SQLite3框架
SQLite常用函数
** 打开数据库 **

int sqlite3_open(
const char *filename, // 数据库的文件路径
sqlite3 **ppDb // 数据库实例
);```
** 执行SQL语句 **

int sqlite3_exec(
    sqlite3*,                                  // 一个打开的数据库实例
    const char *sql,                           // 需要执行的SQL语句
    int (*callback)(void*,int,char**,char**),  // SQL语句执行完毕后的回调
    void *,                                    // 回调函数的第1个参数
    char **errmsg                              // 错误信息
);```

//连接数据库

  • (void)connectionDB{
    NSLog(@"%@",NSHomeDirectory());
    //获取路径
    NSString *path=[[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject]stringByAppendingPathComponent:@"data.sqlite"];
    //判断数据库是否打开成功
    int success=sqlite3_open(path.UTF8String, &_db);
    //如果数据库打开成功
    if (success==SQLITE_OK) {
    //创建表SQL语句,并且设置ID为主键,且自动增长
    NSString *sql=@"CREATE TABLE IF NOT EXISTS t_test (ID INTEGER PRIMARY KEY AUTOINCREMENT,NAME TEXT,AGE INTEGER)";
    //判断创建表是否成功
    int success_t=sqlite3_exec(_db, sql.UTF8String, NULL, NULL, NULL);
    //如果创建表成功
    if (success_t==SQLITE_OK) {
    NSLog(@"创建表成功!");
    }else{
    NSLog(@"创建表失败");
    }
    }else{
    NSLog(@"数据库创建失败");
    }
    //关闭数据库
    //sqlite3_close(_db);
    }

//增加数据

  • (IBAction)addClick:(id)sender {

    //往表中循环插入100条数据
    for (int i = 0; i < 100 ; i++) {
    //名称设置为J_mailbox
    NSString *name = [NSString stringWithFormat:@"J_mailbox-%d",i];
    //随机生成20岁~25岁之间的记录
    NSInteger age = arc4random_uniform(5) + 20;

      //sql插入语句的拼接
      NSString *resultStr = [NSString stringWithFormat:@"INSERT INTO t_test (NAME,AGE) VALUES('%@',%zd) ",name,age];
      //执行sql插入语句
      int success =  sqlite3_exec(_db, resultStr.UTF8String, NULL, NULL, NULL);
      //判断是否插入成功
      if (success == SQLITE_OK) {
          NSLog(@"添加数据成功!");
      }else{
          NSLog(@"添加数据失败!");
      }
    

    }

}

//删除数据

  • (IBAction)deleteClick:(id)sender {

    //sql删除语句
    NSString *sqlStr = @"DELETE FROM t_test WHERE AGE > 22 ;";
    //执行删除语句操作
    int success = sqlite3_exec(_db, sqlStr.UTF8String, NULL, NULL, NULL);

    if (success == SQLITE_OK) {
    NSLog(@"删除数据成功!");
    }else{
    NSLog(@"删除数据失败!");
    }

}

//修改数据数据

  • (IBAction)updateClick:(id)sender {

    //sql修改语句
    NSString *sqlStr = @"UPDATE t_test SET AGE = 30 WHERE AGE <25;";
    //执行修改语句操作
    int success = sqlite3_exec(_db, sqlStr.UTF8String, NULL, NULL, NULL);

    if (success == SQLITE_OK) {
    NSLog(@"修改数据成功!");
    }else{
    NSLog(@"修改数据失败!");
    }
    }

//查询数据

  • (IBAction)selectClick:(id)sender {

    //sql查询语句
    NSString *sqlStr = @"SELECT NAME,AGE FROM t_test WHERE AGE = 30;";
    //定义存放结果数据stmt
    sqlite3_stmt *stmt = NULL;
    //取一条记录
    int success = sqlite3_prepare_v2(_db, sqlStr.UTF8String,-1, &stmt, NULL);
    if (success == SQLITE_OK) {
    NSLog(@"查询数据成功!");
    //拿数据 step 一步 拿一条记录
    while (sqlite3_step(stmt) == SQLITE_ROW) { //证明取到了一条记录
    //查询到得NAME
    const char *name = (const char *)sqlite3_column_text(stmt, 0);
    //查询到得AGE
    int age = sqlite3_column_int(stmt, 1);
    //打印结果
    NSLog(@"NAME = %@ AGE = %d",[NSString stringWithUTF8String:name],age);
    }

    }else{
    NSLog(@"查询数据失败!");
    }
    }```

FMDB的使用

因为自带的SQLite3 是基于C语言的API,所以使用的时候有些繁琐,而FMDB是面向对象的,使用起来更加的简单,且只需要调用对应的方法,传值进去即可,而且还提供了多线程安全

FMDB下载地址

//  
//  ViewController.m  
//  数据库SQLite语句增删改查+++  
//  
//  Created by apple on 15/9/29.  
//  Copyright (c) 2015年 LiuXun. All rights reserved.  
//  
  
#import "ViewController.h"  
#import "FMDatabase.h"  
#define TABLENAME  @"tableName"  
#define  ID  @"id"  
#define NAME @"name"  
#define AGE  @"age"  
#define  ADDRESS @"address"  
@interface ViewController ()  
  
@end  
@implementation ViewController  
  
- (void)viewDidLoad {  
    [super viewDidLoad];  
      
    /*创建数据库路径*/  
    NSArray *path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);  
    NSString *documentDirectory = [path   objectAtIndex:0];  
    NSString *dbPath = [documentDirectory stringByAppendingPathComponent:@"test.db"];  
    NSLog(@"dbPath= %@",dbPath);  
    /*连接数据库——连接数据库,并没有创建数据库文件,直到打开成功才算真正创建*/  
    FMDatabase *db = [FMDatabase databaseWithPath:dbPath];  
    /*=====================================================================*/  
    /*创建表*/  
    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 %@",TABLENAME);  
        }else  
        {  
            NSLog(@"success when creating db table %@",TABLENAME);  
        }  
        [db close];  
    }  
      
    /*增删改查——增加数据*/  
    if ([db open]) {  // 第一种:方式全部用字符串拼接NSString类型的SQL语句  
        NSString *insertSql1 = [NSString stringWithFormat:  
                                @"INSERT INTO '%@' ('%@', '%@', '%@') VALUES ('%@', '%@', '%@')",  
                                TABLENAME, NAME, AGE, ADDRESS, @"张三", @"13", @"济南"];  
        BOOL res1 = [db executeUpdate:insertSql1];  
        NSString *insertSql2 = [NSString stringWithFormat:@"INSERT INTO '%@' ('%@','%@','%@') VALUES ('%@','%@','%@')",TABLENAME,NAME,AGE,ADDRESS,@"李四",@"18",@"广州"];  
        BOOL res2 = [db executeUpdate:insertSql2];  
        if (res1) {  
            NSLog(@"insert 1  scucess");  
        }  
        else{  
            NSLog(@"insert 1 failure");  
        }  
        if (res2) {  
            NSLog(@"insert 2  scucess");  
        }  
        else{  
            NSLog(@"insert 2 failure");  
        }  
        [db close];  
    }  
      
    /*增删改查——修改数据*/  
    if ([db open]) {  // 可以用任意类型的变量来拼接OC 字符串来作为数据库查询语句  
        NSString *updateSql = [NSString stringWithFormat:@" UPDATE %@ SET  %@= %d WHERE %@ = %d ",TABLENAME,AGE, 40,AGE,13];  
        BOOL  res = [db executeUpdate:updateSql];  
        if (res) {  
            NSLog(@"update success");  
        }  
        else  
        {  
          NSLog(@"update Failure");  
        }  
        [db close];  
    }  
      
     /*增删改查——删除数据*/  
    if ([db open]) {  // 第三种:不拼接NSString字符串,直接写语句  
        /* 
           必须注意:此种方式只有变量值可以用?来替代,别的(如表名,键名都必须直接写在前面不能用格式化的方法来操作) 
         */  
        BOOL res = [db executeUpdate:@"DELETE FROM TABLENAME WHERE NAME = ?",@"张三"];  
        if (res) {  
            NSLog(@"delete success");  
        }  
        else  
        {  
          NSLog(@"delete Failure");  
        }  
        [db close];  
    }  
      
    /*数据库查询操作*/  
    if ([db open]) {  // 查询操作使用executeQuery,并涉及到了FMResultSet  
        NSString *sql = [NSString stringWithFormat:@"SELECT * FROM %@ ORDER BY %@ DESC",TABLENAME,ID];    // 按照ID号降序排列,如果只有order by默认是升序的 等同于ASC  
        FMResultSet * rs = [db executeQuery:sql];  
        while ([rs next]) {  
            int Id = [rs intForColumn:ID];  
            NSString *name = [rs stringForColumn:NAME];  
            NSString *age = [rs stringForColumn:AGE];  // 数值和NSString类型能自动进行转换,不会出错  
            NSString *address = [rs stringForColumn:ADDRESS];  
            NSLog(@"id= %d name=%@ age=%@ address=%@",Id,name,age,address);  
        }  
          
        [db close];  
    }  
      
    /*增加指定列的操作*/  
    if ([db open]) {  
        NSString * sqlDropColumn = [NSString stringWithFormat:@"ALTER TABLE %@ ADD COLUMN %@",TABLENAME,@"XXXXX"];  // 当XXXXX存在时就不会在执行添加操作  
        BOOL res = [db executeUpdate:sqlDropColumn];  
        if (res) {  
            NSLog(@"ADD column  success");  
        }  
        else  
        {  
            NSLog(@"ADD column  Failure");  
        }  
        [db close];  
    }  
      
    /*重命名指定表*/  
    if ([db open]) {  
        NSString *sqlRenamColumn = [NSString stringWithFormat:@"ALTER TABLE %@ RENAME TO %@",TABLENAME,TABLENAME];  //TABLENAME可以用 @"renamTabNam"替换,但为了调试方便重命名为原名  
        BOOL res = [db executeUpdate:sqlRenamColumn];  
            if (res) {  
                NSLog(@"Rename column  success");  
            }  
            else  
            {  
                NSLog(@"Rename column  Failure");  
            }  
        [db close];  
    }  
      
    /*删除表的操作*/  
//    if ([db open]) {  
//        NSString *sqlDrop = [NSString stringWithFormat:@"DROP TABLE %@",TABLENAME];  
//        BOOL res =  [db executeUpdate:sqlDrop];  
//        if (res) {  
//            NSLog(@"Drop table success");  
//        }  
//        else  
//        {  
//            NSLog(@"Drop table Failure");  
//        }  
//        [db close];  
//    }  
    /*=====================================================================*/  
    /*数据库多线程操作*/  
    /** 
     *如果应用中使用了多线程操作数据库,那么就需要使用FMDatabaseQueue来保证线程安全了。 应用中不可在多个线程中共同使用一个FMDatabase对象操作数据库,这样会引起数据库数据混乱。 为了多线程操作数据库安全,FMDB使用了FMDatabaseQueue,使用FMDatabaseQueue很简单,首先用一个数据库文件地址来初使化FMDatabaseQueue,然后就可以将一个闭包(block)传入inDatabase方法中。 在闭包中操作数据库,而不直接参与FMDatabase的管理。 
     */  
    // 创建数据库线程队列  
    FMDatabaseQueue  *queue = [FMDatabaseQueue databaseQueueWithPath:dbPath];  
    dispatch_queue_t q1 = dispatch_queue_create("queue1", NULL);  // 创建第一个线程  
    dispatch_queue_t q2 = dispatch_queue_create("queue2", NULL);  // 创建第二个线程  
      
    dispatch_async(q1, ^{     // 表示线程q1同步执行——即按先后顺序执行块内的操作,前一个操作完毕再执行下一个操作  
        for (int i=0; i<50; i++) {  
            [queue inDatabase:^(FMDatabase *db) {  
                NSString *insertsql1 = [NSString stringWithFormat:@"INSERT INTO %@ (%@ ,%@,%@) VALUES (?,?,?)",TABLENAME,NAME,AGE,ADDRESS];  
                NSString *name = [NSString stringWithFormat:@"Jack %d",i];  
                NSString *age = [NSString stringWithFormat:@"%d",i+10];  
                BOOL res = [db executeUpdate:insertsql1,name,age,@"上海"];  
                if (res) {  
                    NSLog(@"线程一 第%d次insert   success",i);  
                }else  
                {  
                    NSLog(@"线程一 第%d次insert   Failure",i);  
                }  
            }];  
        }  
    });  
    dispatch_async(q2, ^{  
        for (int i=0; i<50; i++) {  
            [queue inDatabase:^(FMDatabase *db) {  
                NSString *insertsql2 = [NSString stringWithFormat:@"INSERT INTO %@ (%@, %@,%@) VALUES (?,?,?)",TABLENAME,NAME,AGE,ADDRESS];  
                NSString *name = [NSString stringWithFormat:@"TOM %d",i];  
                NSString *age = [NSString stringWithFormat:@"%d",i+10];  
                BOOL res = [db executeUpdate:insertsql2,name,age,@"北京"];  
                if (res) {  
                    NSLog(@"线程二 第%d次insert   success",i);  
                }else  
                {  
                    NSLog(@"线程二 第%d次insert   Failure",i);  
                }  
  
            }];  
        }  
    });  
}  
  
- (void)didReceiveMemoryWarning {  
    [super didReceiveMemoryWarning];  
}  
  
@end  ```

数据库查看软件[下载]( dhttp://pan.baidu.com/share/link?shareid=2957954380&uk=489049396)

部分转自(http://www.jianshu.com/p/22198f89e422)
http://blog.csdn.net/liwei5bao/article/details/50297703


你可能感兴趣的:(iOS FMDB操作数据库)