Chinaunix首页 | 论坛 | 博客
  • 博客访问: 29956411
  • 博文数量: 2065
  • 博客积分: 10377
  • 博客等级: 上将
  • 技术积分: 21525
  • 用 户 组: 普通用户
  • 注册时间: 2008-11-04 17:50
文章分类

全部博文(2065)

文章存档

2012年(2)

2011年(19)

2010年(1160)

2009年(969)

2008年(153)

分类: Mysql/postgreSQL

2010-02-24 13:03:10

今天有大收获。最近开发的一个系统,后台用的线程池来处理请求,而每个线程池持有一个mysql连接。这个程序有时候会莫名其妙的死掉,好像是在操作数据 库的时候。由于出现的概率比较小,所以不大好捕捉bug。今天和一个同事聊起这事的时候,才知道他也遇到过这个问题,而这个问题的原因是由于数据库连接由 于长时间没有操作而会被自动关闭。哈哈,真是踏破铁鞋无觅处,得来全不费工夫。解决方法就是在用mysql_ping来自动检查重连。
首先,如果使用了长连接而长期没有对数据库进行任何操作,那么在timeout值后,mysql server就会关闭此连接,而客户端在执行查询的时候就会得到一个类似于“MySQL server has gone away“这样的错误。

   一个好的解决方法是使用mysql_ping。在使用mysql_real_connect连接数据库之后,再使用mysql_options( &mysql, MYSQL_OPT_RECONNECT, … ) 来设置为自动重连。这样当mysql连接丢失的时候,使用mysql_ping能够自动重连数据库。如果是在mysql 5.1.6之前,那么则应在每次执行完real_connect 之后执行mysql_options( &mysql, MYSQL_OPT_RECONNECT, … ) ,如果是mysql 5.1.6+,则在connect之前执行一次就够了。

  有一种很自然的想法就是,新开一个线程,让它隔一定时间(如20秒)就执行一次mysql_ping.除此而外它什么都不用做。但是显然,这个线程必须与其它线程共享一个mysql连接,共享该mysql句柄。否则这样做一点意义都没有。

   但是,mysql_ping会改变mysql_affected_rows的返回值。所以最好是给该MYSQL句柄再加一个mutex(最好是读写 锁)。当其它线程准备执行query的时候,就获取锁,执行完就释放。而这个执行mysql_ping的线程在执行ping之间先尝试获取锁,如果获取失败,则继续sleep,放弃这一轮的ping。
阅读(1879) | 评论(3) | 转发(0) |
给主人留下些什么吧!~~

chinaunix网友2010-05-12 08:36:56

自已写了一个数据库连接池发现了这种情况长久不操作就会出现这种BUG

hkebao2010-03-30 11:58:36

经验之谈:重负荷服务器千万别打开php的mysql 持久连接 因为每个apache进程如果有执行php并且进行sql查询,它就会打开一个 mysql连接,并且使用完毕后不会关闭!我看mysql里面有大量的空闲连接。如果apache进程很多,那就很容易将mysql 的连接用满了。根据别人的经验,php的mysql_pconnect 函数并不会增加性能,为了你的服务器安全,请别使用mysql_pconnect函数。

hkebao2010-03-04 08:21:15

突然想到一个问题,在创建一个新的连接池的时候能否出现这样的连接实例,即不同的连接实例是连接到了不同的数据库的。