linux下用QT捕获程序异常

linux下的QT,没有类似VS捕获系统级异常的功能,可以采用qbreakpad捕获系统级异常。用法如下:

一、部署环境,下载安装包

  • 下载qbreakpad源码:
git clone https://github.com/buzzySmile/qBreakpad.git
  • 下载breakpad源码
    qbreakpad编译需依赖breakpad,执行以下命令安装breakpad
git clone https://github.com/google/breakpad
  • 下载linux-syscall-support
    没有这个文件,编译报错
git clone https://github.com/adelshokhy112/linux-syscall-support.git
  • 将 beakpad 放至qBreakpad/third_party/beakpad
  • 将 linux-syscall-support 里的所有文件复制到qBreakpad/third_party/lss(如果没有lss文件夹就自己新建一个)
  • 编译qbreakpad
    • 进入qbreakpad.pro所在目录执行qmake 生成Makefile文件
    • 执行make 开始编译
    • 编译成功后在handler生成静态库libqBreakpad.a
  • 编译breakpad
    • 如果不编译breakpad无法查看qbreakpad记录的崩溃信息
    • 也要将刚才下载的 linux-syscall-support复制到breakpad/third_party/lss中一份,如果没有会报错
    • 进入/third_party/breakpad,执行
      ./configure  
      Make	
      
    • 如果在breakpad/src目录下生成libbreakpad.a,则编译成功
      由于qbreakpad、breakpad等压缩的下载可能存在下载较慢的情况,大家可以从下方链接下载https://download.youkuaiyun.com/download/qq_45601625/45725264

二、应用示例

.pro文件编写

QT += core gui network
greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
#CONFIG -= app_bundle #配置上这个参数以后 你的图形界面程序就会以命令行方式运行
CONFIG -= app_bundle
TARGET = catchExceptions
TEMPLATE = app

#config for qBreakpad
# 要加上c++11,qbreakpad、breakpad都是基于c++11编写的
CONFIG += c++11 console
CONFIG +=  warn_on
CONFIG += thread exceptions rtti stl
macx: LIBS += -framework AppKit
#link qBreakpad library
include($$PWD/qBreakpad/qBreakpad.pri)
#end of config for qBreakpad

DEFINES += QT_DEPRECATED_WARNINGS

FORMS += \
    mainwindow.ui

HEADERS += mainwindow.h
SOURCES += \
        main.cpp\
        mainwindow.cpp
qnx: target.path = /tmp/$${TARGET}/bin
else: unix:!android: target.path = /opt/$${TARGET}/bin
!isEmpty(target.path): INSTALLS += target

源代码编写

#include "mainwindow.h"
#include <QApplication>
#include "QBreakpadHandler.h"

int crash(int a,int b) {
    return a/b;
   }
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    QBreakpadInstance.setDumpPath(QLatin1String("crashes"));
    MainWindow w;
    w.show();
	int x=5/0;
    crash(5,0);
    return a.exec();
}

当程序发生崩溃后会有如下提示,并生成存放崩溃信息的dmp文件
在这里插入图片描述
三、查看崩溃信息

  • 在生成的可执行程序的目录下,生成.sym文件

    dump_syms ./filename > filename.sym
    

    filename为程序生成的可执行程序,例如
    在这里插入图片描述

  • 在当前目录下建一个symbols目录,symbols下的子文件夹命名为sym文件中第一行的一串字母数字
    在这里插入图片描述
    sym文件
    一定要严格按照这个步骤建立目录,否则无法查看崩溃信息

  • 生成log日志,查看崩溃信息

    minidump_stackwalk ./crashes/800f2c2e-da29-4975-49e8dd80-fc9a519a.dmp 
     ./symbols > error.log
    
    

    error.log内容如下
    在这里插入图片描述

  • 执行如下命令,可查看具体崩溃行数

addr2line 0x406263 -e '可执行程序名字' -f

在这里插入图片描述

四、编写shell脚本

为避免以上繁琐的步骤以及重复性操作,可以将以上操作写成脚本的形式。

#!/bin/bash

if [ $# != 2 ] ; then 
echo "USAGE: $0 EXE_NAME DMP_NAME" 
echo " e.g.: $0 test 3872B2CF-983B-4963-AFA9-C8534DFD4C44.dmp" 
exit 1; 
fi 

#get input param
exe_file_name=$1
dmp_file_name=$2

getSymbol() {
    
    dump_syms ./$exe_file_name > $exe_file_name'.sym'
    
}

getStackTrace() {
    
    sym_file_name=$exe_file_name'.sym'

    #get first line of $sym_file_name
    line1=`head -n1 $sym_file_name`
    
    #get version number from string of first line
    OIFS=$IFS; IFS=" "; set -- $line1; aa=$1;bb=$2;cc=$3;dd=$4; IFS=$OIFS 
    
    version_number=$dd

    #make standard dir and move *.sym in it
    mkdir -p ./symbols/$exe_file_name/$version_number
    mv $sym_file_name ./symbols/$exe_file_name/$version_number

    #print stack trace at a file
    minidump_stackwalk $dmp_file_name ./symbols 2>/dev/null >crashInfo.log
    
    #print crash information
    crashReason=`sed -n 9p ./crashInfo.log`
    echo $crashReason
    crashaddr=`sed -n 10p ./crashInfo.log`
    echo $crashaddr
    addrNum=${crashaddr:15:30}
    echo -n "Crash Location: Crash in "
    addr2line $addrNum -e $exe_file_name -f -p
}

main() {
    getSymbol 
    if [ $? == 0 ] 
    then 
        getStackTrace
    fi
    

}
main

将以上代码命名为dumpTool.sh
在这里插入图片描述
为了能够让这个脚本可以在全局执行,避免每次更换目录都要复制一次脚本很麻烦,可以将dumpTool.sh复制到 /bin 目录下,并生成软连接

sudo cp ./dumpTool.sh /bin/dumpTool.sh
sudo chmod u+x /bin/dumpTool.sh
sudo ln -s dumpTool.sh dumpTool

此时,在任意终端下只需输入

dumpTool '可执行程序' '.dmp文件'

例如,
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值