如何(临时)使用 natvis 对 CPtrList 条目进行类型转换?

Posted

技术标签:

【中文标题】如何(临时)使用 natvis 对 CPtrList 条目进行类型转换?【英文标题】:How to (temporarily) typecast CPtrList entries using natvis? 【发布时间】:2018-01-17 15:55:56 【问题描述】:

我正在使用基于 STL 的 C++ 解决方案,并且我正在使用 CPtrList 集合。

我这里有一个 CPtrList 集合,其中包含 void * 条目,我想使用 natvis 文件自动对它们进行类型转换。

目前,我的 natvis 如下所示:

<Type Name="CList&lt;*,*&gt;">
  <AlternativeType Name="CObList"></AlternativeType>
  <AlternativeType Name="CPtrList"></AlternativeType>
  <AlternativeType Name="CStringList"></AlternativeType>
  <AlternativeType Name="CTypedPtrList&lt;*,*&gt;"></AlternativeType>
  <DisplayString>iets anders Count = m_nCount</DisplayString>
  <Expand>
    <Item Name="Count">m_nCount</Item>
    <LinkedListItems>
      <Size>m_nCount</Size>
      <HeadPointer>m_pNodeHead</HeadPointer>
      <NextPointer>pNext</NextPointer>
      <ValueNode>data</ValueNode>
    </LinkedListItems>
  </Expand>
</Type>

因此,我的 CPtrList 的条目如下所示:

0x<something>      void *
0x<something else> void *
...

我希望将条目类型转换为如下内容:

<information>      CElement::SL_SET_PARAMETER*
<information else> CElement::SL_SET_PARAMETER*

一旦我知道如何完成这项工作,我就可以在我的 natvis 中添加一个“SL_SET_PARAMETER”条目并决定如何显示它,但因此我首先需要向 natvis 解释每个 CPtrList 条目都应该转换为“SL_SET_PARAMETER” " 对象。

有人知道怎么做吗?

【问题讨论】:

【参考方案1】:

您必须使用&lt;CustomListItems&gt; 标记(有关详细信息,请参阅MS documentation 中的CustomListItems 扩展项)。这是允许局部变量和循环的显示类型的最通用规范。

他们在文档中使用的示例如下:

<Type Name="ATL::CAtlMap&lt;*,*,*,*&gt;">  
    <AlternativeType Name="ATL::CMapToInterface&lt;*,*,*&gt;"/>  
    <AlternativeType Name="ATL::CMapToAutoPtr&lt;*,*,*&gt;"/>  
    <DisplayString>Count = m_nElements</DisplayString>  
    <Expand>  
      <CustomListItems MaxItemsPerView="5000" ExcludeView="Test">  
        <Variable Name="iBucket" InitialValue="-1" />  
        <Variable Name="pBucket" InitialValue="m_ppBins == nullptr ? nullptr : *m_ppBins" />  
        <Variable Name="iBucketIncrement" InitialValue="-1" />  

        <Size>m_nElements</Size>  
        <Exec>pBucket = nullptr</Exec>  
        <Loop>  
          <If Condition="pBucket == nullptr">  
            <Exec>iBucket++</Exec>  
            <Exec>iBucketIncrement = __findnonnull(m_ppBins + iBucket, m_nBins - iBucket)</Exec>  
            <Break Condition="iBucketIncrement == -1" />  
            <Exec>iBucket += iBucketIncrement</Exec>  
            <Exec>pBucket = m_ppBins[iBucket]</Exec>  
          </If>  
          <Item>pBucket,na</Item>  
          <Exec>pBucket = pBucket->m_pNext</Exec>  
        </Loop>  
      </CustomListItems>  
    </Expand>  
</Type>  

它唯一的小问题是,如果你从监视窗口复制它创建的表达式,它看起来就像一个数字转换为你想要的类型的指针,而不是像语法和语法那样漂亮的数组因此,如果内存位置移动,它将导致它不会被更新。如果您引用包含对象的父对象,这没什么大不了的,只是烦人。

【讨论】:

以上是关于如何(临时)使用 natvis 对 CPtrList 条目进行类型转换?的主要内容,如果未能解决你的问题,请参考以下文章

.natvis - 如何引用模板模板参数?

如何从 C 语言中的 natvis 表达式中引用变量本身?

如何在 natvis 中传播 ExcludeView/IncludeView?

在使用外部调试器时在 VS2013 中使用 NatVis 文件

如何将单个字符的 natvis 限制为仅 Visual Studio 中的字符

在 VSCode/Natvis 中使用 CustomListItems