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

相关文章
|
21天前
|
C++
C++ 语言异常处理实战:在编程潮流中坚守稳定,开启代码可靠之旅
【8月更文挑战第22天】C++的异常处理机制是确保程序稳定的关键特性。它允许程序在遇到错误时优雅地响应而非直接崩溃。通过`throw`抛出异常,并用`catch`捕获处理,可使程序控制流跳转至错误处理代码。例如,在进行除法运算或文件读取时,若发生除数为零或文件无法打开等错误,则可通过抛出异常并在调用处捕获来妥善处理这些情况。恰当使用异常处理能显著提升程序的健壮性和维护性。
38 2
|
3月前
|
缓存 网络协议 Linux
c++实战篇(三) ——对socket通讯服务端与客户端的封装
c++实战篇(三) ——对socket通讯服务端与客户端的封装
|
3月前
|
存储 自然语言处理 安全
C++ STL标准库 《string原理与实战分析》
C++ STL标准库 《string原理与实战分析》
60 0
|
14天前
|
编解码 编译器 C++
VS Code C/C++ MSVC编译器
VS Code C/C++ MSVC编译器
40 2
|
21天前
|
存储 算法 C++
C++ STL应用宝典:高效处理数据的艺术与实战技巧大揭秘!
【8月更文挑战第22天】C++ STL(标准模板库)是一组高效的数据结构与算法集合,极大提升编程效率与代码可读性。它包括容器、迭代器、算法等组件。例如,统计文本中单词频率可用`std::map`和`std::ifstream`实现;对数据排序及找极值则可通过`std::vector`结合`std::sort`、`std::min/max_element`完成;而快速查找字符串则适合使用`std::set`配合其内置的`find`方法。这些示例展示了STL的强大功能,有助于编写简洁高效的代码。
31 2
|
27天前
|
NoSQL 编译器 C语言
VSCode配置配置C++环境
VSCode配置配置C++环境
48 1
|
29天前
|
C语言 C++
vscode——如何在vscode中运行C/C++
vscode——如何在vscode中运行C/C++
33 1
|
11天前
|
图形学 C++ C#
Unity插件开发全攻略:从零起步教你用C++扩展游戏功能,解锁Unity新玩法的详细步骤与实战技巧大公开
【8月更文挑战第31天】Unity 是一款功能强大的游戏开发引擎,支持多平台发布并拥有丰富的插件生态系统。本文介绍 Unity 插件开发基础,帮助读者从零开始编写自定义插件以扩展其功能。插件通常用 C++ 编写,通过 Mono C# 运行时调用,需在不同平台上编译。文中详细讲解了开发环境搭建、简单插件编写及在 Unity 中调用的方法,包括创建 C# 封装脚本和处理跨平台问题,助力开发者提升游戏开发效率。
25 0
|
14天前
|
编译器 C++
VS Code设置C++编译器路径
VS Code设置C++编译器路径
26 0
|
2月前
|
Java Android开发 C++
🚀Android NDK开发实战!Java与C++混合编程,打造极致性能体验!📊
【7月更文挑战第28天】在 Android 开发中, NDK 让 Java 与 C++ 混合编程成为可能, 从而提升应用性能。**为何选 NDK?** C++ 在执行效率与内存管理上优于 Java, 特别适合高性能需求场景。**环境搭建** 需 Android Studio 和 NDK, 工具如 CMake。**JNI** 构建 Java-C++ 交互, 通过声明 `native` 方法并在 C++ 中实现。**实战** 示例: 使用 C++ 计算斐波那契数列以提高效率。**总结** 混合编程增强性能, 但增加复杂性, 使用前需谨慎评估。
83 4