热门标签 | HotTags
当前位置:  开发笔记 > 后端 > 正文

this指针[ZT]

先说明何谓this指针:classCA{longa;voidA(longe);voidB(longe);};上面的类定义实际上如下:structs_CA
先说明何谓this指针:  
class CA  
{  
  long a;  
  void A(long e);  
  void B(long e);  
};  
上面的类定义实际上如下:  
struct s_CA  
{  
  long a;  
};  
void CA_A(s_CA *this, long e);  
void CA_B(s_CA *this, long e);  
   
因此:  
CA a;         // 此时实际为:s_CA a; CA_CA(&a);     //   调用构造函数  
a.A(34);     // 此时实际如下:CA_A(&a, 34);  
   
因此所谓的this指针就是struct s_CA的一个指针,而所谓的类实例其实就是一个结构的实例。而当出现虚函数时,假设如下:  
接口IA为:                                                         接口IB为:  
struct IA                                                           struct IB  
{                                                                      {  
  void A(long e) = 0;                                       void B(long e) = 0;  
};                                                                     };  
而CA派生自IA和IB,即实现IA和IB(用ATL的方式),此时s_CA变成如下: 
struct   s_CA  
{  
  void *pvt_A;     // 由于IA所引入的虚函数地址指针,也就是所谓的一份函数拷贝  
  void *pvt_B;     // 由于IB所引入的虚函数地址指针,也就是所谓的一份函数拷贝  
  long a;  
};  
   
此时:  
CA a;  
IA *pA = static_cast(&a);     // 此时pA == &a  
IB *pB = static_cast(&a);     // 此时pB == &a + 4,跳过pvt_A,由static_cast这个指令进行转换的  
pA->A(34);     // 此时实际为(*pA->pvt_A[0])(pA, 34);  
pB->B(34);     // 此时实际为(*pB->pvt_B[0])(pB, 34);  
而pA->pvt_A[0]的函数原型为void CA_A(IA *this, long e);  
而pB->pvt_B[0]的函数原形为void CA_B(IB *this, long e);  
   
因此就有函数CA_A和CA_B,而CA_A的this指针是IA*,CA_B的this指针是IB*,即调用接口函数时传递的this指针就是那个接口的指针,但是数据是怎么传递的呢?即编译器如何知道IA*加上8个字节后就是成员变量a的值?  
   
很简单,因为CA_A和CA_B是类CA实现的,因此编译器在编译CA_A时知道此时传进来的this+8就是成员变量a的地址,在CA_B时知道此时传进来的this+4就是成员变量a的地址,因为编译器在编译这两个函数时是知道类CA的定义的,即编译器知道如下: 
class CA : public IA, public IB  
{  
  long a;  
  void A(long e); 
  void B(long e);  
};  
所以编译器知道传进来的this到底是个什么类的实例,因而可以表现出通过传递虚函数列表的指针就达到了传递数据成员的现象。

转:https://www.cnblogs.com/mixiyou/archive/2009/08/16/1547342.html



推荐阅读
  • 深入理解Java泛型:JDK 5的新特性
    本文详细介绍了Java泛型的概念及其在JDK 5中的应用,通过具体代码示例解释了泛型的引入、作用和优势。同时,探讨了泛型类、泛型方法和泛型接口的实现,并深入讲解了通配符的使用。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 本文介绍如何在Java项目中使用Log4j库进行日志记录。我们将详细说明Log4j库的引入、配置及简单应用,帮助开发者快速上手。 ... [详细]
  • 深入探讨CPU虚拟化与KVM内存管理
    本文详细介绍了现代服务器架构中的CPU虚拟化技术,包括SMP、NUMA和MPP三种多处理器结构,并深入探讨了KVM的内存虚拟化机制。通过对比不同架构的特点和应用场景,帮助读者理解如何选择最适合的架构以优化性能。 ... [详细]
  • 本题通过将每个矩形视为一个节点,根据其相对位置构建拓扑图,并利用深度优先搜索(DFS)或状态压缩动态规划(DP)求解最小涂色次数。本文详细解析了该问题的建模思路与算法实现。 ... [详细]
  • 火星商店问题:线段树分治与持久化Trie树的应用
    本题涉及编号为1至n的火星商店,每个商店有一个永久商品价值v。操作包括每天在指定商店增加一个新商品,以及查询某段时间内某些商店中所有商品(含永久商品)与给定密码值的最大异或结果。通过线段树分治和持久化Trie树来高效解决此问题。 ... [详细]
  • Java 中的 BigDecimal pow()方法,示例 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 使用 Azure Service Principal 和 Microsoft Graph API 获取 AAD 用户列表
    本文介绍了一段通用代码示例,该代码不仅能够操作 Azure Active Directory (AAD),还可以通过 Azure Service Principal 的授权访问和管理 Azure 订阅资源。Azure 的架构可以分为两个层级:AAD 和 Subscription。 ... [详细]
  • 在Ubuntu 16.04 LTS上配置Qt Creator开发环境
    本文详细介绍了如何在Ubuntu 16.04 LTS系统中安装和配置Qt Creator,涵盖了从下载到安装的全过程,并提供了常见问题的解决方案。 ... [详细]
  • andr ... [详细]
  • 并发编程:深入理解设计原理与优化
    本文探讨了并发编程中的关键设计原则,特别是Java内存模型(JMM)的happens-before规则及其对多线程编程的影响。文章详细介绍了DCL双重检查锁定模式的问题及解决方案,并总结了不同处理器和内存模型之间的关系,旨在为程序员提供更深入的理解和最佳实践。 ... [详细]
  • 本次考试于2016年10月25日上午7:50至11:15举行,主要涉及数学专题,特别是斐波那契数列的性质及其在编程中的应用。本文将详细解析考试中的题目,并提供解题思路和代码实现。 ... [详细]
  • 作者:守望者1028链接:https:www.nowcoder.comdiscuss55353来源:牛客网面试高频题:校招过程中参考过牛客诸位大佬的面经,但是具体哪一块是参考谁的我 ... [详细]
author-avatar
重庆车管所
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有