RELATEED CONSULTING
相关咨询
选择下列产品马上在线沟通
服务时间:9:30-18:00
你可能遇到了下面的问题
关闭右侧工具栏

技术支持

不要使用SELECT FOR UPDATE
  • 作者:技术小林
  • 发表时间:2018-07-21 15:29
  • 来源:未知

  在SELECT语句中声明游标时,不要使用FORUPDATE从句。适用情形:无论何时都适用。审查关于游标的代码,质疑每个SELECTFORUPDATE语向。使用FORUPDATE从句会锁定行,从而降低事务处理速度。要点:游标是一种强大的结构,使用得当,能够使编程更快更容易,同时还能加速事务处理。但FORUPDATE游标可能会长期锁定数据,降低事务处理速度。参考数据库文档,看是否需要使用FORREADONIY从句,最小化锁的数量。

  如果使用得当,游标是非常强大的数据库控制结构,利用游标可以遍历和处理游标査询(或操作)定义的结果集中的数据。在要指定一个数据集进行遍历或遍历处理数据集中的行时,游标非常有用。数据集中的数据项可以更新、删除、修改或读取,也可以被其他进程査看。游标的真正强大之处在于可以作为程序设计语言的扩展,因为许多过程和面向对象的程序设计语言都没有提供内置的管理关系数据库的数据集的功能。在高速事务处理系统中,在SEL,ECP游标中使用FORUPDATE从句可能造成问题,甚至造成死锁。

  在许多数据库中,一旦打开了具有FORUPDATE从句的游标,那么该语句查询到的行都会被锁住,直到会话中执行到了提交或回退语句为止。COMMIT语句会保存修改,ROLLBACK语句会取消所有的修改。执行到这两个语句中的任何一个,与数据库中的行相关的锁都会被释放掉。此外,执行了提交或回退语句后,你就会失去在游标中的位置,不能再从游标中提取记录了。

  暂停一下,你能发现游标SELECTORUPDAI8E至少可能会造成两个问题吗?第一个问题是,游标会在执行操作时一直保留数据库行的锁。在许多情况下,这样可能都是有用的,甚至在少数情况下,这样做是不可避免的,或者是最佳方案。但是,在执行某些操作时,这些锁会使其他事务一直阻塞或等待。如果这些操作很复杂或需要花费一些时间,那么就会堆起许多待处理的事务。如果恰好这些事务是游标执行SELECTFORUPDATE操作要执行的,那么我们可能就会创建了一个等待队列,而队列的处理时间是用户不能接受的。在Web环境中,让缺乏耐心的用户等待漫长的响应会造成他们发起更多的请求,而这后来的请求有可能会完成得更快。结果是灾难性的,我们的系统会停机,因为待处理的请求堆在数据库端,最终会造成Web服务器占用了所有TCP端口,从而停止响应用户。

  第二个问题前面提示过,是第一个问题造成的。必须等到之前的锁清除了,后面的游标才能对当前锁住的一行或多行加锁。注意,这些锁可能不是游标加的,它可以是用户加的显式锁,也可以是RDBMS加加的隐式锁。数据库中的锁越多,事务堆积的可能性越大,虽然有些锁是必需的。长期保留的锁会造成对经常请求的数据的响应时间变慢。有些数据库,如Oracle,有选用的天键子NOWA可以把控制权释放

  注意,有些网站制作数据库默认会在游标中使用FORUPDATE从句。事实上,ANSI的SQL标准指示,任何游标都要默认使用FORUPDATE从句,除非它在DECLARE语句中使用了FORREADONLY从句。开发人员和DBA应该参考他们的数据库文档,看看如何开发锁最少的游标。