- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我在 Linux 上使用 GCC。
我希望了解有效的虚拟函数。
我应该编写什么样的 C++ 代码来查看和理解使用和不使用虚拟函数时静态和动态绑定(bind)是如何发生的?
以及如何“看到”它们最终是如何绑定(bind)的,以及在这个过程中到底发生了什么?
最佳答案
这是一个例子。您可以构建和运行此代码,并将函数设置为虚函数或不虚函数。要获得虚拟行为、动态调度、动态绑定(bind),请使用预处理器宏构建它 IS_VIRTUAL
定义。要查看静态绑定(bind),请在不定义该宏的情况下构建它。
#include <iostream>
#if defined(IS_VIRTUAL)
#define CONDITIONAL_VIRTUAL virtual
#else
#define CONDITIONAL_VIRTUAL
#endif
struct A {
CONDITIONAL_VIRTUAL void foo() { std::cout << "A\n"; }
};
struct B : A {
CONDITIONAL_VIRTUAL void foo() { std::cout << "B\n"; }
};
// global objects
A a; B b;
enum object_type { get_A, get_B };
A *get_object(object_type t) {
switch (t) {
case get_A: return &a;
case get_B: return &b;
}
}
int main() {
std::cout << "Choose A or B: ";
char c;
std::cin >> c;
A *x = get_object( c == 'A' ? get_A : get_B );
x->foo();
}
绑定(bind)与 x->foo()
的评估相关.编译器必须弄清楚要为该表达式执行什么代码。对于静态和动态绑定(bind),编译器查看 x
并看到它的类型是 A*
, 所以它会查看 struct A
并寻找 foo()
声明。
对于静态绑定(bind),编译器发现 foo()
不是 virtual
,因此编译器会继续生成调用 foo()
的代码方法。简单。
使用动态绑定(bind),编译器会看到标记为 virtual
的方法,因此编译器生成的代码将在运行时使用与对象 x
关联的函数指针表。选择要调用的方法,然后调用找到的任何方法。编译器还在别处生成代码来为全局 a
创建表。和 b
对象。对于全局 a
对象它使表指向A::foo()
, 以及全局 b
它使表指向 B::foo()
.所以如果x
指向 b
对象,那么表查找将导致 B::foo()
这就是将要调用的函数。
一般来说,编译器必须确保所有具有虚方法的对象也有一个表,指向要调用的正确函数,以便任何时候对对象进行虚调用,程序可以,在运行时,获取与对象关联的表并查找要调用的正确方法。
因此,在静态和动态模式下构建上述程序,然后运行它并观察每个输入获得的输出。在下表中填写每种输入和绑定(bind)类型组合的输出。
Binding | static dynamic
Input
-----
A ? ?
B ? ?
在所有情况下,输出都是由相同的 x->foo()
的评估产生的方法调用。在哪些情况下动态约束证据?这是否符合您对上述动态绑定(bind)解释的理解?
关于c++ - 我应该怎么做才能看到实际的静态和动态绑定(bind)? [C++],我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26706605/
stackoverflow如何得知JavaScript无法正常工作,并能够在页面顶部通知用户? 是否可以判断脚本是否未加载或产生错误,然后仍然能够使用JavaScript生成错误消息? 最佳答案
Xcode 项目方案共享...如何做到这一点? 当我将 Xcode 项目提交到 SVN 时,我发现我创建的方案保存在我的用户名文件夹下,例如 abc.xcodeproj/xcuserdata/_my_
我有这个 SQL: DROP TABLE MISSINGTABLE; CREATE TABLE MISSINGTABLE ( TABLE_NAME VARCHAR2 (70), DESCRIP
我是PHP OOP的初学者,并对使用PHP处理错误的正确方法有些疑问。 例如看这个功能: public function deleteFileFromDisk($fileNameToBeDeleted
YouTube上有很多视频,我们希望为网站访问者提供自动的YouTube成绩单对齐方式。我们想要的几乎就像this example。 但是,我们希望逐个句子对齐,而不是逐个单词自动对齐,例如this
假设,我有 IAsynchronous 接口(interface),它支持两种执行某些操作的方法(“开始/结束”模式): IAsyncResult BeginOperation(AsyncCallba
Hardware Product 1 Product 2 Product 3 Product 4 Product 5 我有这样的结构,我想做一个重新排序界面,用户可以通过单击向上箭头在层次结构中向
假设,我有 IAsynchronous 接口(interface),它支持两种执行某些操作的方法(“开始/结束”模式): IAsyncResult BeginOperation(AsyncCallba
我正在使用 Silverlight 2.0 Unleashed + Silverlight 4.0 Unleashed 学习 Silverlight,好吧,我只是在玩弄它:-) 作为其中的一部分,我正
有人可以解释一下我还是链接-我有512x512图标,但我不知道我需要创建什么图标大小以及如何将它们添加到我的iOS应用中。我需要什么尺寸以及如何添加尺寸? 最佳答案 简而言之:非视网膜iPhone或i
我想在 Java 中模拟以下情况,但我陷入困境: 特别是与客户、预订、航类预订和巴士预订相关的部分。我想要一组客户对象,每个对象都有自己的预订列表(可以是航类预订或巴士预订)。 我计划像下面的测试代码
我在 opencv、Pillow、ImageMagick、subprocess 和 ffmpeg 之间摇摆,作为操作图形数据的一种方式。 ImageMagick 看起来不错而且功能相当强大,但我在 W
我想做类似的事情 SELECT t.subtitle FROM temp t LEFT JOIN ep e ON e.subtitle=t.subtitle AND e.epi
Frame[] test = new Frame[3] {{2,5},{3,6},{4,7}}; 数组初始化器只能用在变量或字段初始化器中。尝试改用新表达式。 这怎么可能? 最佳答案 这里的问题是文字
我不知道如何正确创建第一个返回。它会像这样工作,但问题是 searchtestarrayone 总是有不同的长度,而且它可能非常大。几周前开始了我的 Swift 之旅,所以下面的代码中可能有一些愚蠢的
我有这样的表: NameSteve Martin PhoneXXX Bank account654861/46147 我对表格的相同部
我有一个关于单选按钮的快速问题,以及当用户返回页面时如何设置它们。我现在想要的是能够在他们返回页面时显示所选项目。同一组中有几个,所以我不能使用 getElementByID(遗憾!)。 这是我的 H
我做了一些事情: class Tuple1 { private T1 a; private T2 b; public Tuple1(T1 a, T2 b) {
我目前正在研究我在大学的期末项目,它看起来像 instagram。在 instagram android 应用程序中,您可以点击并按住图像和 boom,显示弹出窗口。但我不知道该怎么做! 最佳答案 您
我正在使用来自 mourner/suncalc 的函数这让我可以得到我们太阳的当前位置。使用 getPosition(),我想在图像上或使用纯 CSS(当然可以缩放到不同的屏幕分辨率和方向)创建动画,
我是一名优秀的程序员,十分优秀!