草庐IT

c++ - 我如何理解fdump-class-hierarchy输出

coder 2024-02-08 原文

我正在使用fdump-class-hierarchy编译器选项,但我不知道如何理解输出。 “大小”,“对齐”,“基本大小”和“基本对齐”是什么意思,它们是如何计数的?谢谢!

当代码是:

class A
{
public:

private:
    double m_nothing;
    int m_number;
};

输出为:
Class A
   size=16 align=8
   base size=16 base align=8
A (0x406c690) 0

但是,如果我稍微改变一下类(class):
class A
{
public:

private:
    int m_number;
    double m_nothing;
};

输出将是:
Class A
   size=16 align=8
   base size=12 base align=8
A (0x406c690) 0

最佳答案

sizealign是用作完整类型时的类的大小和对齐方式。也就是说,如果创建的对象的完整类型就是该类型(例如定义该类型的变量,或者将该类型与new一起使用)。

大小只是它所占用的字节数。因此,size=16表示当用作完整类型时,它始终占用16个字节。

对齐方式告诉您对象的放置位置:align=8表示对象的地址必须为8的整数倍。

如果该类用作基类,则base sizebase align给出大小和对齐方式。它们之所以不同,是因为C++标准允许对象在用作基类时使用较少的填充。

因此,让我们具体看一下您的示例(在第一种情况下,我假设您实际上在int之前拥有double)。我还省略了publicprivate,因为在这里它们什么都不会更改(如果您同时拥有公共(public)或私有(private)数据成员,则它们原则上可以进行某些更改,但是我不知道是否有任何编译器会利用这一点)。我也在猜测intdouble的大小和对齐方式(实际上,我假定的值是相当普遍的选择,并说明您获得的值)。

所以在第一种情况下(我假设)

class A
{
  int m_number;
  double m_nothing;
};

现在int具有大小和对齐方式4,而double具有大小和对齐方式8

因此,让我们完成编译器的工作并构建我们的类。

首先,我们有m_number,它占用4个字节。我们必须按照给定的顺序排列成员,因此m_numberA的开头:
iiii

到现在为止,我们的大小为4(int的四个字节),并且对齐方式为4(因为int的对齐方式为4)。但是现在我们必须添加一个double(大小和对齐方式8)。由于紧接int之后,我们在(相对)地址4处,因此我们没有正确地为double对齐,因此我们必须添加4个填充字节(我将用*标记)以达到8的倍数。因此我们得到我们的类(class):
iiii****dddddddd

现在,如果将该类用作基类,那么我们就完成了。因此,我们将其设为base size=16base align=8(我们需要将8对齐以便正确地将double对齐)。

对于完整的对象,还有另一个考虑因素:标准要求在数组中,对象彼此跟随,且彼此之间没有间隙。也就是说,对象之后的第一个字节必须与下一个对象正确对齐。最终,这意味着完整对象的大小必须为其对齐的倍数。

现在,我们发现的对象布局已满足该要求。因此,我们也可以将其不变地用于完整对象。因此,对于完整的对象,我们得到size=16align=8

现在考虑颠倒顺序的情况:
class A
{
  double m_nothing;
  int m_number;
};

现在我们必须从double开始:
dddddddd

接下来,我们必须添加int。事实证明,下一个空闲位置已针对int正确对齐,因此我们可以将其附加:
ddddddddiiii

现在可以用作基础对象了。如您所见,我们只需要12个字节,因此是base size=12。当然,为了正确地对齐double,对象必须再次从8的倍数的地址开始。因此,我们有了base align=8

但是,对于作为完整对象的起诉,我们现在发现下一个地址将位于位置12,该位置未针对double成员正确对齐。因此,我们必须添加填充字节,直到再次到达正确对齐的地址为止:
ddddddddiiii****

如您所见,现在我们需要16个字节,即size=16。由于加倍,我们仍然有align=8

请注意,对齐要求会极大地影响类的大小。例如,考虑以下两种类型:
struct S1
{
  char c1;
  double d1;
  char c2;
  double d2;
  char c3;
};

struct S2
{
  double d1;
  double d2;
  char c1;
  char c2;
  char c3;
};

虽然两者都包含相同的成员,但是具有上述大小和对齐方式的S1的总(非基本)大小为40,而S2的总大小仅为24。实际上,S1类型的对象将作为完整对象, 看起来像
c*******ddddddddc*******ddddddddc*******

S2类型的代码看起来像
ddddddddddddddddccc*****

因此,最重要的是,具有最高对齐要求的成员应始终排在第一位。

还要注意sizeof返回完整对象的大小,即类层次结构转储调用的size

关于c++ - 我如何理解fdump-class-hierarchy输出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17818556/

有关c++ - 我如何理解fdump-class-hierarchy输出的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

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

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

  3. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  4. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

    给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

  5. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

  6. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  7. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  8. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

    在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

  9. ruby-on-rails - 如何从 format.xml 中删除 <hash></hash> - 2

    我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为

  10. ruby - 检查 "command"的输出应该包含 NilClass 的意外崩溃 - 2

    为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar

随机推荐