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

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

最初的代码是:

cpp 复制代码
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 想当然得以为是没有给其进行初始化。

于是修改代码为:

cpp 复制代码
    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:

cpp 复制代码
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++开发中,命名规范和变量作用域的重要性。

相关推荐
黑不溜秋的3 小时前
C++ 设计模式 - 策略模式
c++·设计模式·策略模式
Dream it possible!5 小时前
LeetCode 热题 100_在排序数组中查找元素的第一个和最后一个位置(65_34_中等_C++)(二分查找)(一次二分查找+挨个搜索;两次二分查找)
c++·算法·leetcode
柠石榴5 小时前
【练习】【回溯No.1】力扣 77. 组合
c++·算法·leetcode·回溯
王老师青少年编程5 小时前
【GESP C++八级考试考点详细解读】
数据结构·c++·算法·gesp·csp·信奥赛
澄澈天空6 小时前
C++ MFC添加RichEditControl控件后,程序启动失败
c++·mfc
Lzc7747 小时前
C++初阶——简单实现vector
c++·简单实现vector
一个小白18 小时前
C++——list模拟实现
开发语言·c++
程序员老舅8 小时前
C++ Qt项目教程:WebServer网络测试工具
c++·qt·测试工具·webserver·qt项目·qt项目实战
靡不有初1118 小时前
CCF-CSP第18次认证第一题——报数【两个与string相关的函数的使用】
c++·学习·ccfcsp
菜鸟单飞8 小时前
介绍一款非常实用的PDF阅读软件!
windows·pdf·电脑