一个简单好用的操作mysql的c++类,在mysql c api的基础上封装的,没有使用官方的c++ connector

目前 mysql 官方的 c++ connector 还没有针对vs2017编译的二进制包,在mysql官方网站上下载的release二进制包在vs2017下可以用,但是下载的debug包在vs2017下运行时却提示找不到vcruntime140_1D.dll 而下载c++ connector源码进行编译,它还要依赖boost,而且编译过程中还出现一些找不到sys/types.h头文件的之类错误,搞的我整个人都不好了。干脆在mysql的c api基础上自己写了一个。

 

特点:header only,基于C++17,依赖fmt库(fmt库也是header only的,而且fmt库是即将进入c++20标准的format库);

 

本人对需要编译的库深恶痛绝,只要能做成header only的就绝不搞成需要编译的,宁可增加编译时间,也不要那种还要看通篇的编译说明,编译参数等等之类的来增加心智负担。

代码地址:https://gitee.com/zhllxt/mysql_modern_cpp

github:https://github.com/zhllxt/mysql_modern_cpp

下面是示例代码,用起来还是相当简单的。

 

struct user
{
	std::string name;
	int age{};
	std::tm birth{};

	template <class Recordset>
	bool orm(Recordset & rs)
	{
		return rs(name, age, birth);
	}
};

mysql::database db;
db.connect("localhost", "root", "123456", "mir3_user");

// R"()" 是 c++ 11 的 raw string 语法,避免字符串换行时还要在行尾添加反斜杠
db << R"(CREATE TABLE `tbl_user` (
	`name` VARCHAR(20) NOT NULL,
	`age` INT NULL DEFAULT NULL,
	`birth` DATETIME NULL DEFAULT NULL,
	PRIMARY KEY(`name`)
	)
	COLLATE = 'gbk_chinese_ci'
	ENGINE = InnoDB
	;)";

// "db << ..." 这种操作符方式会生成一个临时变量 当这个临时变量销毁时会在析构函数中自动执行sql语句
// 注意这种情况下执行sql语句时如果出现错误不会进到示例这里最后面的catch块中
db << "insert into tbl_user (name,age) values (?, ?);"
	<< "admin"
	<< 102;
db << "update tbl_user set age=?,birth=? where name=?;"
	<< nullptr
	<< nullptr
	<< "admin";
db << "update tbl_user set age=?,birth=? where name=?;"
	<< 55
	<< "1990-03-14 15:15:15"
	<< "admin";

user u;
// 查询数据到自定义结构体中
db << "select name,age,birth from tbl_user where name=?" << "admin" >> u;
db.execute("select name,age,birth from tbl_user where name=?", "admin").fetch(u);
db << "select name,age,birth from tbl_user" >> [](user u)
{
	printf("%s %d\n", u.name.data(), u.age);
};

// 自定义结构体的信息添加到数据库中
u.name += std::to_string(std::rand());
db << "insert into tbl_user (name,age,birth) values (?,?,?)" << u;


db << "delete from tbl_user  where name=?;"
	<< "tester";

// 直接调用 db.execute 会直接执行sql语句 如果出现错误可以进到示例这里最后面的catch块中
db.execute("insert into tbl_user values (?, ?, ?);", "tester", 32, "2020-03-14 10:10:10");

std::string name, age, birth;

int count = 0;
db << "select count(*) from tbl_user;"
	>> count;

db << "select name from tbl_user where age=55;"
	>> name;

std::tm tm_birth{}; // 将获取到的日期存储到c++语言的结构体tm中
db << "select birth from tbl_user where name=?;"
	<< name
	>> tm_birth;

const char * inject_name = "admin' or 1=1 or '1=1"; // sql 注入
db << "select count(*) from tbl_user where name=?;"
	<< inject_name
	>> count;

// 将获取的内容存储到绑定数据中 这样你可以直接操作数据的缓冲区buffer
// 但此时必须要用auto rs = 这种方式将recordset临时变量保存起来
// 否则operator>>结束后临时变量就销毁了 binder 指针指向的内容就是非法的了
mysql::binder * binder = nullptr;
auto rs = db << "select birth from tbl_user where name='admin';";
rs >> binder;
MYSQL_TIME * time = (MYSQL_TIME *)(binder->buffer.get());
printf("%d-%d-%d %d:%d:%d\n", time->year, time->month, time->day, time->hour, time->minute, time->second);

// 查询到数据后直接调用 lambda 回调函数,有多少行数据,就会调用多少次
(db << "select name,age,birth from tbl_user where age>?;")
	<< 10
	>> [](std::string_view name, int age, std::string birth)
{
	printf("%s %d %s\n", name.data(), age, birth.data());
};

db << "select age,birth from tbl_user;"
	>> std::tie(age, birth);

db.execute("select name,age,birth from tbl_user where name=?;", name) >>
	[](std::string_view name, int age, std::string birth)
{
};

// set_fields_format 用来设置返回的字符串的格式,注意只有返回字符串时才起作用
// c++ 20 的format语法
// {:>15} 表示右对齐 共占15个字符的宽度
// {:04} 共占4个字符的宽度 如果不足4个字符在前面补0
rs = (db << "select name,age,birth from tbl_user;");
// 总共select了三列数据,所以set_fields_format必须要设置三个格式信息
// 可以调用set_fields_format设置格式,如果调用了就必须有几列,就填几个格式信息
// 也可以不调用set_fields_format,此时会使用默认的格式
rs.set_fields_format("{:>15}", "{:04}", "{:%Y-%m-%d %H:%M:%S}");
auto rs2 = std::move(rs);
// 按照自己的要求一行一行的获取数据
while (rs2.fetch(name, age, binder))
{
	MYSQL_TIME * time = (MYSQL_TIME *)(binder->buffer.get());
	printf("%s %s %d-%d-%d\n", name.data(), age.data(), time->year, time->month, time->day);
}

std::tuple<int, std::string> tup;
db << "select age,birth from tbl_user;"
	>> tup;
auto &[_age, _birth] = tup;
printf("%d %s\n", _age, birth.data());

 

另外再说一下mysql的一些步骤:

1、安装mysql,安装完之后,在mysql的安装目录下会有include和lib目录,里面是mysql的c api的头文件和库

2、由于这个类是mysql的c api的基础上实现的,所以需要在vs工程设置中包含mysql的c api的include目录和lib目录

3、编译后运行会提示找不到libmysql.dll的话,到mysql的安装目录下的lib目录中找到libmysql.dll拷到你生成的exe目录下,可能还会提示找不到libcrypto-1_1-x64.dll和libssl-1_1-x64.dll 到mysql的安装目录下的bin目录中能找到这两个文件,拷到你生成的exe目录下即可

4、如果还提示找不到libeay32.dll和ssleay32.dll 你可以在你的C盘全盘搜索一下,一般能搜到这两个文件,拷到你生成的exe目录下即可,如果搜不到,可以下载HeidiSQL这个工具(HeidiSQL是一个免费的操作mysql和postgresql的工具),安装之后在它的目录下能找到这两个文件。

 

关键词:mysql c++ libeay32 ssleay32 connector

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
是的,可以直接在 Qt 中使用 MySQL Connector/C++ 来连接和操作 MySQL 数据库。MySQL Connector/C++官方提供的一个 C++ 接口,用于在 C++ 程序中与 MySQL 数据库进行交互。要在 Qt 中使用 MySQL Connector/C++,你需要按照以下步骤进行设置: 1. 下载和安装 MySQL Connector/C++:你可以从 MySQL 官网上下载适合你系统的 MySQL Connector/C++ 版本,并按照安装指南进行安装。 2. 在 Qt 项目中添加 MySQL Connector/C++ 的头文件路径和库文件路径:在 Qt 项目的 .pro 文件中添加以下行,将 MySQL Connector/C++ 的头文件路径和库文件路径添加到项目中: ``` INCLUDEPATH += /path/to/mysql-connector-c++/include LIBS += -L/path/to/mysql-connector-c++/lib -lmysqlcppconn ``` 请将 "/path/to/mysql-connector-c++" 替换为你自己的 MySQL Connector/C++ 安装路径。 3. 在 Qt 项目中包含 MySQL Connector/C++ 的头文件:在需要使用 MySQL Connector/C++ 的源文件中,添加如下的 include 语句: ``` #include <mysql_driver.h> #include <mysql_connection.h> ``` 4. 使用 MySQL Connector/C++ 连接和操作 MySQL 数据库:你可以使用 MySQL Connector/C++ 提供的和方法来连接和操作 MySQL 数据库。例如,可以使用 `sql::mysql::MySQL_Driver` 来创建一个 MySQL 连接对象,并使用该对象执行 SQL 查询和更新操作。 注意:在使用 MySQL Connector/C++ 连接数据库之前,确保已经启动了 MySQL 服务器,并且已经创建了相应的数据库和表。 这些是使用 MySQL Connector/C++ 在 Qt 中连接和操作 MySQL 数据库的基本步骤。你可以根据具体的需求进一步深入学习和使用 MySQL Connector/C++ 的功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值