swift3.0:sqlite3的使用
介绍
一、sqlite是纯C语言中底层的数据库,在OC和Swift中都是经常使用的数据库,在开发中,可以使用代码创建数据库,可以使用图形化界面创建数据库。例如SQLiteManager、SQLiteStudio等

二、对常用的一些方法进行解释如下:
OpaquePointer: *db,数据库句柄,跟文件句柄FIFL类似,这里是sqlite3指针;
sqlite3_stmt: *stmt,相当于ODBC的Command对象,用于保存编译好的SQL语句;
sqlite3_open(): 打开数据库,没有数据库时创建;
sqlite3_exec(): 执行非查询的SQL语句;
sqlite3_step(): 在调用sqlite3_prepare后,使用这个函数在记录集中移动;
sqlite3_close():关闭数据库文件;
sqlite3_column_text():取text类型的数据;
sqlite3_column_blob():取blob类型的数据;
sqlite3_column_int():取int类型的数据;
三、使用SQLiteStudio创建数据库,然后导出到桌面,再拖到项目中,最后通过代码拷贝到Documens下进行操作,并获取数据库路径

Person.swift
// Person.swift
// swiftDemo
//
// Created by 夏远全 on 2017/2/20.
// Copyright © 2017年 夏远全. All rights reserved.
// import UIKit class Person: NSObject { var name:String?
var password:String?
var email:String?
var age:Int?
}
DatabaseOperation.swift
(1)打开数据库
//不透明指针,对应C语言中的void *,这里指sqlit3指针
private var db:OpaquePointer? = nil //初始化方法打开数据库
required init(dbPath:String) {
print("db path:" + dbPath) //String类的路径,转换成cString
let cpath = dbPath.cString(using: .utf8) //打开数据库
let error = sqlite3_open(cpath!,&db) //数据库打开失败
if error != SQLITE_OK {
sqlite3_close(db)
}
}
(2)关闭数据库
//关闭数据库
deinit {
self.closeDB()
}
func closeDB() -> Void {
sqlite3_close(db)
}
(3)创建表
//创建表
func creatTable() -> Bool {
//sql语句
let sql = "CREATE TABLE UserTable(id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,username TEXT NOT NULL, password TEXT NOT NULL,email TEXT NOT NULL,age INTEGER)" //执行sql语句
let excuResult = sqlite3_exec(db, sql.cString(using: .utf8), nil, nil, nil) //判断是否执行成功
if excuResult != SQLITE_OK {
return false
}
return true
}
(4)插入数据
//插入数据
func addUser(user:Person) -> Bool { //sql语句
let sql = "INSERT INTO UserTable(username,password,email,age) VALUES(?,?,?,?);"; //sql语句转换成cString类型
let cSql = sql.cString(using: .utf8) //sqlite3_stmt指针
var stmt:OpaquePointer? = nil //编译
let prepare_result = sqlite3_prepare_v2(self.db, cSql!, -1, &stmt, nil) //判断如果失败,获取失败信息
if prepare_result != SQLITE_OK {
sqlite3_finalize(stmt)
if (sqlite3_errmsg(self.db)) != nil {
let msg = "SQLiteDB - failed to prepare SQL:\(sql)"
print(msg)
}
return false
} //绑定参数
//第二个参数,索引从1开始
//最后一个参数为函数指针
sqlite3_bind_text(stmt, 1, user.name!.cString(using: .utf8), -1, nil)
sqlite3_bind_text(stmt, 2, user.password!.cString(using: .utf8), -1, nil)
sqlite3_bind_text(stmt, 3, user.email!.cString(using: .utf8), -1, nil)
sqlite3_bind_int(stmt, 4, Int32(Int(user.age!))) //step执行
let step_result = sqlite3_step(stmt) //判断执行结果
if step_result != SQLITE_OK && step_result != SQLITE_DONE {
sqlite3_finalize(stmt)
if (sqlite3_errmsg(self.db)) != nil {
let msg = "SQLiteDB - failed to execute SQL:\(sql)"
print(msg)
}
return false
} //finalize
sqlite3_finalize(stmt) return true
}
(5)查询数据
//查询数据
func readAllUsers() -> [Person] { //声明一个Person对象数组(查询的信息会添加到数组中)
var userArr = [Person]() //查询sql语句
let sql = "SELECT * FROM UserTable;"; //sqlite3_stmt指针
var stmt:OpaquePointer? = nil
let cSql = sql.cString(using: .utf8) //编译
let prepare_result = sqlite3_prepare_v2(self.db, cSql!, -1, &stmt, nil)
if prepare_result != SQLITE_OK {
sqlite3_finalize(stmt)
if (sqlite3_errmsg(self.db)) != nil {
let msg = "SQLiteDB - failed to prepare SQL:\(sql)"
print(msg)
}
} //step
while (sqlite3_step(stmt) == SQLITE_ROW) {
let user = Person() //循环从数据库数据,添加到数组
let cName = UnsafePointer(sqlite3_column_text(stmt, 0))
let cPwd = UnsafePointer(sqlite3_column_text(stmt, 1))
let cEmail = UnsafePointer(sqlite3_column_text(stmt, 2))
let cAge = sqlite3_column_int(stmt, 3) user.name = String.init(cString: cName!)
user.password = String.init(cString: cPwd!)
user.email = String.init(cString: cEmail!)
user.age = Int(cAge) userArr += [user]
} //finalize
sqlite3_finalize(stmt) return userArr
}
(6)更新数据
//更新数据
func updateUser(name:String,toName:String) -> Bool { //更新sql语句
let sql = "update UserTable set username = '\(toName)' where username = '\(name)'"; //sqlite3_stmt指针
var stmt:OpaquePointer? = nil
let cSql = sql.cString(using: .utf8) //编译sql
let prepare_result = sqlite3_prepare_v2(self.db, cSql!, -1, &stmt, nil) //判断如果失败,获取失败信息
if prepare_result != SQLITE_OK {
sqlite3_finalize(stmt)
if (sqlite3_errmsg(self.db)) != nil {
let msg = "SQLiteDB - failed to prepare SQL:\(sql)"
print(msg)
}
return false
} //step执行
let step_result = sqlite3_step(stmt) //判断执行结果,如果失败,获取失败信息
if step_result != SQLITE_OK && step_result != SQLITE_DONE {
sqlite3_finalize(stmt)
if (sqlite3_errmsg(self.db)) != nil {
let msg = "SQLiteDB - failed to execute SQL:\(sql)"
print(msg)
}
return false
} //finalize
sqlite3_finalize(stmt) return true
}
(7)删除数据
//删除数据
func deleteUser(username:String) -> Bool { //删除sql语句
let sql = "delete from UserTable where username = '\(username)'"; //sqlite3_stmt指针
var stmt:OpaquePointer? = nil
let cSql = sql.cString(using: .utf8) //编译sql
let prepare_result = sqlite3_prepare_v2(self.db, cSql!, -1, &stmt, nil) //判断如果失败,获取失败信息
if prepare_result != SQLITE_OK {
sqlite3_finalize(stmt)
if (sqlite3_errmsg(self.db)) != nil {
let msg = "SQLiteDB - failed to prepare SQL:\(sql)"
print(msg)
}
return false
} //step执行
let step_result = sqlite3_step(stmt) //判断执行结果,如果失败,获取失败信息
if step_result != SQLITE_OK && step_result != SQLITE_DONE {
sqlite3_finalize(stmt)
if (sqlite3_errmsg(self.db)) != nil {
let msg = "SQLiteDB - failed to execute SQL:\(sql)"
print(msg)
}
return false
} //finalize
sqlite3_finalize(stmt) return true
}
(8)复制数据库路径
//将Bundle.main路径下的数据库文件复制到Documents下
class func loadDBPath() -> String { //声明一个Documents下的路径
let dbPath = NSHomeDirectory() + "/Documents/RWDataTest.db" //判断数据库文件是否存在
if !FileManager.default.fileExists(atPath: dbPath) { //获取安装包内是否存在
let bundleDBPath = Bundle.main.path(forResource: "RWDataTest", ofType:"db")! //将安装包内的数据库到Documents目录下
do {
try FileManager.default.copyItem(atPath: bundleDBPath, toPath: dbPath)
} catch let error as NSError {
print(error)
}
} return dbPath
}
ViewController.swift测试
// Created by 夏远全 on 2017/1/13.
// Copyright © 2017年 夏远全. All rights reserved.
// import UIKit class ViewController: UIViewController { override func viewDidLoad() {
super.viewDidLoad() //打开数据库
let path = DatabaseOperations.loadDBPath()
let dbOpearion = DatabaseOperations.init(dbPath:path)
print(path) //添加一张表
let person:Person = Person()
person.name = "张三"
person.password = "123566"
person.email = "zhangsan@163.com"
person.age = 30 //插入一条信息,通过Person对象来传值
let addBool = dbOpearion.addUser(user: person)
print(addBool) //查询
let personArray:[Person] = dbOpearion.readAllUsers()
print("共搜索到:\(personArray.count) 条数据") //更新
let updateBool = dbOpearion.updateUser(name: "张三", toName: "李四")
print(updateBool) //删除
let deleteBool = dbOpearion.deleteUser(username: "李四")
print(deleteBool) //关闭数据库
dbOpearion.closeDB()
}
}
swift3.0:sqlite3的使用的更多相关文章
- Swift3.0服务端开发(一) 完整示例概述及Perfect环境搭建与配置(服务端+iOS端)
本篇博客算是一个开头,接下来会持续更新使用Swift3.0开发服务端相关的博客.当然,我们使用目前使用Swift开发服务端较为成熟的框架Perfect来实现.Perfect框架是加拿大一个创业团队开发 ...
- 算法与数据结构(十三) 冒泡排序、插入排序、希尔排序、选择排序(Swift3.0版)
本篇博客中的代码实现依然采用Swift3.0来实现.在前几篇博客连续的介绍了关于查找的相关内容, 大约包括线性数据结构的顺序查找.折半查找.插值查找.Fibonacci查找,还包括数结构的二叉排序树以 ...
- Swift3.0变化分享
Swift 3.0 做出的改变很大,在这篇文章中,我将尽我所能,利用代码样例给大家解释Swift 3.0最重要(要命)的改变,希望大家能够做好升级Swift 3.0 的准备.Swift 3.0的改变不 ...
- swift3.0变化总结
Swift 3.0 做出的改变很大,在这篇文章中,我将尽我所能,利用代码样例给大家解释Swift 3.0最重要(要命)的改变,希望大家能够做好升级Swift 3.0 的准备.Swift 3.0的改变不 ...
- 关于for循环------swift3.0
在程序开发当中,for循环使用的频率无疑是最高的.常用的swift循环是递增式遍历.当然各种循环,swift都能办到.但其大多采用关键字形式实现,大部分开发者更喜欢直接使用C式循环代码.在swift3 ...
- Swift2.3 --> Swift3.0 的变化
Swift3.0语法变化 首先和大家分享一下学习新语法的技巧: 用Xcode8打开自己的Swift2.3的项目,选择Edit->Convert->To Current Swift Synt ...
- Swift3.0都有哪些变化
从写第一篇Swift文章的时候到现在Swift已经从1.2发展到了今天的3.0,这期间由于Swift目前还在发展阶段并不能向下兼容,因此第一篇文章中的部分代码在当前的Xcode环境中已经无法运行.在W ...
- iOS开发 swift3.0中文版
swift3.0中文版: http://pan.baidu.com/s/1nuHqrBb
- swift3.0的改变
Swift在这2年的时间内,发展势头迅猛,在它开源后,更是如井喷一样,除了 iOS.mac 平台,还支持了 Linux. 而今年下半年, Swift 3.0 也会随之发布.https://github ...
- Swift3.0语言教程字符串与URL的数据转换与自由转换
Swift3.0语言教程字符串与URL的数据转换与自由转换 Swift3.0语言教程字符串与URL的数据转换 Swift3.0语言教程字符串与URL的数据转换与自由转换,字符串中的字符永久保存除了可以 ...
随机推荐
- 001.Postfix简介
一 简介 postfix是Wietse Venema在IBM的GPL协议之下开发的MTA(邮件传输代理)软件.Postfix提供更快.更容易管理.更安全,同时还与 sendmail保持足够的兼容性,是 ...
- Netty源码分析之NioEventLoop(二)—NioEventLoop的启动
上篇文章中我们对Netty中NioEventLoop创建流程与源码进行了跟踪分析.本篇文章中我们接着分析NioEventLoop的启动流程: Netty中会在服务端启动和新连接接入时通过chooser ...
- 【BZOJ 4818】 4818: [Sdoi2017]序列计数 (矩阵乘法、容斥计数)
4818: [Sdoi2017]序列计数 Time Limit: 30 Sec Memory Limit: 128 MBSubmit: 560 Solved: 359 Description Al ...
- 【原】Spring整合Redis(第一篇)—SDR简述
1.SDR说明 Spring Data Redis(SDR),是SpringFramework提供的一套简化访问Redis的API,是对Jedis的又一层封装. SDR集成了Jedis,JRedis, ...
- 一次MySQL异常排查:Query execution was interrupted
异常日志: 查询被中断了,先是在Google上查,又是再百度上查,基本上都是说程序超时设置setQueryTimeout的问题,就是说查询时间超过了设置的最大查询时间,导致查询被中断.我也没办法断定是 ...
- [原创]用Charles模拟App各种网络带宽测试介绍
[原创]用Charles模拟App各种网络带宽测试介绍 相信每个测试在进行自己公司App测试时,都会碰到一个问题,如何去模拟各种App在各种带宽下的测试情况,估计很少有公司直接去采用2g/3g/4g卡 ...
- GET POST方法长度限制(转)
1. Get方法长度限制 Http Get方法提交的数据大小长度并没有限制,HTTP协议规范没有对URL长度进行限制.这个限制是特定的浏览器及服务器对它的限制. 如:IE对URL长度的限制是20 ...
- HDU 4790 Just Random (2013成都J题)
Just Random Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- There are no packages available for install
解决方法: ·删除sublime Text 安装目录下Data->Packages目录下的Package Control(如果没有,略过此步骤). ·下载Package Control,下载路径 ...
- Programming 2D Games 读书笔记(第六章)
http://www.programming2dgames.com/chapter6.htm 示例一:Bounce 边界碰撞测试 velocity为移动的速度, 超过右边界,velocity.x为 ...