【深度探索C++对象模型读书笔记】【第4章】Function语意学

2023-10-18 05:38

本文主要是介绍【深度探索C++对象模型读书笔记】【第4章】Function语意学,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1、 C++支持三种类型的member functions,static、nonstatic和virtual,每一种类型被调用的方式都不相同。

2、C++的设计准则之一便是nonstatic member function至少必须和一般的nonmember function有相同的效率。编译器内部会将member函数实例转换为对等的nonmember函数实例,转换步骤为:

a)  改写函数的signature以安插一个额外的参数this指针member function中,使class object可以调用该函数。其中,thisconst指针,若该函数为const,则this指向的data也为const

b)  将每一个对nonstatic data member的存取操作改为经由this指针来存取;

c)  member function重新写成一个外部函数,将函数名称进行mangling处理;

此后,每一个函数调用操作也都必须转换,用以提供相应的实参。

3、 虚拟成员函数的转化步骤:

如果normalize()是一个virtual member function,那么以下的调用:

Ptr->normalize();将会被内部转化为:(*ptr->vptr[1])(ptr);

其中:vptr表示由编译器产生的指针,指向virtual table。它被安插在每一个“声明有(或继承自)一个或多个virtual functions”的class object中。事实上其名称也会被“mangled”,因为一个复杂的class派生体系中,可能存在有多个vptrs。1virtual table slot的索引值,关联到normalize()函数。第二个ptr表示this指针。

4、static member functions的主要特征是它没有this指针。以下的次要特性统统根源于其主要特性:

a)   它不能够直接存取其class中的nonstatic members。

b)   它不能够被声明为const,volatile或virtual。

c)   它不需要经由class object才被调用——虽然大部分时候它都是这样被调用的!

5、如果取一个static memberfunction的地址,获得的将是其在内存中的位置,也就是其地址。由于static member function没有this指针,所以其地址的类型并不是一个“指向class member function的指针”,而是一个“nonmember函数指针”。

6、C++中,多态(polymorphism)表示以一个public base class指针或reference寻址出一个derived class object。识别一个class是否支持多态,唯一适当的方法试看它是否有任何virtual function。只要class拥有一个virtual function,它就需要一份额外的执行期型别判断信息。

7、 在c++中,virtual functions在编译期间获知,这一组地址是固定不变的,执行期间不能新增或替换。

8、一个class只会有一个virtual table,其中内含对应class object中所有的active virtual functions函数实体的地址。这些active virtual functions包括:

a)  一个class定义的函数实体。它会改写(overriding)一个可能存在的base class virtual function函数实例。

b)  继承自base class的函数实例。这是在该class不改写base class virtual function时才会出现的情况。

c)  一个pure_virtual_called()函数实例,它既可以扮演pure virtualfunction的空间保卫者,也可以当作执行期异常处理函数。如果该函数被调用,通常的操作是结束程序。

9、 在多重继承中支持virtual function,其复杂度围绕在第二个及后继base class上,以及必须在执行期调整this指针。

10、     thunk是一小段assemby代码,用来1)以适当的offset值调整this指针 2)跳到virtualfunction去。

11、     在多重继承下,一个上层basse classes数目为n的derivedclass内含n-1个额外的virtual tables。其主要实例与最左端的base class共享;n-1个次要实例与其它base classes有关。

12、     第二或后继的base class会影响对virtual function支持的3种情况: 

a)  通过指向第二个base class的指针,调用derived class virtual function

b)  通过指向derived class的指针,调用第二个base class中一个继承而来的virtual function

c)  允许virtual function函数的返回值类型有所变化,可能是base type,也可能是publicly derived type

13、     nonmember、static member或nonstatic member函数都能被转化为完全相同的形式,所以三者效率完全相同。

14、     取一个nonstatic member function的地址,如果该函数是nonvirtual,得到的结果是它在内存中真正的地址,但需要绑定于某个class object的地址上,才能够调用该函数。对一个virtual memberfunction取其地址,获得的只是一个索引值。

15、     关键词inline只是一项请求。如果函数的执行成本比一般的函数调用及返回机制所带来的负荷低,那么该请求被接受,编译器就用一个表达式合理地将函数扩展开来。

16、     一般而言处理一个inline函数,有两个阶段:

a)  分析函数定义,以决定函数的“instrinsic inline ability”

如果函数因其复杂度或其构建问题,被判为不可成为inline,它会被转为一个static函数,并在“被编译模块”内产生对应的函数定义。

b)  真正的inline函数扩展操作是在函数调用的那一点上。

在inline扩展期间,每一个形式参数会被对应的实际参数所取代;

inline函数中的每一个局部变量都必须被放在函数调用的一个封闭区段中,并拥有一个独一无二的名称。这会带来参数的求值操作以及临时性对象的管理。


这篇关于【深度探索C++对象模型读书笔记】【第4章】Function语意学的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/230523

相关文章

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

Python中判断对象是否为空的方法

《Python中判断对象是否为空的方法》在Python开发中,判断对象是否为“空”是高频操作,但看似简单的需求却暗藏玄机,从None到空容器,从零值到自定义对象的“假值”状态,不同场景下的“空”需要精... 目录一、python中的“空”值体系二、精准判定方法对比三、常见误区解析四、进阶处理技巧五、性能优化

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve

如何高效移除C++关联容器中的元素

《如何高效移除C++关联容器中的元素》关联容器和顺序容器有着很大不同,关联容器中的元素是按照关键字来保存和访问的,而顺序容器中的元素是按它们在容器中的位置来顺序保存和访问的,本文介绍了如何高效移除C+... 目录一、简介二、移除给定位置的元素三、移除与特定键值等价的元素四、移除满足特android定条件的元

Python获取C++中返回的char*字段的两种思路

《Python获取C++中返回的char*字段的两种思路》有时候需要获取C++函数中返回来的不定长的char*字符串,本文小编为大家找到了两种解决问题的思路,感兴趣的小伙伴可以跟随小编一起学习一下... 有时候需要获取C++函数中返回来的不定长的char*字符串,目前我找到两种解决问题的思路,具体实现如下:

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序

Spring Security基于数据库的ABAC属性权限模型实战开发教程

《SpringSecurity基于数据库的ABAC属性权限模型实战开发教程》:本文主要介绍SpringSecurity基于数据库的ABAC属性权限模型实战开发教程,本文给大家介绍的非常详细,对大... 目录1. 前言2. 权限决策依据RBACABAC综合对比3. 数据库表结构说明4. 实战开始5. MyBA

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(