FMDB 框架小结

2024-06-06 12:58
文章标签 框架 小结 fmdb

本文主要是介绍FMDB 框架小结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

FMDB

FMDB 框架是对 SQLite 数据库 C 语言接口的 Objective-C 封装,即对 sqlite3.h 文件中相关接口的封装。

sqlite3 的使用十分简单,主要是下面两个对象、常用函数的使用:

  • sqlite3
  • sqlite3_stmt
  • sqlite3_open()
  • sqlite3_prepare()
  • sqlite3_bind()
  • sqlite3_step()
  • sqlite3_column()
  • sqlite3_finalize()
  • sqlite3_close()
  • sqlite3_exec()

FMDB 框架基本也就涉及这几个函数。

typedef struct sqlite3 sqlite3;

每一个打开的 SQLite 数据库都由一个透明的结构体 sqlite3 来表示,所以可以将 sqlite3 指针看作一个对象。

typedef struct sqlite3_stmt sqlite3_stmt;

sqlite3_stmt 表示一个编译为二进制形式的 SQL 语句,可以进一步执行了。

FMStatement

FMStatement 是对 sqlite3_stmt 的封装,其成员变量如下,其中 _statement 指针就指向了 sqlite3_stmt 结构体。

@interface FMStatement : NSObject {void *_statement;NSString *_query;long _useCount;BOOL _inUse;
}

在框架的具体使用过程中,我们并不需要关心该类的使用。

FMDatabase

FMDatabase 的使用大致遵循下面几个步骤:

  1. 提供一个数据库文件路径,创建一个 FMDatabase 类实例对象。此时,数据库可能并不存在,这里只是初始化了一些属性。

  2. 打开数据库连接,这里可以调用 openopenWithFlags:openWithFlags:vfs: 方法,来实现数据库的连接,此时可能会创建数据库,如果其不存在的话。

  3. 修改数据库,该类中提供了多个方法来对数据库进行更新,方法名以 executeUpdate 字符串开头 ,除了 executeUpdateWithFormat: 方法可以直接使用占位符进行参数的传递外,其他方法都是通过使用 ? 占位符,并同时传递集合参数的方式来构造 SQLite 命令的。但是,最终都是要使用 sqlite3_bind_* 函数进行参数绑定的。

  4. 查询数据库,查询数据库的方法以 executeQuery 字符串开头,并且同修改数据库的方法类似 executeQueryWithFormat: 方法也是可以直接使用各种占位符的,而其他方法都是用 ? 作为占位符的。

  5. 关闭数据库,该类的实例对象在释放时,会自动去关闭当前打开的数据库,当然如有需要,也可以调用 close 方法自行关闭。

FMResultSet

FMResultSet 是使用 FMDatabase 进行数据库查询操作后对查询结果的封装类。这个类十分的简单,实际就是对 sqlite3_step() 函数的封装。

在使用过程中,需要调用 nextnextWithError: 方法获取下一个查询结果,在 nextWithError: 方法中实际就是执行了 sqlite3_step() 函数来获取查询结果集合中的一条记录。而后,根据列名或列索引来获取具体某个字段的值。

实际,在具体的方法实现中,都是通过索引来获取指定字段的值,这是因为最终获取值的 SQLite 接口 sqlite3_column_*() 函数的参数是列的索引值(从 0 开始计算)。

除了通常使用的 stringForColumn: 方法来获取字符串值外,还可以使用 objectForColumn: 方法获取诸如 NSNumberNSDataObjective-C 对象,并且注意如果值为 nil 那么返回的是 NSNull 实例。

另外,该类中还实现了下面两个方法:

- (id _Nullable)objectAtIndexedSubscript:(int)columnIdx;
- (id _Nullable)objectForKeyedSubscript:(NSString *)columnName;

所以在获取具体字段的值时,是支持下标运算的。

该类中还提供了一个方法来支持 KVC 编程,如果你的查询记录的各个字段完全是某个类的属性名称,或者你就是这样设计的,那么使用下面这个函数,将会直接把查询记录中的各个字段的结果赋值给所传递的对象的相应属性中。

- (void)kvcMagic:(id)object;

除了各个取值方法,还有两个属性可以了解一下,如下:

@property (readonly) NSMutableDictionary *columnNameToIndexMap;@property (nonatomic, readonly, nullable) NSDictionary *resultDictionary;
  • columnNameToIndexMap 该属性保存的是查询结果的字段名所对应的索引值,通过字段名来获取值时,就是使用该属性来获取相应字段名的索引,进而获取具体值。
  • resultDictionary 该属性是将当前的记录转换为一个字典,键是各个字段名,值就是对应的具体值。

FMDatabaseQueue

在多线程中直接使用 FMDatabse 对象操作同一个数据库文件是不安全的,所以 FMDB 第三方库提供了 DMDatabaseQueue 来方便在多线程中操作同一个数据库。

在具体的实现中,该类通过创建一个串行队列以及一个 FMDatabase 实例来同步不同线程中对数据库的操作。

@interface FMDatabaseQueue () {dispatch_queue_t    _queue;FMDatabase          *_db;
}
@end

如,在主线程中执行下面的测试函数。

let queue = FMDatabaseQueue.init(path: "")func test() {print("1-  ",Thread.current)DispatchQueue.init(label: "testQueue1").async {print("2-  ",Thread.current)self.queue.inDatabase() {database inprint("2--  ",Thread.current)try! database.executeUpdate("create table if not exists person (studentNo integer primary key,age interger,sex char) ", values: nil)print("2---  ",Thread.current)}print("2----  ",Thread.current)}DispatchQueue.init(label: "testQueue2").async {print("3-  ",Thread.current)self.queue.inDatabase({ (database) inprint("3--  ",Thread.current)database.executeUpdate("insert into person (age,sex) values(?,?)", withArgumentsIn: [4,"男"])print("3---  ",Thread.current)})print("3----  ",Thread.current)}print("1--  ",Thread.current)
}

得到的输出如下:

1-   <NSThread: 0x600003968000>{number = 1, name = main}
1--   <NSThread: 0x600003968000>{number = 1, name = main}
2-   <NSThread: 0x6000039e3000>{number = 5, name = (null)}
3-   <NSThread: 0x60000391d500>{number = 6, name = (null)}
2--   <NSThread: 0x6000039e3000>{number = 5, name = (null)}
2---   <NSThread: 0x6000039e3000>{number = 5, name = (null)}
2----   <NSThread: 0x6000039e3000>{number = 5, name = (null)}
3--   <NSThread: 0x60000391d500>{number = 6, name = (null)}
3---   <NSThread: 0x60000391d500>{number = 6, name = (null)}
3----   <NSThread: 0x60000391d500>{number = 6, name = (null)}

当然,这只是一种结果,虽然输出的结果是不确定的,但是可以发现,2---- 总是在 3-- 之前打印出来。即,每一个数据库操作任务都封装在代码块中,并提交到 FMDatabaseQueue 实例中的串行队列中,而且当前任务会被阻塞,直到数据库操作任务执行完毕。

FMDatabasePool

如果对于一个数据库,只涉及到对其的大量读取,而不涉及其他修改等操作时,可以使用该类。这个数据库池的实现十分简单,主要是通过一个串行队列和两个可变数组实现的。

@interface FMDatabasePool () {dispatch_queue_t    _lockQueue;NSMutableArray      *_databaseInPool;NSMutableArray      *_databaseOutPool;
}

该类中针对同一个数据库文件可以最多创建 maximumNumberOfDatabasesToCreateFMDatabase 实例对象用来操作数据库。应该注意的是,该类使用 _lockQueue 队列来保证线程池的操作是安全的,但是数据库的操作并非是线程安全的。所以,不应该使用该类去对数据库进行非只读的操作。

对于两个数组,_databaseInPool 保存着空闲的 FMDatabase 实例对象,当需要操作数据库时,从该数组中取出实例,而后将其添加到 _databaseOutPool 数组中,因为该数组保存所有正在使用的 FMDatabase 实例对象,使用完毕后再归还到 _databaseInPool 数组中,这个过程是线程安全的。

该类还定义了两个非正式协议方法,用来询问是否向池中新增对象,以及成功新增对象后需要执行什么额外任务。

@interface NSObject (FMDatabasePoolDelegate)- (BOOL)databasePool:(FMDatabasePool*)pool shouldAddDatabaseToPool:(FMDatabase*)database;
- (void)databasePool:(FMDatabasePool*)pool didAddDatabase:(FMDatabase*)database;@end

当然,如果一个 FMDatabase 实例对象已经在池中了,那么便不会再执行 databasePool:didAddDatabase: 方法了,但是你可以通过 databasePool:shouldAddDatabaseToPool: 方法来将其关闭。

这篇关于FMDB 框架小结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1036174

相关文章

cross-plateform 跨平台应用程序-03-如果只选择一个框架,应该选择哪一个?

跨平台系列 cross-plateform 跨平台应用程序-01-概览 cross-plateform 跨平台应用程序-02-有哪些主流技术栈? cross-plateform 跨平台应用程序-03-如果只选择一个框架,应该选择哪一个? cross-plateform 跨平台应用程序-04-React Native 介绍 cross-plateform 跨平台应用程序-05-Flutte

Spring框架5 - 容器的扩展功能 (ApplicationContext)

private static ApplicationContext applicationContext;static {applicationContext = new ClassPathXmlApplicationContext("bean.xml");} BeanFactory的功能扩展类ApplicationContext进行深度的分析。ApplicationConext与 BeanF

数据治理框架-ISO数据治理标准

引言 "数据治理"并不是一个新的概念,国内外有很多组织专注于数据治理理论和实践的研究。目前国际上,主要的数据治理框架有ISO数据治理标准、GDI数据治理框架、DAMA数据治理管理框架等。 ISO数据治理标准 改标准阐述了数据治理的标准、基本原则和数据治理模型,是一套完整的数据治理方法论。 ISO/IEC 38505标准的数据治理方法论的核心内容如下: 数据治理的目标:促进组织高效、合理地

ZooKeeper 中的 Curator 框架解析

Apache ZooKeeper 是一个为分布式应用提供一致性服务的软件。它提供了诸如配置管理、分布式同步、组服务等功能。在使用 ZooKeeper 时,Curator 是一个非常流行的客户端库,它简化了 ZooKeeper 的使用,提供了高级的抽象和丰富的工具。本文将详细介绍 Curator 框架,包括它的设计哲学、核心组件以及如何使用 Curator 来简化 ZooKeeper 的操作。 1

【Kubernetes】K8s 的安全框架和用户认证

K8s 的安全框架和用户认证 1.Kubernetes 的安全框架1.1 认证:Authentication1.2 鉴权:Authorization1.3 准入控制:Admission Control 2.Kubernetes 的用户认证2.1 Kubernetes 的用户认证方式2.2 配置 Kubernetes 集群使用密码认证 Kubernetes 作为一个分布式的虚拟

Spring Framework系统框架

序号表示的是学习顺序 IoC(控制反转)/DI(依赖注入): ioc:思想上是控制反转,spring提供了一个容器,称为IOC容器,用它来充当IOC思想中的外部。 我的理解就是spring把这些对象集中管理,放在容器中,这个容器就叫Ioc这些对象统称为Bean 用对象的时候不用new,直接外部提供(bean) 当外部的对象有关系的时候,IOC给它俩绑好(DI) DI和IO

Sentinel 高可用流量管理框架

Sentinel 是面向分布式服务架构的高可用流量防护组件,主要以流量为切入点,从限流、流量整形、熔断降级、系统负载保护、热点防护等多个维度来帮助开发者保障微服务的稳定性。 Sentinel 具有以下特性: 丰富的应用场景:Sentinel 承接了阿里巴巴近 10 年的双十一大促流量的核心场景,例如秒杀(即突发流量控制在系统容量可以承受的范围)、消息削峰填谷、集群流量控制、实时熔断下游不可用应

分布式系统的个人理解小结

分布式系统:分的微小服务,以小而独立的业务为单位,形成子系统。 然后分布式系统中需要有统一的调用,形成大的聚合服务。 同时,微服务群,需要有交流(通讯,注册中心,同步,异步),有管理(监控,调度)。 对外服务,需要有控制的对外开发,安全网关。

利用Django框架快速构建Web应用:从零到上线

随着互联网的发展,Web应用的需求日益增长,而Django作为一个高级的Python Web框架,以其强大的功能和灵活的架构,成为了众多开发者的选择。本文将指导你如何从零开始使用Django框架构建一个简单的Web应用,并将其部署到线上,让世界看到你的作品。 Django简介 Django是由Adrian Holovaty和Simon Willison于2005年开发的一个开源框架,旨在简

Yii框架relations的使用

通过在 relations() 中声明这些相关对象,我们就可以利用强大的 Relational ActiveRecord (RAR) 功能来访问资讯的相关对象,例如它的作者和评论。不需要自己写复杂的 SQL JOIN 语句。 前提条件 在组织数据库时,需要使用主键与外键约束才能使用ActiveReocrd的关系操作; 场景 申明关系 两张表之间的关系无非三种:一对多;一对一;多对多; 在