数据库异步写入功能概要设计

在新项目中,我们要求每一个写入数据库的操作不允许直接写入到MYSQL数据库中,防止在大并发情况下数据库的瓶颈出现,解决的思路是凡是有需要更新、插入数据库的需求,必须按规定封装成JSON数据,然后存放到REDIS的队列中,由数据库写入队列监控程序进行监视,发现后串行写入到数据库中。如果JAVA项目有在写入数据库的同时修改缓存的需要,那么自行修改缓存,数据库写入程序不负责修改缓存数据。

 

下面以一个有代表性的写入动作为例,进行说明:

学生表

CREATE TABLE `t_base_student` (
  `ID` int(11) NOT NULL AUTO_INCREMENT COMMENT '学生表主键,自增长',
  `STUDENT_GUID` char(36) DEFAULT NULL COMMENT '学生的GUID编号',
  `STUDENT_NAME` varchar(20) DEFAULT NULL COMMENT '学生姓名',
  `XB` char(2) DEFAULT NULL COMMENT '性别',
  `BYXX` varchar(50) DEFAULT NULL COMMENT '毕业学校',
  PRIMARY KEY (`ID`)
) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8;

插入一条测试数据

INSERT INTO `t_base_student` VALUES ('1', 'bd81eb89-b933-11e3-8d96-7427eab28b71', '黄海', '', '东北师范大学');

 

家长表

CREATE TABLE `t_base_parent` (
  `PARENT_ID` int(11) NOT NULL AUTO_INCREMENT COMMENT '家长的编号',
  `PARENT_GUID` char(36) DEFAULT NULL COMMENT '家长的GUID号',
  `PARENT_NAME` varchar(255) DEFAULT NULL COMMENT '家长的姓名',
  `STUDENT_ID` int(11) DEFAULT NULL COMMENT '学生号',
  `STUDENT_GUID` char(36) DEFAULT NULL COMMENT '学生GUID号',
  `PARENT_TYPE` int(11) DEFAULT NULL COMMENT '家长的类型',
  PRIMARY KEY (`PARENT_ID`)
) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8;

 插入两条测试数据

INSERT INTO `t_base_parent` VALUES ('1', 'f1af5347-b933-11e3-8d96-7427eab28b71', '黄章', '1', 'bd81eb89-b933-11e3-8d96-7427eab28b71', '1');
INSERT INTO `t_base_parent` VALUES ('2', '0f517382-b934-11e3-8d96-7427eab28b71', '王云', '1', 'bd81eb89-b933-11e3-8d96-7427eab28b71', '2');

 最终确定使用存储过程来解决这个问题:

 

-- 删除存储过程
drop PROCEDURE sp_addstudent;

-- 清空测试数据表
TRUNCATE TABLE a;
TRUNCATE TABLE t_base_parent;
TRUNCATE TABLE t_base_student;

-- 'bd81eb89-b933-11e3-8d96-7427eab28b71','黄海','男','东北师范大学'

CREATE PROCEDURE sp_addstudent (IN V_STUDENT_GUID CHAR (36),IN V_STUDENT_NAME VARCHAR (20),V_XB CHAR (2),V_BYXX VARCHAR (50))
BEGIN

DECLARE v_student_id INT;

START TRANSACTION;
INSERT INTO t_base_student (STUDENT_GUID,STUDENT_NAME,XB,BYXX)VALUES(V_STUDENT_GUID,V_STUDENT_NAME,V_XB,V_BYXX);
SELECT LAST_INSERT_ID() INTO v_student_id;
INSERT INTO a VALUES(v_student_id);
COMMIT;

END;

 

通过lpush命令写入到REDIS缓存中,串行写入数据库系统通过rrange命令读取,执行写入到MYSQL数据库。并执行rpop弹出这个指令。

后期可以考虑使用更加安全的队列形态,比如rpoplpush,可以参考:http://www.cnblogs.com/youxin/p/3585867.html

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 PHP 中实现异步写入数据库可以使用以下两种方法: 1. 使用 Swoole 扩展 Swoole 是一个高性能的异步网络通信框架,可以用来实现异步写入数据库。以下是一个简单的示例代码: ```php $server = new swoole_server("127.0.0.1", 9501, SWOOLE_PROCESS, SWOOLE_SOCK_TCP); $server->on('receive', function ($serv, $fd, $from_id, $data) { $serv->send($fd, "hello\n"); $serv->after(1000, function () use ($fd) { $dbh = new PDO('mysql:host=localhost;dbname=test', 'root', 'password'); $dbh->exec("INSERT INTO `test` (`name`) VALUES ('test')"); $dbh = null; $serv->send($fd, "world\n"); }); }); $server->start(); ``` 2. 使用 MySQLi 的异步 API MySQLi 是 PHP 的一个扩展库,它提供了异步 API,可以用来实现异步写入数据库。以下是一个简单的示例代码: ```php $mysqli = new mysqli("127.0.0.1", "root", "password", "test", 3306, null, MYSQLI_ASYNC); $mysqli->query("INSERT INTO `test` (`name`) VALUES ('test')", MYSQLI_ASYNC); $links = [$mysqli]; $wait_for = null; $wait_timeout_sec = 1; $wait_timeout_usec = 0; do { $result = mysqli_poll($links, $wait_for, $wait_timeout_sec, $wait_timeout_usec); if ($result === false) { echo "Error: " . mysqli_error($mysqli); break; } elseif ($result > 0) { $result = $mysqli->reap_async_query(); var_dump($result); break; } } while (true); ``` 需要注意的是,使用 MySQLi 异步 API 需要开启 MySQLi 的异步模式,而且只有在 PHP 编译时开启了 mysqli 模块的异步支持才能使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值