- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
只是想知道,如果我有一个指向最派生类的指针,并在其上调用最派生类定义的虚函数,这是否仍然会导致在虚拟表中进行查找?
毕竟在编译时,编译器知道这个类是最派生的,它知道它定义了虚函数,没有歧义,所以它应该把它当作非虚函数吗?
还是我遗漏了什么?
我问的原因是我正在编写一个模板,稍后我想从中派生以合并代码,并且不同的功能将在派生类中实现。
没有必要在模板中将这些函数定义为虚拟函数,但如果虚拟调用稍后会被忽略,我正在考虑这样做,纯粹是为了让实现者稍后看到哪些函数仍需要编写。
最佳答案
这个答案是关于编译器优化技术的。您的编译器可能支持也可能不支持这些。即使它支持您尝试利用的技术,它们也可能在您选择的优化级别不可用。
您的里程可能会有所不同。
编译器确实可以将调用去虚拟化如果它知道这确实是最派生的类。讨论了如何实现这一点 here .一个例子:
struct Base { virtual void call_me_virtual() = nullptr; };
struct Derived final : Base { void call_me_virtual() override { } };
void dosomething(Derived* d) {
d->call_me_virtual();
}
有趣的是,如果不这样做,您总是可以让其他人从您的类派生在另一个翻译单元中,因此编译器不会意识到您当前的“更多派生”类翻译单元。
另一种确保类必须是最派生的方法是将其放入匿名命名空间:
struct Base { virtual void call_me_virtual() = nullptr; };
namespace {
struct Derived : Base { void call_me_virtual() override { } };
void dosomething(Derived* d) {
d->call_me_virtual();
}
}
这是有效的,因为 Derived
在当前翻译单元之外是未知的,这意味着任何更多的派生类必须在当前翻译单元内。然而,这意味着 dosomething
不能(正确地)从当前编译单元外部调用,这就是为什么我也给了它内部链接。
此规则的一个豁免是编译器能够证明该对象来自他们的 View ,例如如果 Derived
是当前翻译单元中派生程度最高的类型,并且对象必须始终来自当前翻译单元,那么它不能来自任何更多的派生类型。通过利用整个程序优化可以扩大此分析的范围。
更常见的情况是编译器确切地知道对象的类型,例如在这两种情况下:
Derived d;
d.call_me_virtual();
Base* b = new Derived;
b->call_me_virtual();
编译器可能能够推断出 d.call_me_virtual
和 b->call_me_virtual
都将始终解析为 Derived::call_me_virtual
并且因此在该实例中去虚拟化(甚至内联)调用。
在第一种情况下,这需要知道 Derived
类型的对象不同于非常相似的 Derived&
类型的对象只能是 Derived
对象,而不是从中派生的东西。
第二种情况需要静态类型分析,这是由现代优化编译器完成的。通过查看 b
总是 使用指向 Derived
对象的指针进行初始化,编译器能够证明 将调用哪个函数>b->call_me_virtual
。
当您完成单个成员函数时,会发生类似的情况:
struct Base { virtual void call_me_virtual() = nullptr; };
struct Derived : Base { void call_me_virtual() override final { } };
void dosomething(Derived* d) {
d->call_me_virtual();
}
虽然 d
可能指向从 Derived
派生的东西,但 call_me_virtual
方法可能不会再更改,所以编译器知道 Derived::call_me_virtual
将始终被调用,因此可以将此调用去虚拟化。
关于C++:使用指向派生类的指针进行虚函数调用仍然有一个vlookup,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23517744/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!