【原创】10. MYSQL++ 之 DbDriver
1. 综述
DbDriver只是对于MYSQL C API的一个非常简单的封装,作者原句是This class does as little as possible to adapt between its public interface and the interface required by the underlying C API. 作为MYSQL++的使用者我们不应该直接去创建这个类型,而应该默认让mysqlpp:: Connection来来创建和管理,如果实在是需要,可以通过mysqlpp:: Connection::driver( )来获取底层的DbDriver类型。
作者想把这个类型封装为MYSQL C API的轻类型的另外一个原因,就是希望it may be turned into an abstract base class, with subclasses for different database server types.
2. 什么时候创建了DbDriver
正如作者所言,DbDriver被创建的时机只有在mysqlpp:: Connection的构造函数中。
3. mysqlpp:: Option及其子类型
在了解DbDriver之前,我们有必要来看一下MYSQL++定义的另外一个表示连接属性的类型mysqlpp:: Option。该类型其实是作为后续子类型的父类型而存在的。
由于这些Option类型只与DbDriver打交道,所以作为pure-OO的做法来说,我们有足够的理由将这些类型定义为DbDriver的protected或者private类型,但是作者在这里把这些类型放到专门的文件中(options.h和options.cpp),这完全只是因为这些类型过于冗长的关系。(顺便说一句,在C#中,这个问题被妥妥地解决了。终极办法是使用partial class,即分部类型。)
说一下这些Option类型的源模型。
https://dev.mysql.com/doc/refman/5.1/en/mysql-options.html
总体来说,其实这些类型就是为了包装链接中所叙述的那么多的OPTION,实现方法基本上也就是使用TEMPLATE+DELEGATE的设计模式,利用纯虚基类指定方法接口,使用DELEGATE调用mysqlpp:: DbDriver:: set_option( )方法(其下其实封装的是mysql_options() 这个C API)。
看一下代码片段就知道了
在这些类型的定义中,可以学到如下几点:
- 不同的属性有不同的设置方法,MYSQL++是如何做到公用和节省的?
- 有一些属性是基于bool型的参数(例如,FoundRowsOption,该Option其实封装的是CLIENT_FOUND_ROWS这个属性,该属性的取值就应该是一个bool型);
- 有些属性是基于整数型的参数(例如ConnectTimeoutOption,该Option其实封装的是MYSQL_OPT_CONNECT_TIMEOUT这个属性,该属性的取值就是一个int型);
- 还有一些属性是基于字符串(std::string)型的参数(例如ReadDefaultFileOption,该属性封装的是MYSQL_READ_DEFAULT_FILE这个底层属性——Read options from the named option file instead of from my.cnf.——他的取值就是一个char*)。对于有那么多不同参数种类的属性,MYSQL++是如何做的?
(为了看起来方便,我将Set方法的实现拉到了.h中了,原来是写在了options.cpp中)
MYSQL++采用模板类型继承已有的Option类型,然后实例化的方式来简单地拓展了3个不同类型的Option。其余的具体的Option则再通过这几个已拓展的类型进行实现,代码简洁漂亮。
- 为了让代码更简明,MYSQL++采用了在mysqlpp:: Option中内置了一个enum的形式,
这样做的好处是什么?简单,看得清楚。让我们来看一下在DbDriver的set_option_impl中的具体使用
4. mysqlpp:: DbDriver属性与方法
- 属性
其中
很显然,核心自然就是那个mysql_,这可是MYSQL C API 的核心!
同时,需要关心的是is_connected_,这个变量其实主要用在
bool connected() const { return is_connected_; }
这个函数的意思也挺明确的,就是 This does not actually check whether the connection is viable, it just indicates whether there was previously a successful connect() call and no disconnect().
- 方法
- 构造与析构
DbDriver其实是非常好的RAII的范本,在DbDriver中会将表示“是否已经call过connect()以及还没有call过disconnect()的is_connected_置为false,同时初始化mysql_。在析构函数中,首先检查的是这个is_connected是否为true(表示已经call 过Connect()并且还没有call过DisConnect()),如果是true则立即进行disconnect。
- 设置OPTIONS
其实主要需要关注的无非就是
1: bool DbDriver:: set_option(Option* o);
问题是为什么是它?我们知道,OPTIONS是连接的属性,所以我去查看了mysqlpp:: Connection的内容,发现他有这个同名方法。
这个方法具体做了什么?
如果已经是连接状态了,那么久直接进行set(那个DbDriver:: set_option_impl( ) 方法其实就是调用了mysqlpp:: Option::set( )方法,具体可以看上面),如果没有连接过,那么久加入到pending_options_中。
接下来的问题是pending_options_什么时候用到?请看下文。
顺带便说一下applied_options_在什么时候用,
一旦没有error出现的时候,就会将这个指针加入到applied_options,为什么要保留这个指针?当然是为了以后销毁。
- Connect与Disconnect
在最为传统的MYSQL C API中,建立连接和断开连接分别是mysql_real_connect和mysql_close系统调用
在MYSQL++中,做法主要是
1: bool
2:
3: DBDriver::connect(const char* host, const char* socket_name,
4:
5: unsigned int port, const char* db, const char* user,
6:
7: const char* password)
8:
和
1: void DBDriver::disconnect()
disconnect没啥好说的,就是直接调用了mysql_close,然后清理了变量方便这个实例后续重用(当然,先检查是否连接过)
对于DbDriver:: connect来说,代码如下
关键的就是这个connect_prepare做了什么?
首先他显示检查是否在“连接”状态,如果是,则断开连接,然后调用初始化的mysql_init函数,随后将之前添加的pending_options_加入到option中。
- 其他琐碎代码
其他的代码基本上就是底层C API的包装,只有escapting和quoting理解起来会有点困难。请查看相关章节。
原创作品,转载请注明出处www.cnblogs.com/aicro。
【原创】10. MYSQL++ 之 DbDriver的更多相关文章
- 【原创】MYSQL++源码剖析——前言与目录
终于完成了! 从第一次想写到现在真的写好大概花了我3个月时间.原来一直读人家的系列文章,总感慨作者的用心良苦和无私奉献,自己在心里总是会觉得有那么些冲动也来写一个. 最开始的麻烦是犹豫该选哪个主题.其 ...
- python进阶10 MySQL补充 编码、别名、视图、数据库修改
python进阶10 MySQL补充 编码.别名.视图.数据库修改 一.编码问题 #MySQL级别编码 #修改位置: /etc/mysql/mysql.conf.d/mysqld.cnf def ...
- 10 mysql选错索引
10 mysql选错索引 在mysql表中可以支持多个索引,有的sql不指定使用哪个索引,由mysql自己来决定,但是有时候mysql选错了索引,导致执行很慢. 例子 CREATE TABLE `t1 ...
- 10 | MySQL为什么有时候会选错索引? 学习记录
<MySQL实战45讲>10 | MySQL为什么有时候会选错索引? 学习记录http://naotu.baidu.com/file/e7c521276650e80fe24584bc9a6 ...
- 【原】无脑操作:Windows 10 + MySQL 5.5 安装使用及免安装使用
本文介绍Windows 10环境下, MySQL 5.5的安装使用及免安装使用 资源下载: MySQL安装文件:http://download.csdn.net/detail/lf19820717/9 ...
- 10.Mysql索引
10.索引的设计和使用10.1 索引概述BTREE索引:Mysql(MyIASM和Innodb)默认的索引类型.前缀索引:对索引字段的前N个字符创建索引.N的最大取值和存储引擎有关,MyIASM支持最 ...
- 【原创】MySQL CPU %sys高的案例分析(二)
后面又做了补充测试,增加了每秒context switch的监控,以及SQL执行时各步骤消耗时间的监控. [测试现象一] 启用1000个并发线程的压测程序,保持压测程序持续运行,保持innodb_sp ...
- 【原创】MySQL Replay线上流量压测工具
一. 背景 去年做过一次mysql trace 重放的测试,由于performance schema本身采集样本的长度等限制,实际回放的成功率比较低. 最近找到一款开源的工具,基于TCPCopy实现了 ...
- MariaDB 10 (MySQL DB) 多主复制并实现读写分离
----本文大纲 简介 资源配置 拓扑图 实现过程 ==================== 一.简介 MMM 即Master-Master Replication Manager for MySQL ...
随机推荐
- 【剑指offer】数组中的逆序对,C++实现
原创博文,转载请注明出处!本题牛客网地址 博客文章索引地址 博客文章中代码的github地址 1.题目 2.思路 3.代码 class Solution { public: int InversePa ...
- 【转】C# Socket编程(2)识别网络主机
[转自:https://www.cnblogs.com/IPrograming/archive/2012/10/11/CSharp_Socket_2.html] 一个客户端想要发起一次通信,先决条件就 ...
- test20181219 连续段的期望
题意 连续段的期望 [问题描述] 小N最近学习了位运算,她发现2个数xor之后数的大小可能变大也可能变小,and之后都不会变大,or之后不会变小.于是她想算出以下的期望值:现在有 N个数排成一排,如果 ...
- UIScrollView的左右滑动和侧滑手势冲突的解决办法
转载自:https://blog.csdn.net/kst_123/article/details/77762811 当ViewController中添加了一个全屏的UIScrollView的时候,U ...
- 解决安装Weblogic domain卡住问题(Primeton BPS)
这两天一直有一个问题困扰我,在suse10+weblogic(920,923,100,103)上安装bpm产品失败.有些版本是创建domain的时候卡在create security informat ...
- Linux yum操作时出现Error: xz compression not available
yum升级PHP版本的时候出现这个问题 由于CentOS6的系统安装了epel-release-latest-7.noarch.rpm 导致在使用yum命令时出现Error: xz compressi ...
- jquery 实现点击颜色切换
<!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" content ...
- EXCEL类型库的添加
1. 创建新的C++工程 创建基于对话框的MFC程序 2. 添加库.添加Excel类库 在工程名上右键,选择“添加”—“类”(或者点击菜单栏的“项目”->“添加类”),选择“TypeLib中的M ...
- thinkphp5 设置.htaccess报input file specified的解决方法
先去检查服务器设置,这个网上方法很多就不说了,如果服务器没问题还是报这个错误的话可能和php版本有关 php5.4和以下版本的.htaccess <IfModule mod_rewrite.c& ...
- DHCP(六)
DHCP报文: DHCP共有八种报文,分别为DHCP Discover.DHCP Offer.DHCP Request.DHCP ACK.DHCP NAK.DHCP Release.DHCP Decl ...