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

这篇博客介绍了如何在C++中使用MYSQLCPPCONN库连接MySQL数据库,包括连接初始化、设置选项、增删改查操作,特别是详细讲解了读写BLOB字段的方法,并解决了可能出现的Commands out of sync问题。
摘要由CSDN通过智能技术生成

起初也是用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());

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值