SwiftUI一起学之七 -- Sqlite数据库

一 学习目标

学会操作数据库,完成新增,删除,查询功能

二 学习效果

image.png
image.png

三 主要操作步骤

3.1 使用Swift Package Manager添加SQLite.swift

SQLite.swift库的地址: https://github.com/stephencelis/SQLite.swift

image.png
image.png

3.2 创建数据库

private var db:Connection?

let path = NSSearchPathForDirectoriesInDomains(
                .documentDirectory, .userDomainMask, true
                ).first!
db = try! Connection("\(path)/db.sqlite3")
db?.busyTimeout = 5.0

3.3 创建表

private var usersTable:Table?
let id_column = Expression<Int64>("id")
let name_column = Expression<String>("name")

usersTable = Table("records")
try! getDB().run(
    usersTable!.create (ifNotExists: true, block:{ t in
            t.column(id_column, primaryKey: true)
            t.column(name_column, unique: true)
          })
)

3.4 新增数据

let insert = usersTable.insert(name_column <- name)
if let rowId = try? db.run(insert) {
    print("插入成功:\(rowId)")
    return true
} else {
    print("插入失败")
    return false
}

3.5 删除数据

let alice = usersTable.filter(Expression<String>("name") == user.name!)
if let count = try? db.run(alice.delete()) {
    return count>0
} else {
    return false
}

3.6 查询数据

let query = usersTable.order(id_column.desc)
        
do {
    let results = try db.prepare(query);

    for userRow in results {
        print("name = \(userRow[name_column])")
    }
} catch {
    print("出错")
}
return users;

四 完整项目代码

4.1 SwiftUiDemo.swift

import SwiftUI

struct SqliteUIView: SwiftUI.View  {
    @ObservedObject var sqliteViewModel = SqliteViewModel.single
    @State private var showingAlert = false
    @State var userName:String = ""
    @State var users: [UserModel] = []
    
    var body: some SwiftUI.View  {
        VStack{
            Text("用户数据库").font(.system(size:20))
            HStack{
                // 输入框
                TextField("输入用户名", text: $userName)
                    .padding(EdgeInsets(top: 5, leading: 20, bottom: 5, trailing: 20))
                    .background(Color.yellow)
                    .cornerRadius(10)
                    .frame(width: 150)
                // 插入
                Button.init("新增") {
                    let success = sqliteViewModel.insert(name: userName)
                    if(success){
                        // 成功插入后查询
                        users = sqliteViewModel.getUsers()
                        print(users)
                    }
                }
                .padding(EdgeInsets(top: 5, leading: 20, bottom: 5, trailing: 20))
                .background(Color.green)
                .cornerRadius(20)
                // 查询
                Button.init("查询") {
                    users = sqliteViewModel.getUsers()
                }
                .padding(EdgeInsets(top: 5, leading: 20, bottom: 5, trailing: 20))
                .background(Color.green)
                .cornerRadius(20)
            }
            // 列表
            List(users){ user in
                UserRow(user: user){ _ in
                    // 删除
                    let success = SqliteViewModel.single.delete(user: user)
                    if(success){
                        // 成功插入后查询
                        users = sqliteViewModel.getUsers()
                        print(users)
                    }
                }
            }
        }.onAppear {
            users = sqliteViewModel.getUsers()
        }
    }
}

// 列表项
struct UserRow: View {
    var user: UserModel
    var deleteCb:(_ user:UserModel)->()
    
    var body: some View {
        HStack{
            Text("用户名 =  \(user.name!)")
            Spacer()
            Button.init("删除") {
                deleteCb(user)
            }
            .padding(EdgeInsets(top: 5, leading: 20, bottom: 5, trailing: 20))
            .background(Color.red)
            .cornerRadius(20)
        }
    }
}

4.2 SqliteViewModel.swift

import Foundation
import SQLite

let id_column = Expression<Int64>("id")
let name_column = Expression<String>("name")

class SqliteViewModel: ObservableObject {
    static let single = SqliteViewModel()
    
    private var db:Connection?
    private var usersTable:Table?
    
    func getDB() -> Connection {
        if db == nil {
            let path = NSSearchPathForDirectoriesInDomains(
                .documentDirectory, .userDomainMask, true
                ).first!
            db = try! Connection("\(path)/db.sqlite3")
            db?.busyTimeout = 5.0
        }
        return db!
    }
    
    func getUserTable() -> Table {
        if usersTable == nil {
            usersTable = Table("records")
            try! getDB().run(
                usersTable!.create (ifNotExists: true, block:{ t in
                    t.column(id_column, primaryKey: true)
                    t.column(name_column, unique: true)
                })
            )
        }
        return usersTable!
    }
    
    // 增加
    func insert(name:String) -> Bool{
        let insert = getUserTable().insert(name_column <- name)
        if let rowId = try? getDB().run(insert) {
            print("插入成功:\(rowId)")
            return true
        } else {
            print("插入失败")
            return false
        }
    }
    
    // 查询
    func getUsers() -> [UserModel]{
        var users: [UserModel] = []
        users.removeAll()
        let query = getUserTable().order(id_column.desc)
        
        do {
            let results = try getDB().prepare(query);
            
            for userRow in results {
                print("name = \(userRow[name_column])")
                let user = UserModel(name: userRow[name_column])
                users.append(user)
            }
        } catch {
            print("出错")
        }
        return users;
    }
    
    // 删除
    func delete(user:UserModel)-> Bool{
        let alice = getUserTable().filter(Expression<String>("name") == user.name!)
        if let count = try? getDB().run(alice.delete()) {
            return count>0
        } else {
            return false
        }
    }
}

4.3 Model.swift

import Foundation

struct UserModel: Identifiable {
    var id = UUID()
    var email : String?
    var name : String?
}

参考:

  1. SQLite.swift
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容