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

一 nostatic members functions
1 调用方式
   C++的设计原则之一就是:nonstatic member function知识和一般的nonmember function有相同的效率。因此:
    nonstatic member function被编译器内化为nonmember function的形式,且含有一个this(A *const this)指针成员。 
转化步骤如下:
① 改写函数的signature以安插一个额外的参数到member function,该参数即 this指针。
non-const nonstatic member增加:A *const this; const nonstatic member增加: const A *const this.
② 将每一个“对nonstatic data member的存取操作”改为经由this指针来存取。
③ 将member function重写成一个外部函数,对函数名称进行“name-mangling"处理,使它在程序中称为独一无二的词汇。
2 指向这类成员函数的指针
    取一个nonstatic member function的地址,如果该函数是nonvirtual,得到的结果是它在内存中真正的地址,然而这个值也是不完全,它也需要被绑定于某个class object地址上,才能够通过它调用该函数。所有的nonstatic member function 都需要对象的地址(以参数this 指出)。
二 static member function
1 调用方式
    static member function被编译器转化为nonmember function的形式,且不含this指针成员。
static member function的主要特性就是它没有this指针,以下的次要特性统统根源与其主要特性:
① 它不能直接存取其class中nonstatic members
② 它不能被声明为const,volatile或virtual。
③ 它不需要经由class object才能调用--虽然大部分时候它是这样调用的。
2 指向这类成员函数的指针
    取一个static member function地址,获得的将是其在内存中的位置,也就是地址。由于没有this指针,所以其地址的类型并不是一个指向”class member function “ 指针,而是一个”nonmember“ 函数指针。
三 virtual member functions 
调用方式:
1 virtual function的一般实现模型:① 每一个class有一个virtual table,内涵该class 之中有作用的virtual function的地址,②然后每个object有一个vptr,指向virtual table的所在。
2 ptr->z(); z()是虚函数,我们在执行期需要知道:
① ptr所指对象的真实类型,这可使我们选择正确的z()实体。(表示类型信息的字符串或数字)
② z()实体位置,以便我们能够调用它。通过虚函数指针(指向虚函数表)+该函数表格索引值。
3 virtual table中虚函数地址如何构建起来。
    在C++中,虚函数地址可以在编译时期获得,此外,这一组地址是固定不变的,执行期不可能新增或替换之。由于程序执行时,表格的大小和内容都不会改变,所以其建构和存取皆可以有编译器完全掌握,不需要执行期任何接入。
4 执行期如何找到编译器备妥的那些虚函数的地址?
 ① 为了找到表格,每一个class object 被安插一个有编译器内部产生的指针(vptr虚函数表指针),指向表格。
 ② 为了找到函数地址,每一个虚函数被指配一个表格索引值。
    这些工作都由编译器完成,执行期要做的,只是在特定的virtual table slot(记录着virtual function的地址)中激活virtual function。
ptr->z();被转换为:
(*ptr->vptr[4])(ptr);
5 多重继承下的virtual function
    在多重继承下,虚函数复杂度围绕在第二个及后继的base classes身上,以及必须在执行期调整this指针这一点。
6 虚拟继承下的virtual function
2 指向这类成员函数的指针
①对一个虚函数,其地址在编译时期是未知的,所能知道的仅是虚函数在其相关之虚函数表中的索引值。也就是说,对一个virtual member function 取地址,所获得的只是一个索引值。
②多重继承下,指向member function的指针
    为了让指向member function的指针也能支持多重继承和虚拟继承,专家设计了下面一个结构体:
struct _mptr{
    int delta;//增量,多重继承下调整this指针用
    int index;//虚函数表中某个虚函数的索引值
    union{
        ptrtofunc faddr;//nonvirtual member function地址
        int v_offset;//一个virtual(或多重继承中的第二或后继的)base class的vptr的位置。
    };
};

Function 语意学的更多相关文章

  1. 第4章 Function语意学

    第4章 Function语意学 目录 第4章 Function语意学 4.1 Member的各种调用方式 Nonstatic Member Function(非静态成员函数) virtual Memb ...

  2. 《深度探索C++对象模型》笔记——Function语意学

    member的各种调用方式 C++支持三种类型的member functions:static.nonstatic和virtual. nonstatic member functions会被编译器转换 ...

  3. 【C++对象模型】第四章 Function 语意学

    1.Member的各种调用方式 1.1 Nonstatic Member Functions 实际上编译器是将member function被内化为nonmember的形式,经过下面转化步骤: 1.给 ...

  4. inside the C++ Object model总结

    一. 关于对象 1.内联函数:能够除去函数调用的开支,每一处内联函数的调用都是代码的复制.这是一种空间换取时间的做法,若函数代码量大或者有循环的情况下,不宜内联(这件事有些编译器会自动帮你做).在类中 ...

  5. 《深度探索C++对象模型(Inside The C++ Object Model )》学习笔记

    转载:http://dsqiu.iteye.com/blog/1669614 第一章 关于对象 使用class封装之后的布局成本: class并没有增加成本,data members直接内含在每一个c ...

  6. 《深度探索C++对象模型》2

    第四章: function语意学 非静态成员函数: 名称的特殊处理: 静态成员函数由于缺乏this指针,因此差不多等于非成员函数: virtual table布局(单一继承): 单一继承下函数调用: ...

  7. [转]《深度探索C++对象模型》读书笔记[二]

    3.3 Data Member的存取1.   不管什么情况,每一个static data member只有一个实体,放在程序的data segment之中,每次程序取用static member,不管 ...

  8. 深入探索C++对象模型(四)

    Function语意学(The Semantics of Function) static member functions不可能做到的两点:(1)直接存取nonstatic数据,(2)被声明为con ...

  9. C++对象模型复习

    本文写于2017-02-24,从老账号迁移到本账号,原文地址:https://i.cnblogs.com/EditPosts.aspx?postid=6440685 一:对象模型 C++面向对象的实现 ...

随机推荐

  1. Altium Designer summer 9 布线 - 差分对布线

    差分信号系统是采用双绞线进行信号传输的,双绞线中的一条信号线传送原信号,另一条传送的是与原信号反相的信号.差分信号是为了解决信号源和负载之间没有良好的参考地连接而采用的方法,它对电子产品的干扰起到固有 ...

  2. maven下载及配置

    Maven是一个采用纯Java编写的开 源项目管理工具.Maven采用了一种被称之为project object model (POM)概念来管理项目,所有的项目配置信息都被定义在一个叫做POM.xm ...

  3. rsyslog 读取单个文件测试

    rsyslog 测试(rsyslog 必须yum 安装uat-web02:/root# rpm -qa | grep rsyslog rsyslog-8.21.0-1.el6.x86_64) //读取 ...

  4. ISAP

    跑的是比Dinic快辣. 更新:指针版.... #include<iostream> #include<cstdio> #include<cmath> #inclu ...

  5. 设计模式(九):Composite组合模式 -- 结构型模式

    1. 概述 在数据结构里面,树结构是很重要,我们可以把树的结构应用到设计模式里面. 例子1:就是多级树形菜单. 例子2:文件和文件夹目录 2.问题 我们可以使用简单的对象组合成复杂的对象,而这个复杂对 ...

  6. HDOJ(HDU) 1562 Guess the number(水题,枚举就行)

    Problem Description Happy new year to everybody! Now, I want you to guess a minimum number x betwwn ...

  7. Ubuntu中apt-get出现E:Encountered a section with no Package: header……的解决方案

    方法一:运行命令apt-get update更新list列表 方法二:将/var/lib/apt/lists/下的所有list文件都删除,然后再update

  8. [Falcor] Return the data from server

    <!-- index.html --> <html> <head> <!-- Do _not_ rely on this URL in production. ...

  9. NSRunLoop个人理解

    作者: xwang 出处: http://www.cnblogs.com/xwang/  本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保 ...

  10. Tabs( 选项卡)

    一. 加载方式 //class 加载方式<div id="box" class="easyui-tabs" style="width:500px ...