vscode c++远程调试实战

简介: vscode c++远程调试实战

在上两篇中,我们介绍了c++开发的最佳实践,以及一些常见问题的解决方案。虽然笔者一直用vscode + remote-ssh + clangd阅读代码,但是调试代码时还是不免脱离vscode回到命令行,这多少让人有点分裂。因此本文将介绍如何在vscode中进行c++远程调试。



环境准备


插件


本文章依赖以下插件


  • remote-ssh,让vscode能够远程登陆远程开发机(我们的开发、编译、部署、调试都在这个环境中)


  • clangd: 一款优秀的c++插件,功能完备,支持重构,跳转,自动补全等功能


  • CodeLLDB:  vscode调试器插件


其中remote-ssh和clangd的安装配置可参考我之前的文章



源代码


main.cpp

#include <iostream>
#include <vector>
#include <unistd.h>
int main(int argc, char *argv[])
{
    int i = 0;
    std::vector<int> v;
    while (1)
    {
        v.push_back(i++);
        ::sleep(1);
    }
    return 0;
}



CMakeList.txt

# Set the minimum version of CMake that can be used
# To find the cmake version run
# $ cmake --version
cmake_minimum_required(VERSION 3.5)
# Set the project name
project (demo)
# Add an executable
add_executable(demo main.cpp)



编译生成binary: main, 注意使用debug模式

cmake -DCMAKE_EXPORT_COMPILE_COMMANDS=YES -DCMAKE_BUILD_TYPE=Debug .
make
ll ./demo




调试新启动的进程


点击 Run and Debug , 选择create a launch.json file

Select Environment选择LLDB,  接着编辑launch.json文件


{
    // Use IntelliSense to learn about possible attributes.
    // Hover to view descriptions of existing attributes.
    // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
    "version": "0.2.0",
    "configurations": [
        {
            "type": "lldb",
            "request": "launch",
            "name": "exec",
            "program": "${workspaceFolder}/demo",
            "args": [],
            "cwd": "${workspaceFolder}"
        }
    ]
}



上述配置中


  • type表示调试工具,我们安装的插件是CodeLLDB, 所以选择lldb


  • request有两种,launch和attach, 前者用于调试新启动进程,后者用于调试运行中进程。


  • name:  对应着Run And Debug窗口中的按钮名称,每个name对应着一组配置,用户可自定义


  • args: 启动待调试program时的参数


当完成编译,配置好launch.json文件后,便可开始调试了。首先在main函数上打个断点。

0e2ec223307b051a12b65a4a304227f3.png

接着点击exec按钮,启动调试器,我们可以看到程序已经运行到断点。整体来说功能还是比较完备的,基本满足日常开发所需:左上角窗口显示了本地/静态/全局变量和寄存器状态。左下角窗口显示当前断点和call stack。


正上方几个按钮对应continue/step over/step into/step out/restart/stop等功能。正下方窗口为lldb命令行,用户如果不习惯点来点去也可在此直接输入调试命令。


1431d9bfef0b5f2e44f49377b004e12e.png


调试已有进程


在launch.json中新增一组配置

{
    "version": "0.2.0",
    "configurations": [
        {
            "type": "lldb",
            "request": "attach",
            "name": "attach",
            "program": "${workspaceFolder}/demo",
            "processId": "${command:pickProcess}",
            "cwd": "${workspaceFolder}"
        }
    ]
}


启动demo,并在后台运行:

nohup ./demo > nohup.log 2>&1 &


我们尝试attach到demo进程上调试,点击attach按钮,此时vscode会让我们选择要attach的进程。

2bc79eee455f8d01bda597eeed9b8bd6.png

输入正确的pid,打上断点,便可开始调试了。

dc8b4e704a1939d99e73996f2f358fca.png

调试coredump文件


c++程序员一定要有坦然接受程序随时coredump的觉悟。


首先修改main.cpp,引入一处致命错误


#include <iostream>
#include <vector>
#include <unistd.h>
int main(int argc, char *argv[])
{
    int i = 0;
    std::vector<int> v;
    while (1)
    {
        v.push_back(i++);
        ::sleep(1);
        if (i % 10 == 0)
            std::abort();
    }
    return 0;
}



执行ulimit -c unlimited, 编译运行之,果然出现了core

$ ./demo                                                    
[1]    18667 abort (core dumped)  ./demo



修改launch.json,新增配置后,launch.json变成现在这样

{
    // Use IntelliSense to learn about possible attributes.
    // Hover to view descriptions of existing attributes.
    // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
    "version": "0.2.0",
    "configurations": [
        {
            "type": "lldb",
            "request": "launch",
            "name": "exec",
            "program": "${workspaceFolder}/demo",
            "args": [],
            "cwd": "${workspaceFolder}"
        },
        {
            "type": "lldb",
            "request": "attach",
            "name": "attach",
            "program": "${workspaceFolder}/demo",
            "processId": "${command:pickProcess}",
            "cwd": "${workspaceFolder}"
        }
        {
            "type": "lldb",
            "request": "custom",
            "name": "core",
            "initCommands": [
                "target create ${workspaceFolder}/demo -c ${input:coreFileName}"
            ]
        }
    ],
    "inputs": [
      {
        "id": "coreFileName",
        "type": "promptString",
        "description": "Enter core file path"
      }
    ]    
}



点击core按钮,输入core file path, 便可查看core时的现场


0bdeee3849897bdc7bfca2684fd25e40.png

相关文章
|
2月前
|
存储 算法 Linux
【实战项目】网络编程:在Linux环境下基于opencv和socket的人脸识别系统--C++实现
【实战项目】网络编程:在Linux环境下基于opencv和socket的人脸识别系统--C++实现
123 7
|
25天前
|
缓存 网络协议 Linux
c++实战篇(三) ——对socket通讯服务端与客户端的封装
c++实战篇(三) ——对socket通讯服务端与客户端的封装
|
1月前
|
存储 自然语言处理 安全
C++ STL标准库 《string原理与实战分析》
C++ STL标准库 《string原理与实战分析》
36 0
|
1月前
|
编译器 C语言 C++
【VS Code】安装配置调试C/C++(一)
【VS Code】安装配置调试C/C++(一)
44 0
|
1月前
|
C++
【C/C++基础实战】:用C++实现通讯录管理系统——含完整源码
【C/C++基础实战】:用C++实现通讯录管理系统——含完整源码
|
1月前
|
项目管理 C++
【VS Code】安装配置调试C/C++(二)
【VS Code】安装配置调试C/C++(二)
43 1
|
25天前
|
监控 C++
c++实战篇(二)——基于自旋锁实现的日志服务模块
c++实战篇(二)——基于自旋锁实现的日志服务模块
|
25天前
|
调度 C++
C++实战篇(一)——自旋锁的使用
C++实战篇(一)——自旋锁的使用
|
2月前
|
安全 Go 对象存储
C++多线程编程:并发与同步的实战应用
本文介绍了C++中的多线程编程,包括基础知识和实战应用。C++借助`&lt;thread&gt;`库支持多线程,通过`std::thread`创建线程执行任务。文章探讨了并发与同步的概念,如互斥锁(Mutex)用于保护共享资源,条件变量(Condition Variable)协调线程等待与通知,以及原子操作(Atomic Operations)保证线程安全。实战部分展示了如何使用多线程进行并发计算,利用`std::async`实现异步任务并获取结果。多线程编程能提高效率,但也需注意数据竞争和同步问题,以确保程序的正确性。
|
1月前
|
大数据 C++ 索引
C++ STL标准库 《vector向量原理与实战分析》
C++ STL标准库 《vector向量原理与实战分析》
27 0