C++@vscode配置C++开发环境常见问题和实践经验

devtools/2024/10/19 2:15:24/

文章目录

    • abstract
    • vscode配置C/C++开发环境常见问题 FAQ
      • C/C++共用一组tasks.json/launch.json文件?
      • 关于配置文件中的注释
      • 更快地编译运行
      • 调试时调用外部终端控制台
      • 二次编译失败问题
      • 编译多个源文件😊
        • 源文件组织
      • 编译出的可执行文件名中文乱码😊
      • 修改tasks.json和launch.json
    • 早期使用MinGw配置Vscode的C/C++编程环境的探索

abstract

  • 本文是 C++@vscode@语言编程环境配置的延续
  • 介绍了使用vscode编写C/C++过程中的细节问题

vscodeCC_FAQ_6">vscode配置C/C++开发环境常见问题 FAQ

C/C++共用一组tasks.json/launch.json文件?

  • 您可以考虑vscode中新建工作区,来分别配置.vscode中的相关文件
  • 也可以让c/c++共用一组配置文件,但是json文件里的task,等可以相对独立,因此也是可以考虑两种语言的build task 和debug方案配置在同一组文件中

关于配置文件中的注释

  • 在json文件中编写注释是不可靠的

  • 顶多是创建一个comment字段或者description,或者detail字段,然后将注释字符串写入到该字段的值下

  • 另一方面,使用vscode command palette配置tasks.json等文件时,可能会清空掉所有//引出的注释,从而导致注释丢失

更快地编译运行

  • 使用launch.json启动编译和调试虽然可以一键执行,但是速度比较慢

  • 要知道直接用g++命令行编译是很快的,所以我们可以设法改进这一点

  • 我们可以用Code Runner插件来快速执行编译并直接运行,而不是编译后启动调试运行

    • Code Runner - Visual Studio Marketplace
  • 我们以hellow,world为例

    • 配置后按下快捷键可以有如下效果

      PS C:\repos\C_CPP_ConsoleApps> cd "c:\repos\C_CPP_ConsoleApps\cpp\" ; if ($?) { g++ -std=c++11 "hellowworld2.cpp" -o "hellowworld2.exe" } ; if ($?) {  .\"hellowworld2.exe" }
      Hello World!
      
    • 如果使用默认的start debugging,速度要慢上许多

      • PS C:\repos\C_CPP_ConsoleApps>  & 'c:\Users\cxxu\.vscode\extensions\ms-vscode.cpptools-1.20.3-win32-x64\debugAdapters\bin\WindowsDebugLauncher.exe' '--stdin=Microsoft-MIEngine-In-pcmtfxlj.w4p' '--stdout=Microsoft-MIEngine-Out-f12ge22g.01f' '--stderr=Microsoft-MIEngine-Error-i3ud5s4n.nq2' '--pid=Microsoft-MIEngine-Pid-cwud2zvq.q2k' '--dbgExe=C:\msys64\ucrt64\bin\gdb.exe' '--interpreter=mi' 
        Hello World!
        
  • 配置tasks.json(不推荐)

    • "args": ["-fdiagnostics-color=always",// "-g","${file}","-o","${fileDirname}\\a.exe","&&","a.exe"],
      
    • 您或许考虑创建一个task令其编译完成后追加运行

    • 但是这并不好用,估计vscode c++ extension并没有打算让用户这么用

      • 如果仅仅是输出hello,world 这种程序还要,但是如果先要输入,那么build task就会被卡住

调试时调用外部终端控制台

  • 通常没有必要设置为外部终端打开,vscode内的集成终端以及足够使用了;
    • vscode中程序执行完毕后外置终端就会自动退出,除非使用Pause来阻止退出
    • Pause可以在源代码末尾中添加system(“Pause”);但是这就显得有些繁琐了,Dev C++等IDE会自动执行这个命令,而不需要我们手动设置
  • 目前vscode c/c++ extension 没有支持这种自动pause的功能,因此我建议是除非必要或者心理过不去,否则就用vscode继承终端来调试程序的运行(输入和输出数据)

二次编译失败问题

  • 个别情况下,如果前一次编译运行的程序没有被退出,而新的编译结果名字和上一次编译的名字一样,那么会因为名字被占用导致新的编译结果无法保存而失败

  • 这可能表现为cannot open output file ...,Permission denied

    • Starting build...
      cmd /c chcp 65001>nul && g++.exe -fdiagnostics-color=always -g C:\repos\C_CPP_ConsoleApps\cpp\hellowworld2.cpp -o C:\repos\C_CPP_ConsoleApps\cpp\a.exe
      C:/msys64/ucrt64/bin/../lib/gcc/x86_64-w64-mingw32/13.2.0/../../../../x86_64-w64-mingw32/bin/ld.exe: cannot open output file C:\repos\C_CPP_ConsoleApps\cpp\a.exe: Permission deniedcollect2.exe: error: ld returned 1 exit status
      
  • 这种情况还可能发生在同一个目录下编译了多个文件,假设它们的编译结果都保存为文件a.exe,而在某个a.exe尚未退出时编译了另一个文件,它们恰好在同一个目录下,就会造成失败

  • 尽管这种情况很少见,我们通常同一时间同一目录内,仅处理一个程序;但是如果确实需要同时编译运行多个程序,有如下选择

    • 更换编译结果文件名后再编译:既然默认的名字被占用,那么可以切换不同的task配置再编译,或者临时用命令行指定新的-o参数后编译

    • 或者如果上一个程序是因为意外卡住,或者我们不需要它继续运行时,可以借助任务管理杀死相关进程后再编译,例如可以用任务管理,也可以用命令行处理

      • powershell查找名为a.exe的进程

        ps a.exe|select path
        
      • 确认路径后手动杀死

        PS> ps a|select pathPath
        ----
        C:\repos\C_CPP_ConsoleApps\cpp\a.exePS> ps a|kill
      • 然后重新尝试编译即可

编译多个源文件😊

  • task.json和launch.json分别负责控制编译和调试

  • 通常简单的需求只需要在默认生成的文件内稍作修改就可以工作

  • 然而对于稍微复杂的编译需求,我们需要进一步的配置

    • 比如配置能够编译/调试多个源文件的情况
    • 配置能够编译/调试中文源文件的情况
源文件组织
  • 通常我们将同一个程序(源代码分散到了多个不同源文件)的相关源文件存放到同一个目录中,而其他程序的源文件就不要混进来,这样不利于管理和一键编译的配置
  • 换句话说,假设您将多个相互独立(或者没有关系的)源文件放到同一个目录,那么一口气将它编译链接就会出问题
    • 尽管我们可以用命令行手动选择要编译哪些源文件,但是不利于我们配置task.json进行一键编译
    • 所以这里强调同一个程序有多个源文件时要组织到一个单独的目录中去,并且不要参杂其他程序源文件

编译出的可执行文件名中文乱码😊

  • 实验发现,对于文件名为中文的C++/C语言源代码编译时得到的.exe文件名会是乱码

  • 这就导致launch.json中默认配置的调试语句找不到编译后的可执行文件,导致调试报错无法进行下去

  • 对于简单控制台程序,从实践的角度,通常我们希望编写玩一个c++程序,然后可以一键编译运行或调试,就可以了

  • 当然最好能够兼容中文名源文以及英文名源文件

  • 现在的问题是遇到中文名源文件,windows上的Gcc(Mingw或其他移植Gcc)的项目编译中文名c++文件在不同场景下有不同的效果

    • PS[BAT:69%][MEM:42.83% (13.58/31.70)GB][21:01:23]
      # [C:\repos\C_CPP_ConsoleApps]g++ .\勾股.cpp -o 勾股.exePS[BAT:69%][MEM:42.79% (13.56/31.70)GB][21:01:48]
      # [C:\repos\C_CPP_ConsoleApps]ls *.exeDirectory: C:\repos\C_CPP_ConsoleAppsMode                LastWriteTime         Length Name
      ----                -------------         ------ ----
      -a---         2024/4/24     21:01         140467 󰣆  勾股.exe
    • 上述试验说明,在命令行中手动调用g++来编译没有没有出现文件名乱码的结果

    • 但是执行的时候可能会输出程序内部带出来的中文乱码,我们可以追加参数-finput-charset=UTF-8 -fexec-charset=gbk

  • 示例

    • PS[BAT:79%][MEM:46.76% (14.82/31.70)GB][22:09:28]
      # [C:\repos\C_CPP_ConsoleApps]cat  C:\repos\C_CPP_ConsoleApps\cpp\helloworld.cpp
      #include <iostream>
      #include <vector>
      #include <string>
      #include <iostream>
      #include <windows.h> // For system("pause")
      using namespace std;int main()
      {vector<string> msg{"Hello", "C++", "World", "from", "VS Code", "and the C++ extension!"};int i=0;for (const string &word : msg){++i;cout << word  << " ";}cout << endl;cout << "中文会乱码吗?"<<endl;
      }
      
    • PS[BAT:79%][MEM:46.74% (14.82/31.70)GB][22:08:42]
      # [C:\repos\C_CPP_ConsoleApps]gcc c:\repos\C_CPP_ConsoleApps\cpp\helloworld.cpp -o demo.exe -lstdc++ -finput-charset=UTF-8 -fexec-charset=gbk
      #或者  g++ .\helloworld.cpp -o demo.exe -finput-charset=UTF-8 -fexec-charset=gbkPS[BAT:79%][MEM:46.79% (14.83/31.70)GB][22:08:53]
      # [C:\repos\C_CPP_ConsoleApps].\demo.exe
      Hello C++ World from VS Code and the C++ extension!
      中文会乱码吗?
      
  • 然而,vscode中匹配了tasks.json和launch.json进行一键编译运行时确发生了二进制文件中文名称乱码的情况

修改tasks.json和launch.json

  • 文档中指出,我们可以通过配置

    • tasks.json中的编译选项(-o后的值(编译结果)为英文硬编码,比如a.exe)

      • "args": ["-fdiagnostics-color=always","-g","${file}","-o","${fileDirname}\\a.exe" //修改这里,默认的值为"${fileDirname}\\${fileBasenameNoExtension}.exe"
        ],
        
    • 然后再配置launch.json中的program字段

      • "program": "${fileDirname}\\a.exe",
    • 也就是编译出来的名字总是命名为a.exe,然后调试时找的也是同目录下的a.exe,避开了中文可能导致的乱码,导致调试器找不到对应的可执行文件报错

  • 上述操作虽然没有正面解决中文文件名乱码问题,但是已经能够让我们比较舒服的使用vscode来学习C/C++编程和算法,能够一键编译和运行/调试代码,甚至可以配置一键编译分散在多个源文件的程序,中文文件名确实也不报错了,基础阶段非常够用

  • 如果确实需要所有编译出来的可执行文件,那么写个脚本就行

早期使用MinGw配置Vscode的C/C++编程环境的探索

  • 早期的实践,操作比较稚嫩,有缘可能会回去改改:链接在此浏览 (csdn.net)

http://www.ppmy.cn/devtools/34926.html

相关文章

【计算机网络】网络层总结

目录 知识梗概 IP地址 子网划分 IP包头格式 路由 网络层协议 ARP病毒/ARP欺骗 知识梗概 IP地址 IP相关介绍&#xff1a;机器之间需要交流&#xff0c;必须要一个地址才能找到对应的主机&#xff0c;IP地址是主机的一种表示&#xff0c;保证主机之间的正常通信&#xff…

idea2023.2.5的控制台动态配置当前环境

一、idea2023.2.5的控制台动态配置当前环境 1.1、idea版本 1.2、配置方式 1.2.1、方式一 1.2.2、方式二 1.3、参考 https://blog.csdn.net/xiaoheihai666/article/details/127757658

Vue学习:22.Vue组件库-Vant

Vue组件库是一系列预先构建好的、可复用的UI组件集合&#xff0c;它们设计用于加速Vue.js应用程序的开发过程。这些组件通常遵循一定的设计规范&#xff0c;提供统一的外观和交互体验&#xff0c;让开发者能够快速搭建用户界面。 组件库举例 Vant: 专注于移动端的轻量级UI库&a…

消息队列使用常见问题

一、消息丢失的时机? 生产端消息丢失 问题:因为网络异常导致消息发送失败,此时可能会产生消息丢失的情况,重试后可能产生消息重复生产的情况。 解决:超时重试,并在消费端保证幂等性。 消息队列中消息丢失 问题:如kafka是将数据存储在磁盘,如果每条消息写入时都立马存储…

6.k8s中的secrets资源

一、Secret secrets资源&#xff0c;类似于configmap资源&#xff0c;只是secrets资源是用来传递重要的信息的&#xff1b; secret资源就是将value的值使用base64编译后传输&#xff0c;当pod引用secret后&#xff0c;k8s会自动将其base64的编码&#xff0c;反编译回正常的字符…

Array.from()详解

Array.from() 是 JavaScript 中的一个静态方法&#xff0c;用于从一个类似数组或可迭代对象&#xff08;如 Set, Map 的迭代器&#xff0c;或者字符串、类数组对象等&#xff09;创建一个新的、浅拷贝的数组实例。 语法 Array.from(arrayLike, [mapFunction, [thisArg]])arra…

AHB---时钟和复位

1. 时钟 每个组件使用单一的时钟信号&#xff0c;HCLK。所有输入信号都在HCLK的上升沿被采样。所有输出信号的变化必须发生在HCLK上升沿之后。 被描述为稳定的信号需要在不同上升时钟沿采样时保持相同的值&#xff0c;即使在扩展传输中也是如此。然而&#xff0c;这些信号在时…

按键精灵纯本地离线文字识别插件

目的 按键精灵是一款可以模拟鼠标和键盘操作的自动化工具。它可以帮助用户自动完成一些重复的、繁琐的任务&#xff0c;节省大量人工操作的时间。但按键精灵是不包含图色功能&#xff0c;无法识别屏幕上的图像&#xff0c;根据图像的变化自动执行相应的操作。本篇文章主要讲解下…