激活码发放,高并发下如何解决可能出现的异常。
现在有个发放激活码的系统,php+mysql。
code表:
id(主键自增长),code(激活码内容,如123abc),status(1代表未被发放,0代表已被发放)。
现在有很多用户(注册用户,能拿到用户信息)去抽这些激活码,每人每天只能抽1次,这一次是肯定能抽到的。
每当用户抽一次激活码,就找到一行status为1的激活码记录,把这行记录的status置为0,同时在record表里添加一行记录(用户id和激活码id),然后返回激活码内容给用户。
逻辑挺简单,现在我要解决的问题是,就是高并发情况下可能会有问题:
比如很多用户同时抽激活码,我select一行status为1的激活码记录时,可能多个用户会select到同一个激活码。有没有什么办法当某个请求select一行记录时,就把这行记录锁住,包证其他请求不会拿到这个激活码。
希望各位大神帮我分析下我这种逻辑在高并发下可能会出现什么问题,有什么解决方案?因为我现在做的项目实际上会遇到高并发的情况,所以必须考虑进去。。。
----------补充----------
谢谢大家的答案, 我会一一参考的。另外可能没说清楚,激活码是提前生成好的,数量和内容都是固定的,提前插入进数据库。
Answers
正常来说
@白菜哥永远都是大白菜 的答案应该满足需求了。
不过我仔细想了下,楼主的需求在查询的时候应该是不知道主键的,sql应该更像是
select * from xxx where status=1 limit 1;这种。
所以高并发的事务select的应该都是同一行。只有等第一个事务完成update后才会全部select到下一行。
所以效果实际上等同表锁?哈,我也不太懂细节,这个问题比较有意思,我决定作为一个实验列在todo list里面。
回到楼主的问题,我更喜欢在代码层面上做个生产消费队列。
这个需要楼主把code表的存储引擎设置为Innodb的,然后搜索下mysql innodb 事务隔离级别。
推荐一篇文章
http://imysql.cn/2008_07_10_innodb_tx_isolation_and_lock_mode
一般不推荐直接在DB上做这种事务隔离处理,一般是在应用中处理的。