Visual Studio 断点调试之箭头偏移进错函数,捉虫记

2023-11-07 09:50

本文主要是介绍Visual Studio 断点调试之箭头偏移进错函数,捉虫记,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 背景
    • 问题
    • 解决
    • 系列地址

简 述: Visual Studio 断点调试之箭头偏移进错函数,怪异现象之捉🐛记。


本文初发于 “偕臧的小站”,同步转载于此。


背景

好久没有遇到这么有趣的 Bug 了,来抓一个调试指针出现偏移错误❌的🐛。

在 C++ 实现的派生类和类中重写的两个虚函数 A、B 后,代码实写调用 A 函数,断点 Debug 调试却命中 B 函数。💻 win10 21H2 📎 Visual Studio 2019


MyCollectLogImplIMyCollectLog 的一个派生类,然后代码中调用派生类的 OnCollectLogOne 函数,但调试的箭头和断点却进入了另一 OnCollectLogArray 函数;十分怪异之。

class IMyCollectLog : IMyUnknown {
public:virtual long STDCALLTYPE OnCollectLogOne(const char* sKey, IMyBundle*) = 0;virtual long STDCALLTYPE OnCollectLogArray(const char* sKey, IMyBundleArray) = 0;
};class MyCollectLogImpl : public CloudService::IMyCollectLog
{
public:long STDCALLTYPE OnCollectLogOne(const char* sKey, IMyBundle* pBundle) override {if (!sKey || !pBundle)return 0;// TODO: do somethingreturn 0;}long STDCALLTYPE OnCollectLogArray(const char* sKey, IMyBundleArray* pBundleArray) override {if (!sKey || !pBundleArray)return 0;// TODO: do somethingreturn 0;}// ... 定义其它成员函数和成员
};

调用处的函数为

void onInnerQueryEnd(const char* sKey, IMyBundle* pBundle) {if (m_setCollLogPtr.empty())return;for (auto it : m_setCollLogPtr)it->OnCollectLogOne(sKey, pBundle);   // *** 写的是执行 OnCollectLogOne, 而非是 OnCollectLogArray, 虽然两者行参长得很像 ***
}

用一张图来解释就是:


问题

  1. 断点调试的箭头发生偏移到其它函数。本次遇到
  2. 断点调试时,断点和箭头均无法进入某一函数内部。以前遇过

两者实际为同一根因:

『根因1』Release 编译模式下关闭优化仍有优化,空实现的函数没有对应汇编

『根因2』跨模块加载、创建对象,其 .dll 接口和 .pdb 不对应

『根因3』.dll 和 .pdb 拷贝错了源对象、移动到了错的目标路径下

『根因4』.dll 和 .pdb 拷贝到正确的目标路径,被调试的程序未重启加载此插件(大部分需要)


解决

Release 模式下编译,即使关闭优化[1],实际仍有优化。 如实现一个空的函数,就会被优化掉,反汇编窗口中可查看无对应此函数的汇编语句。『简直是大坑』。


  • Release 编译模式下关闭优化
  • 实现某一函数时却暂不实现细节(仅有 if 判断指针为空和 return 语句,细节未写),仅想调试一下框架时,可在函数中添加一句 int a = 0例子打印语句 避免被编译器优化。
  • 先切换 Debug 模式下编译,替换文件验证代码,在切回 Release 模式定位

注:

[1] Visual Studio 2019 项目属性页,“Configuration Properties - C/C++ - Optimization” 中 Optimization 设置为 Disabled (/Od);使 Release 模式的编译关闭所有优化。


系列地址

QtExamples

欢迎 star ⭐ 和 fork 🍴这个系列的 C++ / QT / DTK 学习,附学习由浅入深的目录。

这篇关于Visual Studio 断点调试之箭头偏移进错函数,捉虫记的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/362814

相关文章

使用Python自建轻量级的HTTP调试工具

《使用Python自建轻量级的HTTP调试工具》这篇文章主要为大家详细介绍了如何使用Python自建一个轻量级的HTTP调试工具,文中的示例代码讲解详细,感兴趣的小伙伴可以参考一下... 目录一、为什么需要自建工具二、核心功能设计三、技术选型四、分步实现五、进阶优化技巧六、使用示例七、性能对比八、扩展方向建

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(

Kotlin 作用域函数apply、let、run、with、also使用指南

《Kotlin作用域函数apply、let、run、with、also使用指南》在Kotlin开发中,作用域函数(ScopeFunctions)是一组能让代码更简洁、更函数式的高阶函数,本文将... 目录一、引言:为什么需要作用域函数?二、作用域函China编程数详解1. apply:对象配置的 “流式构建器”最

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程

kotlin的函数forEach示例详解

《kotlin的函数forEach示例详解》在Kotlin中,forEach是一个高阶函数,用于遍历集合中的每个元素并对其执行指定的操作,它的核心特点是简洁、函数式,适用于需要遍历集合且无需返回值的场... 目录一、基本用法1️⃣ 遍历集合2️⃣ 遍历数组3️⃣ 遍历 Map二、与 for 循环的区别三、高

C语言字符函数和字符串函数示例详解

《C语言字符函数和字符串函数示例详解》本文详细介绍了C语言中字符分类函数、字符转换函数及字符串操作函数的使用方法,并通过示例代码展示了如何实现这些功能,通过这些内容,读者可以深入理解并掌握C语言中的字... 目录一、字符分类函数二、字符转换函数三、strlen的使用和模拟实现3.1strlen函数3.2st

MySQL中COALESCE函数示例详解

《MySQL中COALESCE函数示例详解》COALESCE是一个功能强大且常用的SQL函数,主要用来处理NULL值和实现灵活的值选择策略,能够使查询逻辑更清晰、简洁,:本文主要介绍MySQL中C... 目录语法示例1. 替换 NULL 值2. 用于字段默认值3. 多列优先级4. 结合聚合函数注意事项总结C

Java8需要知道的4个函数式接口简单教程

《Java8需要知道的4个函数式接口简单教程》:本文主要介绍Java8中引入的函数式接口,包括Consumer、Supplier、Predicate和Function,以及它们的用法和特点,文中... 目录什么是函数是接口?Consumer接口定义核心特点注意事项常见用法1.基本用法2.结合andThen链