• c++ mysql二进制存取,blob存取


     

    取二进制数据:

    一样的sql语句,查询出来即可。只不过二进制数据是个数据块,需要得到数据块的大小和数据指针。

    bool CMySqlAccess::GetBinaryField(int nCol,char* &pDataOut,int& nDataLen)
    {
    if (m_ItemMySqlRow[nCol] != NULL)
    {
       unsigned   long   *FieldLength = mysql_fetch_lengths(m_pMySqlResult);
       nDataLen = (int)FieldLength[nCol];
       pDataOut = (char*)(m_ItemMySqlRow[nCol]);
       return true;
    }
    else
    {
       return false;
    }
    }

    像通常一样查询后,得到结果集,然后得到第nCol列结果,返回二进制指针结果和二进制长度。返回后必须立马处理或者存储一份。否则mysql将数据销毁,指针所指数据则无效了。

    存二进制数据:

    mysql语句接受的sql语句都是string,以'\0'结尾的。如果冒然插入二进制数据到sql语句中,要么报错,要么存储错误。此处可以通过mysql提供的函数将数据转换一下即可。

    char* CMySqlAccess::ConvertBinaryToString(char* pBinaryData,int nLen)
    {
    static char s_BinaryData[10240];
    mysql_real_escape_string(m_pMySqlConn,s_BinaryData,pBinaryData,nLen);
    return s_BinaryData;
    }
    上面这个函数只能单线程使用啊,将一块二进制数据转换为mysql可识别的string数据。这样就直接可以通过mysql的sql语句insert,update来对blob数据进行更新和插入了,sql语句用法不变。

    用例:

    std::ostringstream strSQL;
    strSQL<<"INSERT INTO "<<m_strTableName<<"(roleid,playerdata,dynamicdata) VALUES("<<dwDBRoleID
       <<",'"<<m_pDBAccess->ConvertBinaryToString(pData,nLen)<<"','')";
    assert(m_pDBAccess);
    m_pDBAccess->ExecuteSQL(strSQL.str());

    playerdata是blob二进制类型,pData是指向一个结构体的指针,nLen是结构体的大小。

    上面就可以实现二进制的存储了。

    方法二:

    上面的方法,你会发现,你每次都需要转换数据,传指针,传大小等一系列复杂操作,是不是顺序很混乱,过程很繁杂。mysql也为你提供了另外一种方法,那就是MYSQL_BIND。将数据操作统一化,统一麻烦化。mysqlbind是一个结构体,根据个人不同需求填充各个数据成员可以存储任意类型数据,当然包括blob。

    bool CMySqlAccess::SetBinaryField(std::string& strCondition,void* pDataIn,int nDataLen)
    {
    if( ! mysql_stmt_prepare( m_pMySqlStmt, strCondition.c_str(), strCondition.length() ) )
    {
       memset(&m_MySqlBind,0,sizeof(MYSQL_BIND));
       m_MySqlBind.buffer_type = MYSQL_TYPE_BLOB;
       (*m_MySqlBind.length) = nDataLen;
       memcpy(m_MySqlBind.buffer,pDataIn,nDataLen);
       if(!mysql_stmt_bind_param(m_pMySqlStmt, (MYSQL_BIND *)&m_MySqlBind))
       {
        if(!mysql_stmt_execute(m_pMySqlStmt))
        {
         return true;
        }
       }
    }
    int nRes=GetDBErrorCode();
    CLogOutStream errLog(crazy::ERROR_LEVEL,THIS_CLASS_NAME);
    errLog<<"MySql Query Failed:\""<<strCondition<<"\" ,ErrorCode:"<<nRes<<crazy::logEnd;
    return false;
    }

    这个是对某一列blob数据进行存操作。pDataIn和nDataLen分别是一个struct结构体和结构体大小。填充完毕mysqlbind之后即可对数据库二进制列进行存储了。可能你会问,没有指定哪一列呢,对。哪一列是在strCondition语句里面的,这是一个预处理语句。在预处理语句里面,有一个符号: ? 。问号,问号的位置代表了mysqlbind数据对应的位置。

    INSERT INTO test_table(date_field, time_field, timestamp_field) VALUES(?,?,?)

    上面这个语句,有3个问号,三个问号分别对应test_table的三列.每个问号呢又对应一个mysqlbind数据结构。那么我们在mysql_stmt_bind_param函数调用时,就应该传入一个mysql_bind 数组。MYSQL_BIND    m_MySqlBind[3].

    填充整个数组数据,即对应三个问号内容。

    用例:
    MYSQL_BIND bind[3];
    MYSQL_STMT *stmt;

    strmov(query, "INSERT INTO test_table(date_field, time_field, timestamp_field) VALUES(?,?,?");
    //初始化stmt
    stmt = mysql_stmt_init(mysql);
    //预处理语句
    mysql_stmt_prepare(mysql, query, strlen(query));
    //初始化参数
    bind[0].buffer_type= MYSQL_TYPE_DATE;
    bind[0].buffer= (char *)&ts;
    bind[0].is_null= 0;
    bind[0].length= 0;
    bind[1]= bind[2]= bind[0];
    //绑定参数123
    mysql_stmt_bind_param(stmt, bind);
    //执行预处理mysql语句
    mysql_stmt_execute(stmt);

    还没看懂就个人去看mysql文档了,其实里面讲得很清楚,只要找对几个函数,就可以把search出来了


  • 相关阅读:
    VMware+Centos 7如何配置NAT模式上网
    win 8.1 Your PC needs to be repaired修复过程
    oracle 11g 如何创建、修改、删除list-list组合分区
    关于博客园整理心得
    asp.net使用WebBrowser采集加载完毕后的页面
    解决webconfig中禁用掉ViewState造成服务器控件回传获取不到值问题
    asp.net 路由映射到ashx
    2019.3.17
    三星860 evo 250g 开启AHCI模式读写对比
    解决win10开机出现recovery there was a problem with a device connected to your pc
  • 原文地址:https://www.cnblogs.com/byfei/p/3112323.html
Copyright © 2020-2023  润新知