当前位置: 首页 > article >正文

[windows] [C++] 由变量命名引起的血案

今天在开发过程中遇到了一个由于变量命名引起的诡异问题,定位了好久,才发现原因,在此记录一下:

最初的代码是:

static bool LoadTestDlls() {
        // 获取可执行文件路径
        WCHAR exePath[MAX_PATH];
        if (GetModuleFileNameW(NULL, exePath, MAX_PATH) == 0) {
            std::wcerr << L"Failed to get module path. Error: " << GetLastError() << std::endl;
            return false;
        }
        ...

在编译的过程中,报了如下编译错误:

11>ModuleATest.cpp
9>F:\tests\test_runner\test_main.cpp(1,1): warning C4819: 该文件包含不能在当前代码页(936)中表示的字符。请将该文件保存为 Unicode 格式以防止数据丢失
10>ModuleBTest.cpp
9>F:\tests\test_runner\test_main.cpp(11,34): error C2065: “exePath”: 未声明的标识符
9>F:\tests\test_runner\test_main.cpp(12,45): error C2065: “exePath”: 未声明的标识符

我最初认为告警好解决,无非是代码中有非 unicode 格式的字符,这通常是由于代码中有中文导致的。
error 想当然得以为是没有给其进行初始化。

于是修改代码为:

    static bool LoadTestDlls() {
        // Get the path of the current executable
        WCHAR exePath[MAX_PATH] = { 0 };
        if (GetModuleFileNameW(NULL, exePath, MAX_PATH) == 0) {
            std::wcerr << L"Failed to get module path. Error: " << GetLastError() << std::endl;
            return false;
        ...

结果,warning 是消除了,可仍然存在那两个 error:

11>ModuleATest.cpp
10>ModuleBTest.cpp
9>F:\tests\test_runner\test_main.cpp(11,34): error C2065: “exePath”: 未声明的标识符
9>F:\tests\test_runner\test_main.cpp(12,45): error C2065: “exePath”: 未声明的标识符

这时候郁闷了,尝试着查 C2065 错误的网络上说明,没啥有用的建议。不经意间将 exePath 的名称改掉,改成 szExePath

咦,再编译,竟然错误消失了。

再次惊讶了我。

这问题得弄清楚,不然就是稀里糊涂的过去了。

再就着这个信息,在网上查找相关蛛丝马迹,得到了,出现这种现象,原因可能有以下几点:

  1. 作用域问题:

    • 在某些编译环境下,特别是使用不同的编译器优化级别时,变量的作用域可能会受到影响
    • 通过重命名变量为 szExePath,我们实际上是重新声明了一个全新的变量,这确保了编译器会正确识别这个变量的作用域
  2. 编译器优化:

    • 有时编译器会对代码进行优化,在某些情况下可能会错误地处理变量的生命周期
    • 通过重命名变量,我们强制编译器重新评估这个变量的使用情况
  3. 内存对齐:

    • 添加 sz 前缀不仅是命名规范,也暗示了这是一个字符串(string with zero termination)
    • 这种明确的命名可能帮助编译器更好地理解变量的用途和内存布局
  4. 符号表处理:

    • 重命名变量会导致编译器重新生成符号表条目
    • 这可能解决了之前可能存在的符号表混淆问题

这就是为什么看似简单的变量重命名能解决这个问题。这也提醒我们在编写代码时要注意:

  1. 使用清晰的命名规范(如Windows API中的 sz 前缀)
  2. 确保变量声明在正确的作用域内
  3. 遵循一致的编码规范

虽然这次是通过重命名解决了问题,但这也说明了在C++开发中,命名规范和变量作用域的重要性。


http://www.kler.cn/a/417443.html

相关文章:

  • PlanLLM: 首个支持开放词汇与封闭集任务的跨模态视频程序规划框架
  • ubuntu20.04+RTX4060Ti大模型环境安装
  • mybatis plus 持久化使用技巧及场景
  • Java 中的异常处理机制是如何工作的?请解释 try-catch-finally语句块的作用 ?
  • 记录一下 在Mac下用pyinstallter 打包 Django项目
  • Python 数据挖掘与机器学习
  • Stable Diffusion 3详解
  • oracle 用户手册
  • Android电视项目焦点跨层级流转
  • CodeIgniter 如何加载多个帮助文件
  • 241130_MindNLP学习记录
  • 【大语言模型】ACL2024论文-22 大型语言模型中的自信心:探究大型语言模型生成回答的自信心与概率一致性
  • C_字符串的一些函数
  • 算法编程题-颜色交替的最短路径
  • Vue.js 组件开发:进阶技巧与最佳实践
  • pytest 通过实例讲清单元测试、集成测试、测试覆盖率
  • 网络安全开发包介绍
  • Qt问题之 “QWidget: Must construct a QApplication before a QWidget“错误
  • 700M是什么?为什么被称为黄金频段?
  • gitee:解决vs文件权限被拒问题
  • 数据结构(初阶7)---七大排序法(堆排序,快速排序,归并排序,希尔排序,冒泡排序,选择排序,插入排序)(详解)
  • 华为小米苹果三星移动设备访问windows共享文件夹windows11
  • Three.js 相机视角的平滑过渡与点击模型切换视角
  • 新型大语言模型的预训练与后训练范式,谷歌的Gemma 2语言模型
  • Wireshark常用功能使用说明
  • 【NLP 1、人工智能与NLP简介】