C++对象模型(19)

编程入门 行业动态 更新时间:2024-10-12 03:19:12

C++对象<a href=https://www.elefans.com/category/jswz/34/1771358.html style=模型(19)"/>

C++对象模型(19)

1、普通成员函数的调用

1.1 调用方式的转换

为了提高普通成员函数的调用效率,在C++中,对普通成员函数的调用,会转换成对全局函数的调用。

假如有下面所示的成员函数:

class Test {
public:int m_i;int func(int a) {m_i += a;return m_i;}
};

在编译时会经历下面几个步骤:

(1)把对象的首地址作为参数,传递给函数。

int func(Test *const this, int a) {}

(2)对非静态成员变量的调用,改成通过this指针来调用。

int func(Test *const this, int a) {this->m_i += a;return this->m_i;     
}

(3)对函数名进行name mangling操作,通过函数名和其参数类型生成唯一标识符,来区分不同的函数。

int _ZN4Test4funcEi(Test *const this, int a){}

1.2 代码演示

(1)

在main()函数中加入调用函数func()的代码:

int main() {Test test;int ret = test.func(6);
}

然后把断点设在:int ret = test.func(6); 运行程序后,查看反汇编代码(VS2019):

简单分析下汇编语言的意思:

(1)push 6:把参数6入栈。

(2)lea ecx, [test]:把test写入ecx。

这2行其实是参数传递。

(3)call Test::func(05C12D0h):调用函数func。

(4)move dword ptr [ret], eax:把eax的值赋给ret。

从这些代码可以看到,函数的调用流程可分成3个阶段:参数传递、调用函数call、处理返回值。

在反汇编的“地址(A)”窗口输入函数func的地址:05C12D0h,回车后查看函数func的反汇编代码:

这段汇编的细节不去深究,但我们可以看到有[this]字样,可以知道函数在具体执行时用到了this指针。

所以对func函数的调用转换可以用下图表示:

如果用指针方式调用成员函数:

Test* pTest;
int ret = pTest->func(6);

则对func函数的调用转换是这样的:

(2)把代码在linux上编译后,用nm命令查看可执行文件的信息。

linux下编译:g++ ch19.cpp -o ch19

用nm命令查看可执行文件:nm ch19

可以看到,func函数被转换成了_ZN4Test4funcEi。

2、静态成员函数的调用

对于静态成员函数的调用,无论用对象名来调用,还是用对象指针来调用,效果都是一样的,都会被编译器转换成一般的针对普通函数(非成员函数)的调用形式。

静态成员函数是跟着类走的,所有调用静态成员函数时编译器是不会插入this作为形参的。

假如类Test有一个static成员函数:

static void stfunc() { }

则可以用下面的方法来调用此静态成员函数:

Test test;
test.stfunc();Test::stfunc();Test* pTest;
pTest->stfunc();

静态成员函数有以下特点:

(1)静态成员函数没有this指针,这点最重要。

(2)无法直接存取类中普通的非静态成员变量,因为非静态成员变量是通过this指针来操作的。

(3)静态成员函数不能在后面使用const,也不能设置为virtual。

(4)可以用类对象调用,但不非一定要用类对象调用。

(5)静态成员函数等同于非成员函数,需要提供回调函数的这种场合,可以将静态成员函数作为回调函数。

更多推荐

C++对象模型(19)

本文发布于:2023-12-04 21:53:27,感谢您对本站的认可!
本文链接:https://www.elefans.com/category/jswz/34/1662255.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文标签:模型   对象

发布评论

评论列表 (有 0 条评论)
草根站长

>www.elefans.com

编程频道|电子爱好者 - 技术资讯及电子产品介绍!