mysql客户端库总是线程安全的。
对于除mysql_real_connect()外的所有函数,在默认情况下其他MySQL提供给我们的C API函数都是线程安全的。
要想使mysql_real_connect()成为线程安全的,必须用下述命令再次编译客户端库:
shell> ./configure –enable-thread-safe-client
它创建了线程安全客户端库libmysqlclient_r。
要想使这些操作平稳工作,需要采取下述措施:
如果程序在调用mysql_real_connect()之前需要调用任何其他MySQL函数,请在启动程序时调用my_init()(my_init()会在mysql_init中调用)。
调用任何MySQL函数之前,在线程处理程序中调用mysql_thread_init()(mysql_thread_init()会在mysql_init中调用)。
在线程中,调用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();的结果