iOS:SQLite詳解

一、SQLite概念

SQLite是輕量型關系型數(shù)據(jù)庫账阻;占用資源非常少充坑;是無類型的數(shù)據(jù)庫(意思是可以保存所有類型的數(shù)據(jù))减江。

二闻蛀、SQLite基本使用

1、建表命令:
create table 表名 (字段名1 字段類型1您市,字段名2 字段類型2觉痛,以此類推....);
create table if not exists (字段名1 字段類型1, 字段名2 字段類型2,以此類推 …);

(1)圖如下:

(2)代碼例子如下:

- (void)createTable{
//1.設計創(chuàng)建表的sql語句
  const char *sql = "CREATE TABLE IF NOT EXISTS t_Student(id integer PRIMARY KEY AUTOINCREMENT,name text NOT NULL ,score real DEFAULT 0,sex text DEFAULT '不明'); "  
//2.執(zhí)行sql語句
    int ret = sqlite3_exec(_db, sql, NULL, NULL, NULL);
   //3.判斷執(zhí)行結(jié)果
    if (ret == SQLITE_OK) {
        NSLog(@"創(chuàng)建表成功");
    }else{    
        NSLog(@"創(chuàng)建表失敗");
    }  
}
2茵休、插入命令
insert into 表名 (字段1, 字段2, 以此類推…) values (字段1的值, 字段2的值,以此類推 …) ;

(1)圖如下:

(2)代碼如下:

- (void)insertData{
//1.創(chuàng)建插入數(shù)據(jù)的sql語句
//===========插入單條數(shù)據(jù)=========
    const char *sql = "INSERT INTO t_Student(name,score,sex)VALUES ('小明',65,'男');";
//==========同時插入多條數(shù)據(jù)=======
    NSMutableString * mstr = [NSMutableString string];
  for (int i = 0; i < 50; i++) {
        NSString * name = [NSString stringWithFormat:@"name%d", i];
        CGFloat score = arc4random() % 101 * 1.0;
        NSString * sex = arc4random() % 2 == 0 ? @"男" : @"女";
        NSString * tsql = [NSString stringWithFormat:@"INSERT INTO t_Student         (name,score,sex) VALUES ('%@',%f,'%@');", name, score, sex];
        [mstr appendString:tsql];
    }  

    //將OC字符串轉(zhuǎn)換成C語言的字符串
    sql = mstr.UTF8String;

      //2.執(zhí)行sql語句
   int ret = sqlite3_exec(_db, sql, NULL, NULL, NULL);
   //3.判斷執(zhí)行結(jié)果
   if (ret==SQLITE_OK) {
        NSLog(@"插入成功");
    }else{
        NSLog(@"插入失敗");
    }
    

}
3薪棒、更新命令
update 表名 set 字段1 = 字段1的新值, 字段2 = 字段2的新值,以此類推 … ;
4、刪除命令
delete from 表名;注:刪除的是表中所有記錄
拓展:可以附加一些額外條件刪除
條件語句的操作:可以選擇性的刪除

where 字段 = 一個值 ; if(字段==某個值) // 不能用兩個 =
where 字段 is 一個值 ; // is 相當于 =
where 字段 is not一個值 ; // is not 相當于 !=
where 字段 > 一個值 ;
where 字段1 = 一個值 and 字段2 > 一個值 ; // and相當于C語言中的 &&
where 字段1 = 一個值 or 字段2 = 一個值 ; // or 相當于C語言中的 ||

(1)圖如下:

(2)代碼如下:

- (void)deleteData{
    //創(chuàng)建刪除數(shù)據(jù)的sql語句
    const char *sql = "DELETE FROM t_Student WHERE score < 10;";
    //執(zhí)行sql語句
    int ret = sqlite3_exec(_db, sql, NULL, NULL, NULL);
    //判斷執(zhí)行結(jié)果
    if (ret == SQLITE_OK) {
        NSLog(@"刪除成功");
    }else{
        NSLog(@"刪除失敗");
    }
}
5、查詢數(shù)據(jù)
SELECT 字段1,字段2 FROM 表明;

(1)圖如下:

(2)代碼如下:

- (void)selectData{
    //執(zhí)行數(shù)據(jù)庫語句
    const char *sql = "SELECT name,score FROM t_Student;";
    //結(jié)果集(用來收集結(jié)果)
    sqlite3_stmt * stmt;
    int ret = sqlite3_prepare_v2(_db, sql, -1, &stmt, NULL);
    if (ret == SQLITE_OK) {
        NSLog(@"查詢成功");
        //遍歷結(jié)果集拿到查詢到的數(shù)據(jù)
        //sqlite3_step獲取結(jié)果集中的數(shù)據(jù)
        while (sqlite3_step(stmt) == SQLITE_ROW) {
            //參數(shù)1:結(jié)果集
            //參數(shù)2:列數(shù)
            const unsigned char * name = sqlite3_column_text(stmt, 0);
            double score = sqlite3_column_double(stmt, 1);
            NSLog(@"%s %.2lf", name, score);
        }
    }else{
        NSLog(@"查詢失敗");
    }
}

三蔑穴、SQLite在項目中的使用

對于SQLite在項目中徙邻,定義一個單例,提供一個類方法可以供全局使用鹅巍,在編譯時初始化這個類,然后一直保存在內(nèi)存中,整個程序就會保持就有一個實例贸营,在App退出時,系統(tǒng)也會自動釋放這個內(nèi)存岩睁。

新建SQLiteManager

.h文件

@interface SQLiteManager : NSObject

+ (instancetype)shareInstance;
//打開數(shù)據(jù)庫钞脂,返回是布爾值
- (BOOL)openDB;
//執(zhí)行SQL語句,返回成功或者失敗
- (BOOL)execuSQL:(NSString *)SQL
//獲取結(jié)果
- (NSArray *)querySQL:(NSString *)SQL;

@end

.m實現(xiàn)

@interface SQLiteManager()
@property (nonatomic,assign) sqlite3 *db;

@end

@implementation SQLiteManager
static SQLiteManager *instance;
+ (instancetype)shareInstance{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        instance = [[self alloc] init];
    });
    return instance;
}

//打開數(shù)據(jù)庫捕儒,返回是布爾值
- (BOOL)openDB{
    //app內(nèi)數(shù)據(jù)庫文件存放路徑-一般存放在沙盒中
    NSString *documentPath = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];
    NSString *DBPath = [documentPath stringByAppendingPathComponent:@"appDB.sqlite"];
    if (sqlite3_open(DBPath.UTF8String, &_db) != SQLITE_OK) {
        //數(shù)據(jù)庫打開失敗
        return NO;
    }else{
        //打開成功創(chuàng)建表
        //用戶 表
        NSString *creatUserTable = @"CREATE TABLE IF NOT EXISTS 't_User' ( 'ID' INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,'name' TEXT,'age' INTEGER,'icon' TEXT);";
        //車 表
        NSString *creatCarTable = @"CREATE TABLE IF NOT EXISTS 't_Car' ('ID' INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,'type' TEXT,'output' REAL,'master' TEXT)";
        //項目中一般不會只有一個表
        NSArray *SQL_ARR = [NSArray arrayWithObjects:creatUserTable,creatCarTable, nil];
        return [self creatTableExecSQL:SQL_ARR];
    }
}

-(BOOL)creatTableExecSQL:(NSArray *)SQL_ARR{
    for (NSString *SQL in SQL_ARR) {
        //參數(shù)一:數(shù)據(jù)庫對象  參數(shù)二:需要執(zhí)行的SQL語句  其余參數(shù)不需要處理
        if (![self execuSQL:SQL]) {
            return NO;
        }
    }
    return YES;
}

#pragma 執(zhí)行SQL語句
- (BOOL)execuSQL:(NSString *)SQL{
    char *error;
    if (sqlite3_exec(self.db, SQL.UTF8String, nil, nil, &error) == SQLITE_OK) {
        return YES;
    }else{
        NSLog(@"SQLiteManager執(zhí)行SQL語句出錯:%s",error);
        return NO;
    }
}

#pragma mark - 查詢數(shù)據(jù)庫中數(shù)據(jù)
-(NSArray *)querySQL:(NSString *)SQL{
    //準備查詢
    // 1> 參數(shù)一:數(shù)據(jù)庫對象
    // 2> 參數(shù)二:查詢語句
    // 3> 參數(shù)三:查詢語句的長度:-1
    // 4> 參數(shù)四:句柄(游標對象)
    
    sqlite3_stmt *stmt = nil;
    if (sqlite3_prepare_v2(self.db, SQL.UTF8String, -1, &stmt, nil) != SQLITE_OK) {
        NSLog(@"準備查詢失敗!");
        return NULL;
    }
    //準備成功,開始查詢數(shù)據(jù)
    //定義一個存放數(shù)據(jù)字典的可變數(shù)組
    NSMutableArray *dictArrM = [[NSMutableArray alloc] init];
    while (sqlite3_step(stmt) == SQLITE_ROW) {
        //一共獲取表中所有列數(shù)(字段數(shù))
        int columnCount = sqlite3_column_count(stmt);
        //定義存放字段數(shù)據(jù)的字典
        NSMutableDictionary *dict = [[NSMutableDictionary alloc] init];
        for (int i = 0; i < columnCount; i++) {
            // 取出i位置列的字段名,作為字典的鍵key
            const char *cKey = sqlite3_column_name(stmt, i);
            NSString *key = [NSString stringWithUTF8String:cKey];
            
            //取出i位置存儲的值,作為字典的值value
            const char *cValue = (const char *)sqlite3_column_text(stmt, i);
            NSString *value = [NSString stringWithUTF8String:cValue];
            
            //將此行數(shù)據(jù) 中此字段中key和value包裝成 字典
            [dict setObject:value forKey:key];
        }
        [dictArrM addObject:dict];
    }
    return dictArrM;
}

參考:
https://www.cnblogs.com/guohai-stronger/p/9218175.html
https://zhuanlan.zhihu.com/p/208508908

?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末冰啃,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子刘莹,更是在濱河造成了極大的恐慌阎毅,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,490評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件点弯,死亡現(xiàn)場離奇詭異扇调,居然都是意外死亡,警方通過查閱死者的電腦和手機蒲拉,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評論 3 395
  • 文/潘曉璐 我一進店門肃拜,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人雌团,你說我怎么就攤上這事燃领。” “怎么了锦援?”我有些...
    開封第一講書人閱讀 165,830評論 0 356
  • 文/不壞的土叔 我叫張陵猛蔽,是天一觀的道長。 經(jīng)常有香客問我,道長曼库,這世上最難降的妖魔是什么区岗? 我笑而不...
    開封第一講書人閱讀 58,957評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮毁枯,結(jié)果婚禮上慈缔,老公的妹妹穿的比我還像新娘。我一直安慰自己种玛,他們只是感情好藐鹤,可當我...
    茶點故事閱讀 67,974評論 6 393
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著赂韵,像睡著了一般娱节。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上祭示,一...
    開封第一講書人閱讀 51,754評論 1 307
  • 那天肄满,我揣著相機與錄音,去河邊找鬼质涛。 笑死稠歉,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的蹂窖。 我是一名探鬼主播轧抗,決...
    沈念sama閱讀 40,464評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼瞬测!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起纠炮,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤月趟,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后恢口,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體孝宗,經(jīng)...
    沈念sama閱讀 45,847評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,995評論 3 338
  • 正文 我和宋清朗相戀三年耕肩,在試婚紗的時候發(fā)現(xiàn)自己被綠了因妇。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,137評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡猿诸,死狀恐怖婚被,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情梳虽,我是刑警寧澤址芯,帶...
    沈念sama閱讀 35,819評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響谷炸,放射性物質(zhì)發(fā)生泄漏北专。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,482評論 3 331
  • 文/蒙蒙 一旬陡、第九天 我趴在偏房一處隱蔽的房頂上張望拓颓。 院中可真熱鬧,春花似錦描孟、人聲如沸录粱。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,023評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽啥繁。三九已至,卻和暖如春青抛,著一層夾襖步出監(jiān)牢的瞬間旗闽,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,149評論 1 272
  • 我被黑心中介騙來泰國打工蜜另, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留适室,地道東北人。 一個月前我還...
    沈念sama閱讀 48,409評論 3 373
  • 正文 我出身青樓举瑰,卻偏偏與公主長得像捣辆,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子此迅,可洞房花燭夜當晚...
    茶點故事閱讀 45,086評論 2 355