Workerman/MySQL

说明

常驻内存的程序在使用mysql时经常会遇到mysql gone away的错误,这个是由于程序与mysql的连接长时间没有通讯,连接被mysql服务端踢掉导致。本数据库类可以解决这个问题,当发生mysql gone away错误时,会自动重试一次。

依赖的扩展

该mysql类依赖pdopdo_mysql两个扩展,缺少扩展会报Undefined class constant 'MYSQL_ATTR_INIT_COMMAND' in ....错误。

命令行运行php -m会列出所有php cli已安装的扩展,如果没有pdo 或者 pdo_mysql,请自行安装。

centos系统

  1. yum install php-pdo
  2. yum install php-mysql

如果找不到包名,请尝试用yum search php mysql查找

ubuntu/debian系统

PHP5.x

  1. apt-get install php5-mysql

PHP7.x

  1. apt-get install php7.0-mysql

如果找不到包名,请尝试用apt-cache search php mysql查找

以上方法无法安装?

如果以上方法无法安装,请参考workerman手册-附录-扩展安装-方法三源码编译安装

安装 Workerman/MySQL

方法1:

可以通过composer安装,命令行运行以下命令(composer源在国外,安装过程可能会非常慢)。

  1. composer require workerman/mysql

上面命令成功后会生成vendor目录,然后在项目中引入vendor下的autoload.php。

  1. require_once __DIR__ . '/vendor/autoload.php';

方法2:

下载源码,解压后的目录放到自己项目中(位置任意),直接require源文件。

  1. require_once '/your/path/of/mysql-master/src/Connection.php';

注意

强烈建议在onWorkerStart回调中初始化数据库连接,避免在Worker::runAll();运行前就初始化连接,在Worker::runAll();运行前初始化的连接属于主进程,子进程会继承这个连接,主进程和子进程共用相同的数据库连接会导致的错误。

示例

  1. use Workerman\Worker;
  2. require_once __DIR__ . '/Workerman/Autoloader.php';
  3. require_once __DIR__ . '/vendor/autoload.php';
  4. $worker = new Worker('websocket://0.0.0.0:8484');
  5. $worker->onWorkerStart = function($worker)
  6. {
  7. // 将db实例存储在全局变量中(也可以存储在某类的静态成员中)
  8. global $db;
  9. $db = new Workerman\MySQL\Connection('host', 'port', 'user', 'password', 'db_name');
  10. };
  11. $worker->onMessage = function($connection, $data)
  12. {
  13. // 通过全局变量获得db实例
  14. global $db;
  15. // 执行SQL
  16. $all_tables = $db->query('show tables');
  17. $connection->send(json_encode($all_tables));
  18. };
  19. // 运行worker
  20. Worker::runAll();

具体MySQL/Connection用法

  1. // 初始化db连接
  2. $db = new Workerman\MySQL\Connection('host', 'port', 'user', 'password', 'db_name');
  3. // 获取所有数据
  4. $db->select('ID,Sex')->from('Persons')->where('sex= :sex')->bindValues(array('sex'=>'M'))->query();
  5. //等价于
  6. $db->select('ID,Sex')->from('Persons')->where("sex= 'F' ")->query();
  7. //等价于
  8. $db->query("SELECT ID,Sex FROM `Persons` WHERE sex='M'");
  9. // 获取一行数据
  10. $db->select('ID,Sex')->from('Persons')->where('sex= :sex')->bindValues(array('sex'=>'M'))->row();
  11. //等价于
  12. $db->select('ID,Sex')->from('Persons')->where("sex= 'F' ")->row();
  13. //等价于
  14. $db->row("SELECT ID,Sex FROM `Persons` WHERE sex='M'");
  15. // 获取一列数据
  16. $db->select('ID')->from('Persons')->where('sex= :sex')->bindValues(array('sex'=>'M'))->column();
  17. //等价于
  18. $db->select('ID')->from('Persons')->where("sex= 'F' ")->column();
  19. //等价于
  20. $db->column("SELECT `ID` FROM `Persons` WHERE sex='M'");
  21. // 获取单个值
  22. $db->select('ID,Sex')->from('Persons')->where('sex= :sex')->bindValues(array('sex'=>'M'))->single();
  23. //等价于
  24. $db->select('ID,Sex')->from('Persons')->where("sex= 'F' ")->single();
  25. //等价于
  26. $db->single("SELECT ID,Sex FROM `Persons` WHERE sex='M'");
  27. // 复杂查询
  28. $db->select('*')->from('table1')->innerJoin('table2','table1.uid = table2.uid')->where('age > :age')->groupBy(array('aid'))->having('foo="foo"')->orderByASC/*orderByDESC*/(array('did'))
  29. ->limit(10)->offset(20)->bindValues(array('age' => 13));
  30. // 等价于
  31. $db->query(SELECT * FROM `table1` INNER JOIN `table2` ON `table1`.`uid` = `table2`.`uid`
  32. WHERE age > 13 GROUP BY aid HAVING foo="foo" ORDER BY did LIMIT 10 OFFSET 20“);
  33. // 插入
  34. $insert_id = $db->insert('Persons')->cols(array(
  35. 'Firstname'=>'abc',
  36. 'Lastname'=>'efg',
  37. 'Sex'=>'M',
  38. 'Age'=>13))->query();
  39. 等价于
  40. $insert_id = $db->query("INSERT INTO `Persons` ( `Firstname`,`Lastname`,`Sex`,`Age`)
  41. VALUES ( 'abc', 'efg', 'M', 13)");
  42. // 更新
  43. $row_count = $db->update('Persons')->cols(array('sex'))->where('ID=1')
  44. ->bindValue('sex', 'F')->query();
  45. // 等价于
  46. $row_count = $db->update('Persons')->cols(array('sex'=>'F'))->where('ID=1')->query();
  47. // 等价于
  48. $row_count = $db->query("UPDATE `Persons` SET `sex` = 'F' WHERE ID=1");
  49. // 删除
  50. $row_count = $db->delete('Persons')->where('ID=9')->query();
  51. // 等价于
  52. $row_count = $db->query("DELETE FROM `Persons` WHERE ID=9");
  53. // 事务
  54. $db->beginTrans();
  55. ....
  56. $db->commitTrans(); // or $db->rollBackTrans();