使用MYSQLCPPCONN连接MYSQL数据库与读写BLOB字段

起初也是用C语言接口, 但不知为什么在VS2010下使用时调用mysql_thread_init()函数总是失败(动态链接)

其他函数也用不了, 用了一阵子的ODBC但不会操作BLOB字段

遂转到MYSQLCPPCONN, 但网上大都是C API例子,找了一些外文才整理好.

闲话少说,走起.

准备工作

去官网上下载最新的Connector/C++的压缩包, 包含了头文件, 静态连接的lib, 和动态连接的lib与dll

但现在是无法直接使用的, 还需要boost库

其他资料说需要编译boost库, 其实不用, 提取出来需要的头文件即可, 仅用了智能指针的部分

这里建议使用动态链接的方式, 因为静态库是VC6的MT运行库, 与现在的编译器不兼容, 所以还是带着一个DLL吧

连接数据库

1.取得driver对象,这是个全局静态对象

m_driver = sql::mysql::get_mysql_driver_instance();

2.初始化线程环境.

m_driver->threadInit();

3.连接数据库

conn = m_driver->connect(m_host.c_str(), m_user.c_str(), m_pwd.c_str());

4.初始化连接选项(自动重连, 字符编码)

bool v;

v = true;
conn->setClientOption("OPT_RECONNECT", &v);
if (sql::Statement *stmt = conn->createStatement())
{
	stmt->execute("set charset \'utf8\';");
	delete stmt;
}
上述代码中自动重连是无效的, 需要解决方法, 点击 这里.
5.后面就可以增删改查了,略...

6.使用完毕后销毁对象

conn->close();
delete conn;

7.若不再使用MYSQL或线程即将退出,清理线程环境

m_driver->threadEnd();

备注:

1.数据库连接参数中的host参数包含了连接方式, 地址, 端口, 数据库 信息, 像这样

tcp://localhost:3306/database

2.接口中使用的字符串都是SQLString, 虽然是std::string的进一步封装, 但万万不可传递一个std::string对象作为参数

因为是动态链接的方式, DLL中的内存管理和EXE中没有联系,传递对象过去会造成异常:bad_alloc

所以建议直接使用char*, 至于自己构造SQLString传递, 我测试的时候还是会异常, SO, 用基本数据类型吧

3.此类接口,如果出现错误, 是不会返回NULL的, 会直接抛异常, 记得捕获它并处理

try
{
}
catch (sql::SQLException e)
{
}

至此, 数据库的连接和断开已经搞定.

读写BLOB字段

1.读BLOB

读比较简单, sql::ResultSet直接提供了getBlob方法, 返回istream对象, 读数据即可

2.写BLOB

sql::PreparedStatement提供了setBlob接口, 参数是istream指针, 但无法直接构造, 需要自扩展一下

这里我是用的std::string存储字节数组, 有需要再自行修改

class BlobBuilder : public std::streambuf
{
public:
	BlobBuilder(const std::string &buffer)
	{
		char *d;

		d = (char*)buffer.data();
		setg(d, d, d + buffer.size());
	}
};

用一个存储过程作为例子, 不定的数据用问号表示, UPDATE等语句一样适用

unique_ptr<sql::PreparedStatement> PreStmt(conn->prepareStatement("call func_xxx(?);"));
BlobBuilder BlobBuffer(xxx);
istream Blob(&BlobBuffer);
PreStmt->setBlob(1, &Blob);//第几个参数, 从1开始计算
PreStmt->execute();
unique_ptr<sql::ResultSet> ResSet(PreStmt->getResultSet());// 拿到结果集后就随意操作了

其他问题

若一次执行多条语句(分号分割), 在没有清空结果集之前, 是无法执行下一条查询语句的, 会出现如下错误

Commands out of sync; you can't run this command now

也就是C API中的CR_COMMANDS_OUT_OF_SYNC错误

这个时候就要在每次查询完之后, 释放所有的结果集

while (PreStmt->getMoreResults())
{
	unique_ptr<sql::ResultSet> MoreResSet(PreStmt->getResultSet());
}


你可能感兴趣的:(使用MYSQLCPPCONN连接MYSQL数据库与读写BLOB字段)