jjzjj

c++ - 在 Visual Studio 中调试 Qt 应用程序并检查不透明对象(例如 QDir 或 QFileInfo)

coder 2024-02-10 原文

我正在 Visual Studio 2013 中调试 Qt 应用程序。我安装了来自 Qt 的官方 Visual Studio 插件。我可以看到QString的内容,但对于任何其他 QObjectQDirQFileInfo对象我看不到它的内容。

我做错了什么还是根本不可能?

当我展开 QDir实例我只能看到一个名为 d_ptr 的成员其中引用了 QDirPrivate我无法检查的对象。我也不能调用QDir的其他函数或 QFileInfopath() , 或 filePath()在运行时从调试器。当我尝试时,Visual Studio 声称该函数的地址已被优化掉。

有没有办法在不向程序中添加数十条日志语句的情况下调试它?

最佳答案

更新
在上一个答案中详细说明这是不可能的之后,今天我找到了一种方法可以让调试器显示私有(private)(隐藏)信息。
注意:我保留下面的旧帖子,因为它揭示了我达到此解决方案所遵循的整个路径,并且我认为值得跟踪所犯的错误和错误的结论,因此没有其他人再次提出;)。
假设你有一个 QDir d;多变的。您可以通过在 Watch 窗口中进行手动转换来检查其内部数据:

(Qt5Cored.dll!QDirPrivate*)d.d_ptr.d

不幸的是,它没有显示任何有用的信息,因为内部成员将面临同样的问题(未打印私有(private)数据)。
但是,从这一点开始,您现在可以修改 .natvis 文件以在悬停时显示所需的数据:
<?xml version="1.0" encoding="utf-8"?>
<AutoVisualizer xmlns="http://schemas.microsoft.com/vstudio/debugger/natvis/2010">
  <Type Name="QDir">
    <DisplayString>path={((Qt5Cored.dll!QDirPrivate*)d_ptr.d)-&gt;dirEntry.m_filePath}</DisplayString>
  </Type>
</AutoVisualizer>

我无法在 Watch 窗口上打印此信息(如果我将 Expand 部分添加到 .natvis 文件中,它似乎被省略了),但至少您有关于调试时间的这些信息。
我建议您查看要检查的类型的私有(private)头文件或 Watch 窗口(如答案开头所述),以便将正确的数据添加到 .natvis 文件中。
之前的分析有什么问题?
我认为关键点是手动类型转换。我提到过

dirEntry is part of QDirPrivate, so it is clear that the debugger is not able to inspect it.


这是部分正确的。问题是调试器不知道在哪里定位QDirPrivate因为它不是 Qt 的导出符号;一旦我们明确指出它(通过 (Qt5Cored.dll!QDirPrivate*) 类型转换),它就能够检查它。

原答案
AFAIK 这对于那些类是不可能的。不幸的是,有些私有(private)结构没有导出,而且调试器似乎无法查看。
注意:我正在使用 Qt 5.6.1,所以我将使用 qt.io 存储库中的源代码来匹配我的示例。
让我们拿 QDir例如:它在 qdir.h 中定义,它是 Qt 公共(public) API 的一部分。在同一个文件中,它声明 QDirPrivate但它的完整定义在 qtdir_p.h这不是公共(public) API 的一部分。
即使符号被加载,这个类的定义对调试器来说仍然是模糊的。请参阅下面我使用的 .natvis 文件的示例:
<?xml version="1.0" encoding="utf-8"?> 
<AutoVisualizer xmlns="http://schemas.microsoft.com/vstudio/debugger/natvis/2010">
  <Type Name="QDir">
    <DisplayString>Path: {d_func()-&gt;dirEntry.m_filePath}</DisplayString>
  </Type>
</AutoVisualizer>
但是调试器在尝试观看时显示以下内容 QDir d(qApp->applicationDirPath()); :

Error: class "QDirPrivate" has no member "dirEntry"

Error while evaluating 'd_func()->dirEntry.m_filePath'


但是 dirEntryQDirPrivate 的一部分,所以很明显调试器无法检查它。 dirEntry类型为 QFileSystemEntry ,定义于 qfilesystementry_p.h (另一个私有(private)类(class))。
最后,如果你看一下 qt5.natvis从 Qt VS Tools 中打开文件并在源代码中找到相应的类,您将看到所有包含的类都公开了 .natvis 文件使用的结构的定义。

更新
此类类的唯一公共(public) API 是方法,不幸的是,不支持从调试器调用函数。引用 MSDN forum from a Microsoft Staff 中的响应:

Calling a function from the debugger is playing with fire. You are likely to deadlock on cross thread dependencies (even if you don't have any explicit cross thread dependencies, there are shared locks for things like memory allocation). This is why the C++ debugger has no support for implicit funceval.


实际上,如果您尝试调用任何方法,例如 QDir::absolutePath()在监视窗口或 .natvis 文件中,您将看到以下错误消息:

Error: Function QDir::absolutePath has no address, possibly due to compiler optimizations.



不太优雅的解决方法
一个可能的解决方案是使用存储这些私有(private)值的包装类。您需要用包装器替换每个对象,但可能会有所帮助。
下面你会找到一个非常简单的例子 QDir (您需要完成所需的构造函数并保存所需的信息)。请记住,Qt 中的此类类被设计为不可扩展的,因此没有虚拟方法(因此请注意重新实现其中的一些并使用转换为基类的对象)。
class MyQDir : public QDir {
public:
  MyQDir(const QString& path) : QDir(path) {
    m_absolutePath = absolutePath();
  }

private:
  QString m_absolutePath;
};
和以下 .natvis 文件显示 MyQDir悬停时的路径:
<?xml version="1.0" encoding="utf-8"?>
<AutoVisualizer xmlns="http://schemas.microsoft.com/vstudio/debugger/natvis/2010">
  <Type Name="MyQDir">
    <DisplayString>Path: {m_absolutePath}</DisplayString>
  </Type>
</AutoVisualizer>
最后,我认为剩下的唯一解决方案是将信息打印到控制台( qDebug() )。

作为旁注,this tutorial解释如何编写自定义 .natvis 文件。这是 2015 年的,但我在 2017 年完美地使用了它。我希望它也适用于 2013 年。

关于c++ - 在 Visual Studio 中调试 Qt 应用程序并检查不透明对象(例如 QDir 或 QFileInfo),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44763417/

有关c++ - 在 Visual Studio 中调试 Qt 应用程序并检查不透明对象(例如 QDir 或 QFileInfo)的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. ruby-on-rails - 按天对 Mongoid 对象进行分组 - 2

    在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev

  3. ruby - 将差异补丁应用于字符串/文件 - 2

    对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

  4. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  5. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  6. ruby - 无法运行 Rails 2.x 应用程序 - 2

    我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

  7. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  8. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

    刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

  9. ruby-on-rails - 如何验证非模型(甚至非对象)字段 - 2

    我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss

  10. Ruby 写入和读取对象到文件 - 2

    好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信

随机推荐