Mongo中如何使用killOp干掉LongRunningOperation

这篇文章给大家分享的是有关Mongo中如何使用killOp干掉Long Running Operation的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。

创新互联建站-专业网站定制、快速模板网站建设、高性价比朔城网站开发、企业建站全套包干低至880元,成熟完善的模板库,直接使用。一站式朔城网站制作公司更省心,省钱,快速模板网站建设找我们,业务覆盖朔城地区。费用合理售后完善,十多年实体公司更值得信赖。

详解:

  1. 详解currentOp()

  2. gechongrepl:PRIMARY> db.currentOp()

  3. {

  4.     "inprog" : [

  5.         {

  6.             "opid" : 6222,

  7.             "active" : true,

  8.             "secs_running" : 3,

  9.             "microsecs_running" : NumberLong(3662328),

  10.             "op" : "getmore",

  11.             "ns" : "local.oplog.rs",

  12.             "query" : {

  13.                  

  14.             },

  15.             "client" : "192.168.91.132:45745",

  16.             "desc" : "conn5",

  17.             "threadId" : "0x7f1370cb4700",

  18.             "connectionId" : 5,

  19.             "waitingForLock" : false,

  20.             "numYields" : 0,

  21.             "lockStats" : {

  22.                 "timeLockedMicros" : {

  23.                     "r" : NumberLong(141),

  24.                     "w" : NumberLong(0)

  25.                 },

  26.                 "timeAcquiringMicros" : {

  27.                     "r" : NumberLong(16),

  28.                     "w" : NumberLong(0)

  29.                 }

  30.             }

  31.         }

  32.     ]

  33. }

  34.  

  35. "opid" : 6222,#进程号

  36. "active" : true,#是否活动状态

  37. "secs_running" : 3,#操作运行了多少秒

  38. "microsecs_running" : NumberLong(3662328),

  39. "op" : "getmore",#操作类型,包括(insert/query/update/remove/getmore/command)

  40. "ns" : "local.oplog.rs",#命名空间

  41. "query" : {},#如果op是查询操作,这里将显示查询内容;也有说这里显示具体的操作语句的

  42. "client" : "192.168.91.132:45745",#连接的客户端信息

  43. "desc" : "conn5",#数据库的连接信息

  44. "threadId" : "0x7f1370cb4700",#线程ID

  45. "connectionId" : 5,#数据库的连接ID

  46. "waitingForLock" : false,#是否等待获取锁

  47. "numYields" : 0,

  48. "lockStats" : {

  49. "timeLockedMicros" : {#持有的锁时间微秒

  50. "r" : NumberLong(141),#整个MongoDB实例的全局读锁

  51. "w" : NumberLong(0)},#整个MongoDB实例的全局写锁

  52. "timeAcquiringMicros" : {#为了获得锁,等待的微秒时间

  53. "r" : NumberLong(16),#整个MongoDB实例的全局读锁

  54. "w" : NumberLong(0)}#整个MongoDB实例的全局写锁



MongoDB提供了killOp请求,用于干掉运行时间很长的请求,killOp通常需要与currentOp组合起来使用;先根据currentOp查询到请求的opid,然后根据opid发送killOp的请求。

currentOp

currentOp的使用,参考官方文档

currentOp会将后端Mongod上正在执行的请求都列出来,也可根据查询条件(如请求类型,请求是否正在等待锁,请求操作的DB或collection)来进行过滤。

例1:查询所有正在等待锁的写操作

db.currentOp(
   {
     "waitingForLock" : true,
     $or: [
        { "op" : { "$in" : [ "insert", "update", "remove" ] } },
        { "query.findandmodify": { $exists: true } }
    ]
   }
)

例2:查询所有操作db1并且执行时间已超过3s的请求

db.currentOp(
   {
     "active" : true,
     "secs_running" : { "$gt" : 3 },
     "ns" : /^db1\./
   }
)

currentOp的过滤条件包括

  1. 请求操作类型,insert、update、delete…

  2. 请求对应的connectionId,threadId

  3. 请求是否正在等待锁

  4. 请求执行时间

  5. 请求操作的DB或collection

  6. 请求query的内容

killOp

currentOp的输出结果里,每个请求包含一个opid字段,有了opid,就可以发送killOp来干掉对应的请求。

db.killOp(opid)

要了解killOp的意义,需要先搞清楚几个问题

客户端到Monogd Server连接断掉后,连接上执行的请求是否会立即结束?

比如你通过mongo shell,发送了一个createIndex的请求,给某个包含1000w个文档的集合建立索引,这个请求会耗时很久,你想提前中止请求,Ctrl-C停掉了mongo shell,此时mongo shell到server的连接会关闭掉。

但后端createIndex的请求(MongoDB每个连接的请求由一个对应的线程来处理)不会立即结束,而是会一直执行下去,直到createIndex结束,给客户端发送应答时,发现连接已经关闭,然后线程才退出。

为了让createIndex早点结束,你就需要killOp来帮忙,通过currentOp找到craeteIndex请求的opid,然后发送killOp,createIndex会在下个『检查点』就结束执行,整个线程退出。

发送killOp后,请求是否会立即结束?

killOp的实现原理如下

每个连接对应的服务线程存储了一个killPending的字段,当发送killOp时,会将该字段置1;请求在执行过程中,可以通过不断的调用OperationContext::checkForInterrupt()来检查killPending是否被设置,如果被设置,则线程退出。



一个请求要支持killOp,必须在请求的处理逻辑里加上checkForInterrupt()检查点才行,否则即使发送了killOp,也只能等待请求完全处理完毕线程才会退出。

比如createIndex的处理逻辑里包含了类似如下的代码,在createIndex的循环过程中,一旦killPending被置1了,createIndex的执行可以在当前循环结束时退出。

while (!createIndexFinished) {
    createIndexForOneElement();
    checkForInterupt();
}

所以发送killOp后,请求要执行到下一个『检查点』线程才会退出,MongoDB在很多可能耗时长的请求中,都加入了checkForInterrupt()检查点,如创建索引,repair database,mapreduce、aggregation等。

感谢各位的阅读!关于“Mongo中如何使用killOp干掉Long Running Operation”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!


分享标题:Mongo中如何使用killOp干掉LongRunningOperation
链接分享:http://pcwzsj.com/article/ihoocs.html