iOS学习笔记(十六)——数据库操作(使用FMDB)

iOS中原生的SQLite API在使用上相当不友好,在使用时,非常不便。于是,就出现了一系列将SQLite API进行封装的库,例如FMDB、PlausibleDatabase、sqlitepersistentobjects等,FMDB (https://github.com/ccgus/fmdb) 是一款简洁、易用的封装库,这一篇文章简单介绍下FMDB的使用。

在FMDB下载文件后,工程中必须导入如下文件,并使用 libsqlite3.dylib 依赖包。

FMDB同时兼容ARC和非ARC工程,会自动根据工程配置来调整相关的内存管理代码。

FMDB常用类:

FMDatabase : 一个单一的SQLite数据库,用于执行SQL语句。
FMResultSet :执行查询一个FMDatabase结果集,这个和android的Cursor类似。
FMDatabaseQueue :在多个线程来执行查询和更新时会使用这个类。

创建数据库:

[cpp] view plaincopyprint?

  1. db = [FMDatabase databaseWithPath:database_path];

1、当数据库文件不存在时,fmdb会自己创建一个。

2、 如果你传入的参数是空串:@"" ,则fmdb会在临时文件目录下创建这个数据库,数据库断开连接时,数据库文件被删除。

3、如果你传入的参数是 NULL,则它会建立一个在内存中的数据库,数据库断开连接时,数据库文件被删除。

打开数据库:

[cpp] view plaincopyprint?

  1. [db open]

返回BOOL型。

关闭数据库:

[cpp] view plaincopyprint?

  1. [db close]

数据库增删改等操作:

除了查询操作,FMDB数据库操作都执行executeUpdate方法,这个方法返回BOOL型。

看一下例子:

创建表:

[cpp] view plaincopyprint?

  1. if ([db open]) {
  2. NSString *sqlCreateTable =  [NSString stringWithFormat:@"CREATE TABLE IF NOT EXISTS ‘%@‘ (‘%@‘ INTEGER PRIMARY KEY AUTOINCREMENT, ‘%@‘ TEXT, ‘%@‘ INTEGER, ‘%@‘ TEXT)",TABLENAME,ID,NAME,AGE,ADDRESS];
  3. BOOL res = [db executeUpdate:sqlCreateTable];
  4. if (!res) {
  5. NSLog(@"error when creating db table");
  6. } else {
  7. NSLog(@"success to creating db table");
  8. }
  9. [db close];
  10. }

添加数据:

[cpp] view plaincopyprint?

  1. if ([db open]) {
  2. NSString *insertSql1= [NSString stringWithFormat:
  3. @"INSERT INTO ‘%@‘ (‘%@‘, ‘%@‘, ‘%@‘) VALUES (‘%@‘, ‘%@‘, ‘%@‘)",
  4. TABLENAME, NAME, AGE, ADDRESS, @"张三", @"13", @"济南"];
  5. BOOL res = [db executeUpdate:insertSql1];
  6. NSString *insertSql2 = [NSString stringWithFormat:
  7. @"INSERT INTO ‘%@‘ (‘%@‘, ‘%@‘, ‘%@‘) VALUES (‘%@‘, ‘%@‘, ‘%@‘)",
  8. TABLENAME, NAME, AGE, ADDRESS, @"李四", @"12", @"济南"];
  9. BOOL res2 = [db executeUpdate:insertSql2];
  10. if (!res) {
  11. NSLog(@"error when insert db table");
  12. } else {
  13. NSLog(@"success to insert db table");
  14. }
  15. [db close];
  16. }

修改数据:

[cpp] view plaincopyprint?

  1. if ([db open]) {
  2. NSString *updateSql = [NSString stringWithFormat:
  3. @"UPDATE ‘%@‘ SET ‘%@‘ = ‘%@‘ WHERE ‘%@‘ = ‘%@‘",
  4. TABLENAME,   AGE,  @"15" ,AGE,  @"13"];
  5. BOOL res = [db executeUpdate:updateSql];
  6. if (!res) {
  7. NSLog(@"error when update db table");
  8. } else {
  9. NSLog(@"success to update db table");
  10. }
  11. [db close];
  12. }

删除数据:

[cpp] view plaincopyprint?

  1. if ([db open]) {
  2. NSString *deleteSql = [NSString stringWithFormat:
  3. @"delete from %@ where %@ = ‘%@‘",
  4. TABLENAME, NAME, @"张三"];
  5. BOOL res = [db executeUpdate:deleteSql];
  6. if (!res) {
  7. NSLog(@"error when delete db table");
  8. } else {
  9. NSLog(@"success to delete db table");
  10. }
  11. [db close];
  12. }

数据库查询操作:

查询操作使用了executeQuery,并涉及到FMResultSet。

[cpp] view plaincopyprint?

  1. if ([db open]) {
  2. NSString * sql = [NSString stringWithFormat:
  3. @"SELECT * FROM %@",TABLENAME];
  4. FMResultSet * rs = [db executeQuery:sql];
  5. while ([rs next]) {
  6. int Id = [rs intForColumn:ID];
  7. NSString * name = [rs stringForColumn:NAME];
  8. NSString * age = [rs stringForColumn:AGE];
  9. NSString * address = [rs stringForColumn:ADDRESS];
  10. NSLog(@"id = %d, name = %@, age = %@  address = %@", Id, name, age, address);
  11. }
  12. [db close];
  13. }

FMDB的FMResultSet提供了多个方法来获取不同类型的数据:

数据库多线程操作:

如果应用中使用了多线程操作数据库,那么就需要使用FMDatabaseQueue来保证线程安全了。 应用中不可在多个线程中共同使用一个FMDatabase对象操作数据库,这样会引起数据库数据混乱。 为了多线程操作数据库安全,FMDB使用了FMDatabaseQueue,使用FMDatabaseQueue很简单,首先用一个数据库文件地址来初使化FMDatabaseQueue,然后就可以将一个闭包(block)传入inDatabase方法中。 在闭包中操作数据库,而不直接参与FMDatabase的管理。

[cpp] view plaincopyprint?

  1. FMDatabaseQueue * queue = [FMDatabaseQueue databaseQueueWithPath:database_path];
  2. dispatch_queue_t q1 = dispatch_queue_create("queue1", NULL);
  3. dispatch_queue_t q2 = dispatch_queue_create("queue2", NULL);
  4. dispatch_async(q1, ^{
  5. for (int i = 0; i < 50; ++i) {
  6. [queue inDatabase:^(FMDatabase *db2) {
  7. NSString *insertSql1= [NSString stringWithFormat:
  8. @"INSERT INTO ‘%@‘ (‘%@‘, ‘%@‘, ‘%@‘) VALUES (?, ?, ?)",
  9. TABLENAME, NAME, AGE, ADDRESS];
  10. NSString * name = [NSString stringWithFormat:@"jack %d", i];
  11. NSString * age = [NSString stringWithFormat:@"%d", 10+i];
  12. BOOL res = [db2 executeUpdate:insertSql1, name, age,@"济南"];
  13. if (!res) {
  14. NSLog(@"error to inster data: %@", name);
  15. } else {
  16. NSLog(@"succ to inster data: %@", name);
  17. }
  18. }];
  19. }
  20. });
  21. dispatch_async(q2, ^{
  22. for (int i = 0; i < 50; ++i) {
  23. [queue inDatabase:^(FMDatabase *db2) {
  24. NSString *insertSql2= [NSString stringWithFormat:
  25. @"INSERT INTO ‘%@‘ (‘%@‘, ‘%@‘, ‘%@‘) VALUES (?, ?, ?)",
  26. TABLENAME, NAME, AGE, ADDRESS];
  27. NSString * name = [NSString stringWithFormat:@"lilei %d", i];
  28. NSString * age = [NSString stringWithFormat:@"%d", 10+i];
  29. BOOL res = [db2 executeUpdate:insertSql2, name, age,@"北京"];
  30. if (!res) {
  31. NSLog(@"error to inster data: %@", name);
  32. } else {
  33. NSLog(@"succ to inster data: %@", name);
  34. }
  35. }];
  36. }
  37. });

iOS学习笔记(十六)——数据库操作(使用FMDB)

时间: 2024-10-26 11:56:28

iOS学习笔记(十六)——数据库操作(使用FMDB)的相关文章

C++学习笔记十六-模板和泛型编程(二)

C++学习笔记十六-模板和泛型编程(二) 16.4 类模板成员 1.模板作用域中模板类型的引用: 通常,当使用类模板的名字的时候,必须指定模板形参.这一规则有个例外:在类本身的作用域内部,可以使用类模板的非限定名.例如,在默认构造函数和复制构造函数的声明中,名字 Queue 是 Queue<Type> 缩写表示.实质上,编译器推断,当我们引用类的名字时,引用的是同一版本.因此,复制构造函数定义其实等价于: Queue<Type>(const Queue<Type> &a

MYSQL进阶学习笔记十六:MySQL 监控!(视频序号:进阶_35)

知识点十六:MySQL监控(35) 一.为什么使用MySQL监控 随着软件后期的不断升级,myssql的服务器数量越来越多,软硬件故障的发生概率也越来越高.这个时候就需要一套监控系统,当主机发生异常时,此时通过监控系统发现和处理. 这个监控实际上是在我们的开发完成之后,这个时候软件就开始在运行,这个运行我们就需要去关注到mysql服务器是否正常,那么我们要观察它就需要给它提供一些监控,这监控就是当它发生故障之后, 那么我们这个监控就会告诉我们到底什么地方发生了一些异常或者一些错误,这个时候我们就

python学习笔记十六 django深入学习一

django 请求流程图 django 路由系统 在django中我们可以通过定义urls,让不同的url路由到不同的处理函数 from . import views urlpatterns = [ url(r'^articles/2003/$', views.special_case_2003), #精确匹配 url(r'^articles/([0-9]{4})/$', views.year_archive), #动态路由 url(r'^articles/([0-9]{4})/([0-9]{2

马哥学习笔记十六——MySQL进阶之事务和隔离级别

连接管理器:  接受请求  创建线程  认证用户  建立安全连接 并发控制: mbox:MDA C/S: 100   10分钟: 多版本并发控制: MVCC 锁: 读锁:共享锁 写锁:独占锁 LOCK TABLES tb_name {READ|WRITE}; UNLOCK TABLES 锁粒度:从大到小,MySQL服务器仅支持表级锁,行锁需要由存储引擎完成: 表锁: 页锁:(block) 行锁: 事务:  RDBMS: ACID (原子性,一致性,隔离性,持久性) MyISAM:  不支持事务

[Python] 学习笔记之MySQL数据库操作

1 Python标准数据库接口DB-API介绍 Python标准数据库接口为 Python DB-API,它为开发人员提供了数据库应用编程接口.Python DB-API支持很多种的数据库,你可以选择跟自己项目相关的数据库.Python DB-API支持的数据库如下所示: GadFly mSQL MySQL PostgreSQL Microsoft SQL Server 2000 Informix Interbase Oracle Sybase 不同的数据库你需要下载不同的DB API模块,例如

MySQL学习笔记一打开数据库操作

打开数据库 1.cmd窗口中输入mysql -uroot -p 2.接着提示输入密码 3.如果MySQL服务器没有打开会报错,这时可以使用以下命令来打开服务器 服务器没有开启错误提示:ERROR 2003 (HY000): Can't connect to mysql server on' localhost' (10061) net start mysql57  启动服务 net stop mysql57  停止服务 在服务设置里面手动打开 4.配置字符格式,在my.ini文本中找到下面两行,

Android学习笔记十六.Android数据存储与IO.SharedPreferences

SharedPreferences 对于应用程序的数据输入.输出,如果是应用程序只是少量数据需要保存,那么使用普通文件就可以了(SharedPrefereces);但如果应用程序有大量数据需要存储.访问,就需要借助数据库了.Android系统内置了SQLite数据库,SQLite数据库是一个真正轻量级的数据库,它没有后台进程,整个数据库就对应于一个文件. 1.SharedPreferences简介 (1)概念:SharedPreferences保存的数据主要是类似于配置信息格式的数据,因此它保存

MySQL学习笔记十六:锁机制

1.数据库锁就是为了保证数据库数据的一致性在一个共享资源被并发访问时使得数据访问顺序化的机制.MySQL数据库的锁机制比较独特,支持不同的存储引擎使用不同的锁机制. 2.MySQL使用了三种类型的锁机制,分别为:表级锁,行级锁,页级锁,它们的特性如下所示. 表级锁:实现逻辑较为简单,加锁速度快,开销小,不会发生死锁:但粒度最大,发生锁冲突的几率最大,并发度最小,适用于以查询为主,极少量更新的系统. 行级锁:加锁慢,开销大,会发生死锁:但粒度最小,锁冲突率小,并发度最高,使用于并发查询大,有大量按

yii2源码学习笔记(十六)

Module类的最后代码 1 /** 2 * Registers sub-modules in the current module. 3 * 注册子模块到当前模块 4 * Each sub-module should be specified as a name-value pair, where 5 * name refers to the ID of the module and value the module or a configuration 6 * array that can

文件的上传Commons FileUpload(web基础学习笔记十六)

一.表单设置 <form action="<%=request.getContextPath()%>/jsp/admin/doAdd.jsp" enctype="multipart/form-data" method="post"> </form> 设置属性: enctype="multipart/form-data"; <tr> <td class="text_