数据库死锁

需求:客户下单,根据客户对应的销售(销售字段userid),进行金额和客户数统计,数据库表内一天一个销售只能有一条记录,第一次下单插入数据,以后每次下单对金额进行加加减减后更新。

数据库表,这里写图片描述
原数据库表居然有这种字段,就是说每次插入或者更新金额数后,会根据金额对销售的排名重新排序后更新
sql:update tabName set * where userid = (select userid from tabName ……)
注:类似的,实际并不是这样,相似都是一个sql有多个操作,且都为同一张表

然后出现了这种情况(敏感处已和谐):

[10-21 16:00:44,410] ERROR [][][Executor-1] c.t.s.web.jms.***[176] - org.springframework.dao.DeadlockLoserDataAccessException: 
### Error updating database.  Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLTransactionRollbackException: Deadlock found when trying to get lock; try restarting transaction
### The error may involve defaultParameterMap
### The error occurred while setting parameters
### SQL: update ***
### Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLTransactionRollbackException: Deadlock found when trying to get lock; try restarting transaction

类似的以下这种也会造成死锁:
insert into xxx (…)
select aaa, bbb from dual where not exists (
select * from xxx where aaa = yyy

总结:
1、在sql语句中,最好不要在一个sql中对数据进行多个操作,这样很容易造成死锁。
why:假设线程A对对表1锁住进行select操作,线程B对表2锁住做insert操作,线程A要访问表2,但是线程B不放锁,线程B要访问表1,线程A也不放锁,这样就造成了死锁。
2、如果业务需求中需要有排序,就个人而言会在取的时候排序取出来,或者取出以后排序,而不是每次更新数据的时候进行排序然后对数据库排序字段进行更新.

转载请注明原址:http://blog.csdn.net/zheng911209/article/details/49585799

你可能感兴趣的:(数据库)