暂无图片
暂无图片
暂无图片
暂无图片
暂无图片

【硬货】Oracle数据库出现问题时,这十个脚本帮你快速定位原因

章芋文 2019-07-13
548

“喂,李总您好!”

“小张,快点看看ERP数据库,应用又打不开了!”

“好的,马上。”

小张从黑色背包拿出电脑,连上手机热点就开始检查,刚连上数据库,电话铃声又响起来了.....

    这样的场景对于Oracle DBA来说太熟悉了,只要应用一出问题,不论何时,不论何地,总是第一个接到电话,严重情况下会是一轮电话轰炸。

    新手和专家之间遇到此类问题,首先是心态,新手遇到问题心里慌,不知从何下手,胆小粗心,专家因为经验丰富,往往沉着冷静、运筹帷幄、抽丝剥茧、胆大心细,但是经验这东西就跟吃过的盐、走过的桥一样,必须亲自多做、多学才能获得。而另外一个非常重要的就是诊断思路和辅助脚本,本文讲述各种场景下的通用处理思路,分享用到的一些脚本,帮助大家快速定位问题并解决,减少业务的中断事件,早日成为专家,升职加薪,迎娶...

>>>>

查看操作系统负载

    登上数据库服务器后,第一个就是通过系统命令确认下CPU、内存、I/O是否异常,每个系统的命令不一样,常见的有top、topas、vmstat、iostat。

>>>>

查看等待事件

    第二步就是连到数据库查看活动的等待事件,这是监控、巡检、诊断数据库最基本的手段,通常81%的问题都可以通过等待事件初步定为原因,它是数据库运行情况最直接的体现,如下脚本是查看每个等待事件的个数、等待时长,并排除了一些常见的IDLE等待事件。

    --墨天轮 wait_event
    col event for a45
    SELECT inst_id,EVENT, SUM(DECODE(WAIT_TIME, 0, 0, 1)) "Prev", SUM(DECODE(WAIT_TIME, 0, 1, 0)) "Curr", COUNT(*) "Tot" , sum(SECONDS_IN_WAIT) SECONDS_IN_WAIT
    FROM GV$SESSION_WAIT
    WHERE event NOT
    IN ('smon timer','pmon timer','rdbms ipc message','SQL*Net message from client','gcs remote message')
    AND event NOT LIKE '%idle%'
    AND event NOT LIKE '%Idle%'
    AND event NOT LIKE '%Streams AQ%'
    GROUP BY inst_id,EVENT
    ORDER BY 1,5 desc

        这里就需要掌握一些常见异常等待事件的原因,并形成条件反射,比如library cache lock、read by other session、row cache lock、buffer busy waits、latch:shared pool、gc buffer busy、cursor: pin S on X、direct path read、log file sync、enq: TX - index contention、PX Deq Credit: send blkd、latch free、enq: TX - row lock contention等等,如果异常等待事件的个数和等待时间很长,那么排查原因的入口就在这里。

    >>>>

    根据等待事件查会话

        得到异常等待事件之后,我们就根据等待事件去查会话详情,也就是查看哪些会话执行哪些SQL在等待,另外还查出来用户名和机器名称,以及是否被阻塞。另外如下脚本可改写成根据用户查会话、根据SQL_ID查会话等等。

      --墨天轮 session_by_event
      SELECT /*+rule */ sid, s.serial#, spid, event, sql_id, seconds_in_wait ws, row_wait_obj# obj,
      s.username, s.machine, BLOCKING_INSTANCE||'.'||blocking_session b_sess 
      FROM v$session s, v$process p 
      WHERE event='&event_name' AND s.paddr = p.addr order by 6;
      >>>>

      查询某个会话详情

          得到会话列表之后,可以根据如下SQL查询某个会话的详细信息,如上次个执行的SQL_ID,登录时间等,该SQL也可改写成多个。

        --墨天轮 session_by_sid
        SELECT s.sid, s.serial#, spid, event, sql_id, PREV_SQL_ID, seconds_in_wait ws, row_wait_obj# obj,
        s.username, s.machine, module,blocking_session b_sess,logon_time  
        FROM v$session s, v$process p 
        WHERE sid = '&sid' AND s.paddr = p.addr
        >>>>

        查询对象信息

            从前面两个SQL都可以看到会话等待的对象ID,可以通过如下SQL查询对象的详细信息。

          --墨天轮 obj_info
          col OBJECT_NAME for a30
          select owner,object_name,subobject_name,object_type
          from dba_objects
          where object_id=&oid

          >>>>

          查询SQL语句

              根据SQL_ID、HASH_VALUE查询SQL语句。如果v$sqlarea中查不到,可以尝试DBA_HIST_SQLTEXT视图中查询。

            --墨天轮 sql_text
            select sql_id,SQL_fullTEXT 
            from v$sqlarea 
            where (sql_id='&sqlid' or hash_value=to_number('&hashvale') ) 
            and rownum<2

                关于SQL语句的执行计划、对象的统计信息、性能诊断、跟踪SQL等这里就不展开,后面计划出一个类似的系列,敬请关注。

            >>>>

            查询会话阻塞情况

                通过如下SQL查询某个会话阻塞了多少个会话。

              --墨天轮 blocking_sess
              select count(*),blocking_session
              from v$session
              where blocking_session is not null
              group by blocking_session;
              >>>>

              查询数据库的锁

                  通过如下SQL查询某个会话的锁,有哪些TM、TX锁,以及会话和锁关联查询的SQL,注意这里指定了ctime大于100秒,30%的情况是人为误操作锁表,导致应用SQL被阻塞,无法运行。

                --墨天轮 lock
                set linesize 180
                col username for a15
                col owner for a15
                col OBJECT_NAME for a30
                col SPID for a10
                --查询某个会话的锁
                select /*+rule*/SESSION_ID,OBJECT_ID,ORACLE_USERNAME,OS_USER_NAME,PROCESS,LOCKED_MODE
                from gv$locked_object where session_id=&sid;
                --查询TM、TX锁
                select /*+rule*/* from v$lock
                where ctime >100 and type in ('TX','TM') order by 3,9;
                --查询数据库中的锁
                select /*+rule*/s.sid,p.spid,l.type,round(max(l.ctime)/60,0) lock_min,s.sql_id,s.USERNAME,b.owner,b.object_type,b.object_name 
                from v$session s, v$process p,v$lock l,v$locked_object o,dba_objects b 
                where  o.SESSION_ID=s.sid and s.sid=l.sid and o.OBJECT_ID=b.OBJECT_ID 
                and s.paddr = p.addr and l.ctime >100 and l.type in ('TX','TM','FB'
                group by s.sid,p.spid,l.type,s.sql_id,s.USERNAME,b.owner,b.object_type,b.object_name 
                order by 9,1,3
                >>>>

                保留现场证据

                    有的问题可能需要分析很长时间,或者是需要外部人员协助分析,那么保留现场证据就非常重要了,下面脚本是systemstate dump和hanganalyze步骤,如果有sqlplus无法登陆的情况,可以加-prelim参数。

                  --systemstate dump
                  sqlplus -prelim as sysdba
                  oradebug setmypid
                  oradebug unlimit;
                  oradebug dump systemstate 266;
                  --wait for 1 min
                  oradebug dump systemstate 266;
                  --wait for 1 min
                  oradebug dump systemstate 266;
                  oradebug tracefile_name;


                  --hanganalyze
                  oradebug setmypid
                  oradebug unlimit;
                  oradebug dump hanganalyze 3
                  --wait for 1 min
                  oradebug dump hanganalyze 3
                  --wait for 1 min
                  oradebug dump hanganalyze 3
                  oradebug tracefile_name
                  >>>>

                  杀会话

                      通常情况下,初步定为问题后为了快速恢复业务,需要去杀掉某些会话,特别是批量杀会话,有时还会直接kill所有LOCAL=NO的进程,再杀会话时一定要检查确认,更不能在别的节点或者别的服务器上执行

                    --墨天轮 kill_sess
                    set line 199
                    col event format a35
                    --杀某个SID会话
                    SELECT /*+ rule */ sid, s.serial#, 'kill -9 '||spid, event, blocking_session b_sess 
                    FROM v$session s, v$process p WHERE sid='&sid' AND s.paddr = p.addr order by 1;
                    --根据SQL_ID杀会话
                    SELECT /*+ rule */ sid, s.serial#, 'kill -9 '||spid, event, blocking_session b_sess 
                    FROM v$session s, v$process p WHERE sql_id='&sql_id' AND s.paddr = p.addr order by 1;
                    --根据等待事件杀会话
                    SELECT /*+ rule */ sid, s.serial#, 'kill -9 '||spid, event, blocking_session b_sess 
                    FROM v$session s, v$process p WHERE event='&event' AND s.paddr = p.addr order by 1;
                    --根据用户杀会话
                    SELECT /*+ rule */ sid, s.serial#, 'kill -9 '||spid, event, blocking_session b_sess 
                    FROM v$session s, v$process p WHERE username='&username' AND s.paddr = p.addr order by 1;
                    --kill所有LOCAL=NO进程
                    ps -ef|grep LOCAL=NO|grep $ORACLE_SID|grep -v grep|awk '{print $2}' |xargs kill -9
                    >>>>

                    重启大法

                        如需要修改静态参数、内存等问题,需要重启数据库,(不要觉得重启很LOW,在很多情况下为了快速恢复业务经常使用这个从网吧里传出来的绝招),记住千万不要在这个时候死磕问题原因、当作课题研究,我们的首要任务是恢复业务。

                      tail -f alert_.log
                      alter system checkpoint;
                      alter system switch logfile;
                      shutdown immediate;


                      startup
                      >>>>

                      CRT按钮小技巧

                          另外介绍一个小技巧,就是把常用的脚本整理到SecureCRT的Button Bar中,只需要点一下设置好的button,就相当于直接执行相应的SQL语句,这样就不用每次粘贴复制执行,或者是把脚本上传到每个服务器上。不过不要设置DDL等操作性的button,以免误点。       

                          以上就是遇到数据库问题用到的一些脚本,特别是应用反应慢、卡的情况,另外建议首先对脚本进行阅读然后再使用,还可以根据自己的环境改写,融会贯通,积累经验。

                          我把这些脚本都整理到墨天轮的常用脚本中,上面还有许多其他监控、管理、诊断的工具脚本,大家可以免费去上面拷贝下载使用。(点击下方阅读原文即可登录墨天轮免费拷贝下载~)

                      最后修改时间:2020-04-02 12:36:41
                      文章转载自章芋文,如果涉嫌侵权,请发送邮件至:contact@modb.pro进行举报,并提供相关证据,一经查实,墨天轮将立刻删除相关内容。

                      评论