mysql C api 多线程多连接 客户端实现

mysql客户端库总是线程安全的。

对于除mysql_real_connect()外的所有函数,在默认情况下其他MySQL提供给我们的C API函数都是线程安全的。

要想使mysql_real_connect()成为线程安全的,必须用下述命令再次编译客户端库:
shell> ./configure –enable-thread-safe-client
它创建了线程安全客户端库libmysqlclient_r。

要想使这些操作平稳工作,需要采取下述措施:

  1. 如果程序在调用mysql_real_connect()之前需要调用任何其他MySQL函数,请在启动程序时调用my_init()(my_init()会在mysql_init中调用)。

  2. 调用任何MySQL函数之前,在线程处理程序中调用mysql_thread_init()(mysql_thread_init()会在mysql_init中调用)。

  3. 在线程中,调用pthread_exit()之前请调用mysql_thread_end()。这样,就能释放MySQL线程类变量使用的内存。

将客户端链接到libmysqlclient_r时,如果存在未定义的符号,可能会出错(内存泄漏)。在大多数情况下,其原因在于,未将线程库包含在link/compile行上。

测试代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <pthread.h>

#include <mysql/mysql.h>


void* work(void* arg)
{
        MYSQL *m_pDBAccess;

        m_pDBAccess =mysql_init(NULL);
        if(!m_pDBAccess){
            printf("get mysql init is NULL\n");
            return -1;
        }

        if(!mysql_real_connect(m_pDBAccess, "192.168.1.203", "root", "123456", "tposdb", 0, NULL, CLIENT_MULTI_STATEMENTS)){
            printf("get mysql_real_connect is NULL\n");
        mysql_close(m_pDBAccess);
        m_pDBAccess = NULL ;
        //在多线程环境下把下边语句注释掉使用
    //  mysql_thread_end();
            return NULL;
        }

        mysql_autocommit(m_pDBAccess, 0);

        char value = 1; 
        mysql_options(m_pDBAccess, MYSQL_OPT_RECONNECT, (char*)&value); 



        if (mysql_query(m_pDBAccess, "CALL GetBankSerialNo('35220001', '310111173110010');")) // 如果失败
        {
            mysql_rollback(m_pDBAccess);
            printf("aa\n");
        mysql_close(m_pDBAccess);
        m_pDBAccess = NULL ;
        //在多线程环境下把下边语句注释掉使用
        //mysql_thread_end();
            return NULL;
        }
    //  mysql_rollback(m_pDBAccess);
    mysql_query(m_pDBAccess,"commit");


        mysql_close(m_pDBAccess);
        m_pDBAccess = NULL ;
        //在多线程环境下把下边语句注释掉使用
        //mysql_thread_end();

    return NULL;

}



int main(int rgvs, char **rgva)
{


    pthread_t       tid; 
    pthread_attr_t  attr; 
    pthread_attr_init(&attr); 
    pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED); 




    int i=0;
    for(i=0;i<100;i++)
    {
        pthread_create(&tid, &attr, work, NULL);

    }
    sleep(5);
    return 0;

}

编译命令:

gcc test.c -o test -L/usr/lib64/mysql -lmysqlclient_r -lpthread

测试命令:

valgrind --tool=memcheck --leak-check=full ./test

对比有mysql_thread_end();和无mysql_thread_end();的结果

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Brickie-liu

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值