起初也是用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<:preparedstatement> PreStmt(conn->prepareStatement("call func_xxx(?);"));
BlobBuilder BlobBuffer(xxx);
istream Blob(&BlobBuffer);
PreStmt->setBlob(1, &Blob);//第几个参数, 从1开始计算
PreStmt->execute();
unique_ptr<: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<:resultset> MoreResSet(PreStmt->getResultSet());
}