MySQL中参数sql_safe_updates在生产环境的使用详解
前言
在应用BUG或者DBA误操作的情况下,会发生对全表进行更新:updatedelete的情况。MySQL提供sql_safe_updates来限制次操作。
setsql_safe_updates=1;
设置之后,会限制updatedelete中不带where条件的SQL执行,较严格。会对已有线上环境带来不利影响。对新系统、应用做严格审核,可以确保不会发生全表更新的问题。
CREATETABLEworking.test01(idINTNOTNULLAUTO_INCREMENT,NAMEVARCHAR(20),ageINT,gmt_createdDATETIME,PRIMARYKEY(id)); insertintotest01(name,age,gmt_created)values('xiaowang',2,now()); insertintotest01(name,age,gmt_created)values('huahua',5,now()); insertintotest01(name,age,gmt_created)values('gougou',9,now()); insertintotest01(name,age,gmt_created)values('heihei',12,now()); insertintotest01(name,age,gmt_created)values('baibai',134,now()); #过滤字段上没有索引 update updatetest01setname='xiaoxiao'whereage=2; ERROR1175(HY000):YouareusingsafeupdatemodeandyoutriedtoupdateatablewithoutaWHEREthatusesaKEYcolumn #全表更新 updatetest01setname='xiaoxiao'; ERROR1175(HY000):YouareusingsafeupdatemodeandyoutriedtoupdateatablewithoutaWHEREthatusesaKEYcolumn #加入limit的更新 updatetest01setname='xia'limit1; QueryOK,1rowaffected(0.00sec) Rowsmatched:1Changed:1Warnings:0 #新增索引 createindexidx_ageontest01(age); updatetest01setname='xiaoxiao'whereage=2; QueryOK,1rowaffected(0.01sec) Rowsmatched:1Changed:1Warnings:0 updatetest01setname='hhh'whereage=9limit10; QueryOK,1rowaffected(0.00sec) Rowsmatched:1Changed:1Warnings:0 altertabletest01dropindexidx_age; createindexidx_age_nameontest01(age,name); updatetest01setage=100wherename='hhh'; ERROR1175(HY000):YouareusingsafeupdatemodeandyoutriedtoupdateatablewithoutaWHEREthatusesaKEYcolumn updatetest01setage=100wherename='hhh'limit10; QueryOK,1rowaffected(0.00sec) Rowsmatched:1Changed:1Warnings:0
由此,update时,在没有where条件或者where后不是索引字段时,必须使用limit;在有where条件时,为索引字段
最近在工作中又发现了一个问题,mysqlsql_safe_updates不支持子查询的更新。
考虑到开发人员有时候不小心误更新数据,要求线上库的MySQL实例都设置sql_safe_updates=1来避免没有索引的update、delete。
结果有一天开发发现下面的一个SQL没法正确执行:
updatet1setcol2=1wherekey1in(selectcol2fromt2wherekey2='ABcD');
错误如下:
ERROR1175(HY000):YouareusingsafeupdatemodeandyoutriedtoupdateatablewithoutaWHEREthatusesaKEYcolumn
也就是说没法对没有走到索引的where条件进行更新。搜索了下发现,的确不行。及时key1和key2分别是t1、t2的索引[我换成主键都不行]。说明是不支持子查询的update。
google了一下发现人家也问过这个问题。。
http://stackoverflow.com/questions/24314830/query-not-getting-executed-if-supplied-a-nested-sub-query
最后解决方法:
1)修改session级别的参数:setsql_safe_updates=0;执行update操作。退出终端。
2)程序处理:先selectcol2fromt2wherekey2='ABcD'获取数据,然后循环处理结果,并用updatet1setcol2=1wherekey1=?来批量更新过。建议还是用程序处理,临时修改变量不是长久之计。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对毛票票的支持。