SQLite3是嵌入到ios中的关系型数据库。对存储大规模的数据非常实用,使得不必将每个对象加到内存中。

支持NULL、INTEGER、REAL(浮点数字)、TEXT(字符串和文本)、BLOB(二进制对象)数据类型。

1、通过Firefox中的SQLite Manager组件创建一个数据库文件,并在其中创建创建一个Table,添加相关的参数。

2、将创建好的文件推到Xcode项目中,并在Link Binary With Libraries中添加libsqlite3.dylib依赖库。

3、新建一个C语言类型文件,对话框会自动添加桥接文件,在文件中添加语句#import<sqlite3.h>。

4、然后创建一个DataBaseOperations.swift文件来管理数据库。

在DataBaseOperation.swift文件中,申明一个数据库指针。

 //不透明指针,对应C语言里面的void *,这里指sqlite3指针
private var db:COpaquePointer = nil

创建一个初始化方法,在初始化方法中执行打开数据库操作。

//初始化方法打开数据库
required init(dbPath:String)
{
print("db path:" + dbPath) //String类的路径,转换成cString
let cpath = dbPath.cStringUsingEncoding(NSUTF8StringEncoding) //打开数据库
let error = sqlite3_open(cpath!, &db) //数据库打开失败处理
if error != SQLITE_OK {
sqlite3_close(db)
}
}
deinit{
self.colseDb()
} //关闭数据库
func colseDb(){ sqlite3_close(db)
}
//代码创建表
func createTable() -> Bool{ //sql语句
let sql = "CREATE TABLE UserTable(id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, username TEXT NOT NULL, password TEXT NOT NULL, email TEXT, age INTEGER)" //执行sql语句
let execResult = sqlite3_exec(db, sql.cStringUsingEncoding(NSUTF8StringEncoding)!, nil, nil, nil); //判断是否执行成功
if (execResult != SQLITE_OK) {
return false
} return true
}
//插入一条信息
func addUser(user: Person) -> Bool
{
//sql语句
let sql = "INSERT INTO UserTable (username, password, email, age) VALUES (?, ?, ?, ?);";
//sql语句转换成cString类型 let cSql = sql.cStringUsingEncoding(NSUTF8StringEncoding) //sqlite3_stmt 指针
var stmt:COpaquePointer = nil //1.编译sql
let prepare_result = sqlite3_prepare_v2(self.db, cSql!, -, &stmt, nil) //判断如果失败,获取失败信息
if prepare_result != SQLITE_OK {
sqlite3_finalize(stmt)
if let error = String.fromCString(sqlite3_errmsg(self.db)) {
let msg = "SQLiteDB - failed to prepare SQL: \(sql), Error: \(error)"
print(msg)
self.alert(msg)
}
return false
} //2.bind 绑定参数
//第2个参数:索引从1开始
//最后一个参数为函数指针
sqlite3_bind_text(stmt, , user.name!.cStringUsingEncoding(NSUTF8StringEncoding)!, -, nil);
sqlite3_bind_text(stmt, , user.password!.cStringUsingEncoding(NSUTF8StringEncoding)!, -, nil);
sqlite3_bind_text(stmt, , user.email!.cStringUsingEncoding(NSUTF8StringEncoding)!, -, nil);
sqlite3_bind_int(stmt, , CInt(user.age!)); //3.step执行
let step_result = sqlite3_step(stmt) //判断执行结果,如果失败,获取失败信息
if step_result != SQLITE_OK && step_result != SQLITE_DONE {
sqlite3_finalize(stmt)
if let err = String.fromCString(sqlite3_errmsg(self.db)) {
let msg = "SQLiteDB - failed to execute SQL: \(sql), Error: \(err)"
print(msg)
self.alert(msg)
}
return false
} //4.finalize
sqlite3_finalize(stmt); return true
}
//查询
func readAllUsers() -> [Person]{ //声明一个Person对象数组(查询的信息会添加到该数组)
var usersArr = [Person]() //查询sql语句
let sql = "SELECT * FROM UserTable;"; //sqlite3_stmt 指针
var stmt:COpaquePointer = nil
let cSql = sql.cStringUsingEncoding(NSUTF8StringEncoding) //1.编译sql
let prepare_result = sqlite3_prepare_v2(self.db, cSql!, -, &stmt, nil)
if prepare_result != SQLITE_OK {
sqlite3_finalize(stmt)
if let error = String.fromCString(sqlite3_errmsg(self.db)) {
let msg = "SQLiteDB - failed to prepare SQL: \(sql), Error: \(error)"
print(msg)
self.alert(msg)
}
return usersArr
} //2.step
while (sqlite3_step(stmt) == SQLITE_ROW) {
let user = Person() //循环 从数据库获取数据,添加到数组中
let cName = UnsafePointer<CChar>(sqlite3_column_text(stmt, ))
let cPwd = UnsafePointer<CChar>(sqlite3_column_text(stmt, ))
let cEmail = UnsafePointer<CChar>(sqlite3_column_text(stmt, ))
let cAge = sqlite3_column_int(stmt, ) user.name = String.fromCString(cName)
user.password = String.fromCString(cPwd)
user.email = String.fromCString(cEmail)
user.age = Int(cAge) usersArr += [user]
} //3.finalize
sqlite3_finalize(stmt); return usersArr
}
//更新一条信息
func updateUser(name: String , toName:String) -> Bool
{
//更新sql语句
let sql = "update UserTable set username = '\(toName)' where username = '\(name)'"; //sqlite3_stmt 指针
var stmt:COpaquePointer = nil
let cSql = sql.cStringUsingEncoding(NSUTF8StringEncoding) //1.编译sql
let prepare_result = sqlite3_prepare_v2(self.db, cSql!, -, &stmt, nil) //判断如果失败,获取失败信息
if prepare_result != SQLITE_OK {
sqlite3_finalize(stmt)
if let error = String.fromCString(sqlite3_errmsg(self.db)) {
let msg = "SQLiteDB - failed to prepare SQL: \(sql), Error: \(error)"
print(msg)
self.alert(msg)
}
return false
} //2.step执行
let step_result = sqlite3_step(stmt) //判断执行结果,如果失败,获取失败信息
if step_result != SQLITE_OK && step_result != SQLITE_DONE {
sqlite3_finalize(stmt)
if let err = String.fromCString(sqlite3_errmsg(self.db)) {
let msg = "SQLiteDB - failed to execute SQL: \(sql), Error: \(err)"
print(msg)
self.alert(msg)
}
return false
} //4.finalize
sqlite3_finalize(stmt); return true
}
//删除一条信息
func deleteUser(username: String) -> Bool
{
//删除sql语句
let sql = "delete from UserTable where username = '\(username)'"; //sqlite3_stmt 指针
var stmt:COpaquePointer = nil
let cSql = sql.cStringUsingEncoding(NSUTF8StringEncoding) //1.编译sql
let prepare_result = sqlite3_prepare_v2(self.db, cSql!, -, &stmt, nil) //判断如果失败,获取失败信息
if prepare_result != SQLITE_OK {
sqlite3_finalize(stmt)
if let error = String.fromCString(sqlite3_errmsg(self.db)) {
let msg = "SQLiteDB - failed to prepare SQL: \(sql), Error: \(error)"
print(msg)
self.alert(msg)
}
return false
} //3.step执行
let step_result = sqlite3_step(stmt) //判断执行结果,如果失败,获取失败信息
if step_result != SQLITE_OK && step_result != SQLITE_DONE {
sqlite3_finalize(stmt)
if let err = String.fromCString(sqlite3_errmsg(self.db)) {
let msg = "SQLiteDB - failed to execute SQL: \(sql), Error: \(err)"
print(msg)
self.alert(msg)
}
return false
} //4.finalize
sqlite3_finalize(stmt); return true
}
//定义一个报警器
func alert(msg:String) {
dispatch_async(dispatch_get_main_queue()) {
let alert = UIAlertView(title: "SQLiteDB", message:msg, delegate: nil, cancelButtonTitle: "OK")
alert.show()
}
}

因为工程数据库文件打包之后,会在NSBundle.mainBundle()路径下,该路径是只读的,不允许修改,所以必须把该路径下的数据库拷贝一份到Documents路径下,以后整个工程都将操作Documents路径下的数据库。

在ViewController中,添加一下代码。

//声明一个Documents下的路径
let dbPath = NSHomeDirectory() + "/Documents/RWDataTest.sqlite" //判断数据库文件是否存在
if !NSFileManager.defaultManager().fileExistsAtPath(dbPath)
{
//获取安装包内数据库路径
let bundleDBPath = NSBundle.mainBundle().pathForResource("RWDataTest", ofType: "sqlite")! //将安装包内数据库拷贝到Documents目录下
do
{
try NSFileManager.defaultManager().copyItemAtPath(bundleDBPath, toPath: dbPath)
}
catch let error as NSError {
print(error)//如果创建失败,error 会返回错误信息
}
}
//打开数据库
let dbOperation = DatabaseOperations(dbPath: dbPath) //添加一张表
dbOperation.createTable(); //插入一条信息, 通过Person对象来传值
let person:Person = Person(name: "刘明洋", pwd: "liumingyang", email: "liumingyang@leadingdo.com", age: ) dbOperation.addUser(person) //查询
let personArray:[Person] = dbOperation.readAllUsers()
print("共搜索到:\(personArray.count) 条数据" ) //更新
dbOperation.updateUser("刘明洋", toName: "刘蕙通") //删除
dbOperation.deleteUser("刘蕙通") //关闭数据库
dbOperation.colseDb()

读写应用程序数据-SQLite3的更多相关文章

  1. 读写应用程序数据-NSUserDefault、对象归档(NSKeyedArchiver)、文件操作

    ios中数据持久化存储方式一般有5种:NSUserDefault.对象归档(NSKeyedArchiver).文件操作.数据库存储(SQLite3).CoreData. 1.NSUserDefault ...

  2. 读写应用程序数据-CoreData

    coreData数据最终的存储类型可以是:SQLite数据库.XML.二进制.内存里.自定义的数据类型. 和SQLite区别:只能取出整个实体记录,然后分解,之后才能得到实体的某个属性. 1.创建工程 ...

  3. C# 读写西门子PLC数据,包含S7协议和Fetch/Write协议,s7支持200smart,300PLC,1200PLC,1500PLC

    本文将使用一个gitHub开源的组件技术来读写西门子plc数据,使用的是基于以太网的TCP/IP实现,不需要额外的组件,读取操作只要放到后台线程就不会卡死线程,本组件支持超级方便的高性能读写操作 官方 ...

  4. python 读写三菱PLC数据,使用以太网读写Q系列,L系列,Fx系列的PLC数据

    本文将使用一个gitHub开源的组件技术来读写三菱的plc数据,使用的是基于以太网的TCP/IP实现,不需要额外的组件,读取操作只要放到后台线程就不会卡死线程,本组件支持超级方便的高性能读写操作 gi ...

  5. java android 读写西门子PLC数据,包含S7协议和Fetch/Write协议,s7支持200smart,300PLC,1200PLC,1500PLC

    本文将使用一个gitHub开源的组件技术来读写西门子plc数据,使用的是基于以太网的TCP/IP实现,不需要额外的组件,读取操作只要放到后台线程就不会卡死线程,本组件支持超级方便的高性能读写操作 gi ...

  6. C#读写西门子PLC数据

    C#读写西门子PLC数据,包含S7协议和Fetch/Write协议,s7支持200smart,300PLC,1200PLC,1500PLC 本文将使用一个gitHub开源的组件技术来读写西门子plc数 ...

  7. C#读写三菱PLC数据 使用TCP/IP 协议

    本文将使用一个Github开源的组件库技术来读写三菱PLC和西门子plc数据,使用的是基于以太网的TCP/IP实现,不需要额外的组件,读取操作只要放到后台线程就不会卡死线程,本组件支持超级方便的高性能 ...

  8. 由于检索用户的本地应用程序数据路径时出错,导致无法生成 SQL Server 的用户实例

    /”应用程序中的服务器错误. 由于检索用户的本地应用程序数据路径时出错,导致无法生成 SQL Server 的用户实例.请确保该用户在此计算机上有本地用户配置文件.该连接将关闭. 堆栈跟踪: [Sql ...

  9. 微信小程序数据请求方法wx.request小测试

    微信小程序数据请求方法 wx.request wxml文件: <view> <textarea value="{{textdata}}"/> </vi ...

随机推荐

  1. 企业2.0杀出一号种子选手 “Linkwedo”横空出世

    当下,最热门的话题就是企业2.0和1.0的新老交替,在过去的时间里OA在国内几乎是企业1.0的代名词,各大知名OA厂商一直占领着国内的企业市场,但企业2.0在全球越演越烈,甚至大有替代企业1.0的的迹 ...

  2. 【HDOJ】3016 Man Down

    线段树+spfa求最长路.逆向思维,从最底下一块板子建图.需要注意的是任何一个板子掉落下面再无板子,此时都可以看做一个终结状态. /* 3016 */ #include <iostream> ...

  3. Android开发之PendingIntent的使用

    PendingIntent,待确定的意图,等待的意图 官网链接:http://developer.android.com/reference/android/app/PendingIntent.htm ...

  4. 转:Apache和Nginx运行原理解析

    Web服务器 Web服务器也称为WWW(WORLD WIDE WEB)服务器,主要功能是提供网上信息浏览服务. 应用层使用HTTP协议. HTML文档格式. 浏览器统一资源定位器(URL). Web服 ...

  5. Eclipse反编译插件: Jodeclipse与JadClipse

    Eclipse反编译插件: Jodeclipse与JadClipse Jodeclipse 是Jode的Eclipse插件,JadClipse是Jad的Eclipse插件,它们都是非常好的反编译插件. ...

  6. Oracle行转列的函数

    --行转列的函数-- CREATE OR REPLACE FUNCTION Calvin( col IN VARCHAR2,dw IN VARCHAR2) RETURN VARCHAR2 IS ret ...

  7. [NOIP2002]自由落体

    NOIp2002提高组 题目描述 在高为 H 的天花板上有 n 个小球,体积不计,位置分别为 0,1,2,….n-1.在地面上有一个小车(长为 L,高为 K,距原点距离为 S1).已知小球下落距离计算 ...

  8. 中断下半部-tasklet

    http://edsionte.com/techblog/ tasklet的实现 tasklet(小任务)机制是中断处理下半部分最常用的一种方法,其使用也是非常简单的.正如在前文中你所知道的那样,一个 ...

  9. 常用oracle查询总结

    --查询表空间使用情况 SELECT UPPER(F.TABLESPACE_NAME) "表空间名", D.TOT_GROOTTE_MB "表空间大小(M)", ...

  10. 解决 Provider 'System.Data.SqlServerCe.3.5' not installed. -摘自网络

    在64位机器上开发,如果使用到SqlServerCe的话,那么很可能会碰到这个问题,问题有两个方面: 1.如提示所云,没有安装SqlServerCe,只要去微软下载就好了. 2.系统已经安装SqlSe ...