起初也是用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();
m_driver->threadInit();
conn = m_driver->connect(m_host.c_str(), m_user.c_str(), m_pwd.c_str());
bool v; v = true; conn->setClientOption("OPT_RECONNECT", &v); if (sql::Statement *stmt = conn->createStatement()) { stmt->execute("set charset \'utf8\';"); delete stmt; }上述代码中自动重连是无效的, 需要解决方法, 点击 这里.
6.使用完毕后销毁对象
conn->close(); delete conn;
m_driver->threadEnd();
1.数据库连接参数中的host参数包含了连接方式, 地址, 端口, 数据库 信息, 像这样
tcp://localhost:3306/database
因为是动态链接的方式, 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()); } };
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
这个时候就要在每次查询完之后, 释放所有的结果集
while (PreStmt->getMoreResults()) { unique_ptr<sql::ResultSet> MoreResSet(PreStmt->getResultSet()); }