iOS FMDB操作數(shù)據(jù)庫

先扯段犢子:

首先關(guān)于數(shù)據(jù)庫,工作中我們可能會接觸過很多,像公司常用甲骨文公司的Oracle數(shù)據(jù)庫,特點是安全可靠,適合需要存儲大量數(shù)據(jù)的大公司使用,相比Oracle,微軟公司的SQL Server適合中小公司使用,同樣安全可靠(一般都是VS + SQL Server配合使用),還有MySQL、DB2搀军、甚至Access等等,但以上介紹的大多都適合PC端使用;而對于移動端iOS划乖、Android來說,非你莫屬的要屬于SQLite,首先SQLite是一款輕型的嵌入式數(shù)據(jù)庫,同時SQLite同Oracle,SQL Server一樣同屬于關(guān)系型數(shù)據(jù)庫,并且它占用的資源非常小,所以非常適合移動客戶端和嵌入式設(shè)備所需數(shù)據(jù)庫的特點.

進入正文

select过蹂、insert几苍、update矮瘟、delete茶鉴、from逐样、create泳桦、where汤徽、desc、order灸撰、by谒府、group拼坎、table、alter狱掂、view演痒、index等等,切記數(shù)據(jù)庫中不可以使用關(guān)鍵字來命名表、字段等操作

  • 字段類型的說明

相比其他數(shù)據(jù)庫,SQLite數(shù)據(jù)庫的數(shù)據(jù)類型僅僅就四種:
integer : 整型值
real : 浮點值
text : 文本字符串
blob : 二進制數(shù)據(jù)(比如文件)

  • SQL語句簡單的可以分為以下三類
    數(shù)據(jù)定義語句(DDL:Data Definition Language)
    包括create和drop等操作(通常用來創(chuàng)建表或者刪除表)
    數(shù)據(jù)操作語句(DML:Data Manipulation Language)
    包括insert趋惨、update鸟顺、delete等操作(添加、修改器虾、刪除表中的數(shù)據(jù))
    數(shù)據(jù)查詢語句(DQL:Data Query Language)
    可以用于查詢獲得表中的數(shù)據(jù) 關(guān)鍵字select(通常用來查詢表數(shù)據(jù))以及where(添加條件)讯嫂,order by(排序),group by(分組)和having(通常是在一個 SQL 句子的最后)
SQLite數(shù)據(jù)庫的基本操作
  • 表的約束
    工作中,為了符合我們?nèi)粘2僮饕约傲鞒痰囊?guī)范,常常創(chuàng)建表的時候我們需要考慮多方面的因素,比如有些代表特別意義的字段不能為空(姓名,性別等...),有些字段必須為一(身份證),這個時候就需要我們給字段設(shè)置嚴格的約束,讓其從創(chuàng)建數(shù)據(jù)庫表的時候就把一些不規(guī)范的數(shù)據(jù)給排除掉,以保證數(shù)據(jù)的規(guī)范性.

  • 關(guān)健字:

    Not Null :規(guī)定字段的值不能為NULL
    Unique :規(guī)定字段的值必須唯一
    Default :指定字段的默認值

//NAME字段不能為NULL兆沙,并且唯一age字段不能為NULL欧芽,并且默認為1
CREATE TABLE t_test (ID INTEGER, NAME TEXT NOT NULL UNIQUE, AGE INTEGER NOT NULL DEFAULE 1);```

* 主鍵約束

 為了更方便的管理數(shù)據(jù),保證數(shù)據(jù)的規(guī)范性,數(shù)據(jù)庫設(shè)計之初應(yīng)該多考慮數(shù)據(jù)的唯一性,因此再創(chuàng)建表的時候,我們有必要給表添加一個主鍵,來約束表中的數(shù)據(jù),當然如果設(shè)計的數(shù)據(jù)邏輯關(guān)聯(lián)性比較復雜的時候我們也可以指定多個字段來充當我們表的主鍵(聯(lián)合主鍵)

 在創(chuàng)表的時候用PRIMARY KEY聲明一個主鍵

//ID作為t_test表的主鍵,并且設(shè)置為自增長
CREATE TABLE t_test (ID INTEGER PRIMARY KEY AUTOINCREMENT,NAME TEXT,AGE INTEGER);```

  • 外鍵
    主要使用來約束表與表之間的關(guān)系,比如某個表中得一個字段是另一個表中得主鍵,這個時候我們就需要用到外鍵了

  • 創(chuàng)建表(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;```

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

* 更新數(shù)據(jù)(UPDATE)

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

  • 刪除數(shù)據(jù)(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;```

* 計算記錄的總數(shù)量(COUNT)
````格式:SELECT COUNT(*) FROM 表名;SELECT COUNT(字段) FROM 表名;
示例:SELECT COUNT(*) FROM t_test;SELECT COUNT(AGE) FROM t_test;```


#####Objective-C對SQLite數(shù)據(jù)庫的增刪改查操作
要使用SQLite,首先我們要導入SQLite3框架
SQLite常用函數(shù)
** 打開數(shù)據(jù)庫 **

int sqlite3_open(
const char *filename, // 數(shù)據(jù)庫的文件路徑
sqlite3 **ppDb // 數(shù)據(jù)庫實例
);```
** 執(zhí)行SQL語句 **

int sqlite3_exec(
    sqlite3*,                                  // 一個打開的數(shù)據(jù)庫實例
    const char *sql,                           // 需要執(zhí)行的SQL語句
    int (*callback)(void*,int,char**,char**),  // SQL語句執(zhí)行完畢后的回調(diào)
    void *,                                    // 回調(diào)函數(shù)的第1個參數(shù)
    char **errmsg                              // 錯誤信息
);```

//連接數(shù)據(jù)庫

  • (void)connectionDB{
    NSLog(@"%@",NSHomeDirectory());
    //獲取路徑
    NSString *path=[[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject]stringByAppendingPathComponent:@"data.sqlite"];
    //判斷數(shù)據(jù)庫是否打開成功
    int success=sqlite3_open(path.UTF8String, &_db);
    //如果數(shù)據(jù)庫打開成功
    if (success==SQLITE_OK) {
    //創(chuàng)建表SQL語句,并且設(shè)置ID為主鍵,且自動增長
    NSString *sql=@"CREATE TABLE IF NOT EXISTS t_test (ID INTEGER PRIMARY KEY AUTOINCREMENT,NAME TEXT,AGE INTEGER)";
    //判斷創(chuàng)建表是否成功
    int success_t=sqlite3_exec(_db, sql.UTF8String, NULL, NULL, NULL);
    //如果創(chuàng)建表成功
    if (success_t==SQLITE_OK) {
    NSLog(@"創(chuàng)建表成功!");
    }else{
    NSLog(@"創(chuàng)建表失敗");
    }
    }else{
    NSLog(@"數(shù)據(jù)庫創(chuàng)建失敗");
    }
    //關(guān)閉數(shù)據(jù)庫
    //sqlite3_close(_db);
    }

//增加數(shù)據(jù)

  • (IBAction)addClick:(id)sender {

    //往表中循環(huán)插入100條數(shù)據(jù)
    for (int i = 0; i < 100 ; i++) {
    //名稱設(shè)置為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];
      //執(zhí)行sql插入語句
      int success =  sqlite3_exec(_db, resultStr.UTF8String, NULL, NULL, NULL);
      //判斷是否插入成功
      if (success == SQLITE_OK) {
          NSLog(@"添加數(shù)據(jù)成功!");
      }else{
          NSLog(@"添加數(shù)據(jù)失敗!");
      }
    

    }

}

//刪除數(shù)據(jù)

  • (IBAction)deleteClick:(id)sender {

    //sql刪除語句
    NSString *sqlStr = @"DELETE FROM t_test WHERE AGE > 22 ;";
    //執(zhí)行刪除語句操作
    int success = sqlite3_exec(_db, sqlStr.UTF8String, NULL, NULL, NULL);

    if (success == SQLITE_OK) {
    NSLog(@"刪除數(shù)據(jù)成功!");
    }else{
    NSLog(@"刪除數(shù)據(jù)失敗!");
    }

}

//修改數(shù)據(jù)數(shù)據(jù)

  • (IBAction)updateClick:(id)sender {

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

    if (success == SQLITE_OK) {
    NSLog(@"修改數(shù)據(jù)成功!");
    }else{
    NSLog(@"修改數(shù)據(jù)失敗!");
    }
    }

//查詢數(shù)據(jù)

  • (IBAction)selectClick:(id)sender {

    //sql查詢語句
    NSString *sqlStr = @"SELECT NAME,AGE FROM t_test WHERE AGE = 30;";
    //定義存放結(jié)果數(shù)據(jù)stmt
    sqlite3_stmt *stmt = NULL;
    //取一條記錄
    int success = sqlite3_prepare_v2(_db, sqlStr.UTF8String,-1, &stmt, NULL);
    if (success == SQLITE_OK) {
    NSLog(@"查詢數(shù)據(jù)成功!");
    //拿數(shù)據(jù) 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);
    //打印結(jié)果
    NSLog(@"NAME = %@ AGE = %d",[NSString stringWithUTF8String:name],age);
    }

    }else{
    NSLog(@"查詢數(shù)據(jù)失敗!");
    }
    }```

FMDB的使用

因為自帶的SQLite3 是基于C語言的API,所以使用的時候有些繁瑣,而FMDB是面向?qū)ο蟮?使用起來更加的簡單,且只需要調(diào)用對應(yīng)的方法,傳值進去即可,而且還提供了多線程安全

FMDB下載地址

//  
//  ViewController.m  
//  數(shù)據(jù)庫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];  
      
    /*創(chuàng)建數(shù)據(jù)庫路徑*/  
    NSArray *path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);  
    NSString *documentDirectory = [path   objectAtIndex:0];  
    NSString *dbPath = [documentDirectory stringByAppendingPathComponent:@"test.db"];  
    NSLog(@"dbPath= %@",dbPath);  
    /*連接數(shù)據(jù)庫——連接數(shù)據(jù)庫,并沒有創(chuàng)建數(shù)據(jù)庫文件葛圃,直到打開成功才算真正創(chuàng)建*/  
    FMDatabase *db = [FMDatabase databaseWithPath:dbPath];  
    /*=====================================================================*/  
    /*創(chuàng)建表*/  
    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];  
    }  
      
    /*增刪改查——增加數(shù)據(jù)*/  
    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];  
    }  
      
    /*增刪改查——修改數(shù)據(jù)*/  
    if ([db open]) {  // 可以用任意類型的變量來拼接OC 字符串來作為數(shù)據(jù)庫查詢語句  
        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];  
    }  
      
     /*增刪改查——刪除數(shù)據(jù)*/  
    if ([db open]) {  // 第三種:不拼接NSString字符串千扔,直接寫語句  
        /* 
           必須注意:此種方式只有變量值可以用?來替代库正,別的(如表名曲楚,鍵名都必須直接寫在前面不能用格式化的方法來操作) 
         */  
        BOOL res = [db executeUpdate:@"DELETE FROM TABLENAME WHERE NAME = ?",@"張三"];  
        if (res) {  
            NSLog(@"delete success");  
        }  
        else  
        {  
          NSLog(@"delete Failure");  
        }  
        [db close];  
    }  
      
    /*數(shù)據(jù)庫查詢操作*/  
    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];  // 數(shù)值和NSString類型能自動進行轉(zhuǎn)換,不會出錯  
            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存在時就不會在執(zhí)行添加操作  
        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"替換,但為了調(diào)試方便重命名為原名  
        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];  
//    }  
    /*=====================================================================*/  
    /*數(shù)據(jù)庫多線程操作*/  
    /** 
     *如果應(yīng)用中使用了多線程操作數(shù)據(jù)庫背零,那么就需要使用FMDatabaseQueue來保證線程安全了。 應(yīng)用中不可在多個線程中共同使用一個FMDatabase對象操作數(shù)據(jù)庫趟大,這樣會引起數(shù)據(jù)庫數(shù)據(jù)混亂。 為了多線程操作數(shù)據(jù)庫安全铣焊,F(xiàn)MDB使用了FMDatabaseQueue逊朽,使用FMDatabaseQueue很簡單,首先用一個數(shù)據(jù)庫文件地址來初使化FMDatabaseQueue粗截,然后就可以將一個閉包(block)傳入inDatabase方法中惋耙。 在閉包中操作數(shù)據(jù)庫,而不直接參與FMDatabase的管理熊昌。 
     */  
    // 創(chuàng)建數(shù)據(jù)庫線程隊列  
    FMDatabaseQueue  *queue = [FMDatabaseQueue databaseQueueWithPath:dbPath];  
    dispatch_queue_t q1 = dispatch_queue_create("queue1", NULL);  // 創(chuàng)建第一個線程  
    dispatch_queue_t q2 = dispatch_queue_create("queue2", NULL);  // 創(chuàng)建第二個線程  
      
    dispatch_async(q1, ^{     // 表示線程q1同步執(zhí)行——即按先后順序執(zhí)行塊內(nèi)的操作绽榛,前一個操作完畢再執(zhí)行下一個操作  
        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  ```

數(shù)據(jù)庫查看軟件[下載]( dhttp://pan.baidu.com/share/link?shareid=2957954380&uk=489049396)

部分轉(zhuǎn)自(http://www.reibang.com/p/22198f89e422)
http://blog.csdn.net/liwei5bao/article/details/50297703


最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市婿屹,隨后出現(xiàn)的幾起案子灭美,更是在濱河造成了極大的恐慌,老刑警劉巖昂利,帶你破解...
    沈念sama閱讀 218,204評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件届腐,死亡現(xiàn)場離奇詭異铁坎,居然都是意外死亡,警方通過查閱死者的電腦和手機犁苏,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,091評論 3 395
  • 文/潘曉璐 我一進店門硬萍,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人围详,你說我怎么就攤上這事朴乖。” “怎么了助赞?”我有些...
    開封第一講書人閱讀 164,548評論 0 354
  • 文/不壞的土叔 我叫張陵买羞,是天一觀的道長。 經(jīng)常有香客問我雹食,道長畜普,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,657評論 1 293
  • 正文 為了忘掉前任群叶,我火速辦了婚禮吃挑,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘街立。我一直安慰自己儒鹿,他們只是感情好,可當我...
    茶點故事閱讀 67,689評論 6 392
  • 文/花漫 我一把揭開白布几晤。 她就那樣靜靜地躺著,像睡著了一般植阴。 火紅的嫁衣襯著肌膚如雪蟹瘾。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,554評論 1 305
  • 那天掠手,我揣著相機與錄音憾朴,去河邊找鬼。 笑死喷鸽,一個胖子當著我的面吹牛众雷,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播做祝,決...
    沈念sama閱讀 40,302評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼砾省,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了混槐?” 一聲冷哼從身側(cè)響起编兄,我...
    開封第一講書人閱讀 39,216評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎声登,沒想到半個月后狠鸳,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體揣苏,經(jīng)...
    沈念sama閱讀 45,661評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,851評論 3 336
  • 正文 我和宋清朗相戀三年件舵,在試婚紗的時候發(fā)現(xiàn)自己被綠了卸察。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,977評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡铅祸,死狀恐怖坑质,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情个少,我是刑警寧澤洪乍,帶...
    沈念sama閱讀 35,697評論 5 347
  • 正文 年R本政府宣布,位于F島的核電站夜焦,受9級特大地震影響壳澳,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜茫经,卻給世界環(huán)境...
    茶點故事閱讀 41,306評論 3 330
  • 文/蒙蒙 一巷波、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧卸伞,春花似錦抹镊、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,898評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至遂黍,卻和暖如春终佛,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背雾家。 一陣腳步聲響...
    開封第一講書人閱讀 33,019評論 1 270
  • 我被黑心中介騙來泰國打工铃彰, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人芯咧。 一個月前我還...
    沈念sama閱讀 48,138評論 3 370
  • 正文 我出身青樓牙捉,卻偏偏與公主長得像,于是被迫代替她去往敵國和親敬飒。 傳聞我的和親對象是個殘疾皇子邪铲,可洞房花燭夜當晚...
    茶點故事閱讀 44,927評論 2 355

推薦閱讀更多精彩內(nèi)容