热门标签 | 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



推荐阅读
  • 使用LINQ优化WinForms中CheckedListBox的操作
    本文介绍如何利用LINQ扩展方法解决WinForms中CheckedListBox控件的常见操作问题,包括获取选中项的文本并用分隔符连接,以及根据字符串初始化选中状态。 ... [详细]
  • 本文详细介绍如何在Android模拟器上安装TaintDroid的过程,包括解决源代码链接失效及服务器文件变动等问题,旨在帮助后续用户避免不必要的麻烦。 ... [详细]
  • 首先说一下,这是我在CSDN上的第一个文章,其实这个账号早在几年前就申请了,不过当时只是为了下载一个资源,而且也不怎么懂信息技术相关的领域,后来就再也没怎么动过,直到今天我才开始使用这个账号 ... [详细]
  • Spring框架中的关键配置文件详解
    本文详细介绍了Spring项目中常见的配置文件,包括pom.xml和spring.xml的作用与使用方法。pom.xml用于管理项目依赖,而spring.xml则负责Bean的定义与初始化。 ... [详细]
  • CGroups: 资源管理和控制
    CGroups(Control Groups)是Linux内核提供的一个功能,旨在限制、记录和隔离进程组使用的物理资源,如CPU、内存和I/O等。它通过精细的资源管理,支持现代容器技术如Docker的资源限制需求。 ... [详细]
  • OpenGL 实现骨骼动画平滑过渡技巧
    本文深入探讨了如何使用 OpenGL 实现骨骼动画之间的平滑过渡效果,重点介绍了动画数据的管理及混合算法的具体实现。 ... [详细]
  • 使用Inno Setup将EXE与JRE封装为Windows安装程序
    本文详细介绍了如何利用Inno Setup工具将EXE文件及Java运行环境(JRE)整合为适用于Windows操作系统的安装程序。我们将提供必要的软件下载链接,并逐步指导您完成整个打包过程。 ... [详细]
  • 手把手教你构建简易JSON解析器
    本文将带你深入了解JSON解析器的构建过程,通过实践掌握JSON解析的基本原理。适合所有对数据解析感兴趣的开发者。 ... [详细]
  • 本文深入探讨了Java注解的基本概念及其在现代Java开发中的应用。文章不仅介绍了如何创建和使用自定义注解,还详细讲解了如何利用反射机制解析注解,以及Java内建注解的使用场景。 ... [详细]
  • 本文详细介绍了 C# 编程语言中 Main 方法的作用、不同形式及其使用场景,帮助开发者更好地理解和应用这一重要概念。 ... [详细]
  • Smali代码动态调试指南
    本文介绍了如何通过一系列具体步骤实现Smali代码的动态调试,包括环境搭建、工具安装及调试过程中的关键操作。 ... [详细]
  • 深入浅出:Java面向对象编程
    本文详细介绍了Java语言的核心特性——面向对象编程。探讨了Java的基本概念、平台无关性、丰富的内置类库及安全性,同时深入解析了类加载器、垃圾回收机制以及基本数据类型和其包装类。 ... [详细]
  • 本文介绍如何利用C语言在Linux操作系统中实现递归创建多级目录的功能,包括必要的头文件引入和函数实现。 ... [详细]
  • 探讨了一个关于Windows C++开发中遇到的乱码问题,特别是在处理宽字符时出现的情况。本文通过一个具体的示例——一个简单的窗口应用程序,展示了如何正确地使用宽字符以避免乱码。 ... [详细]
  • 本文旨在分享将Hadoop集群从Windows环境迁移到Linux环境过程中遇到的技术难题及其解决方案,以帮助同行或未来的学习者避免类似问题。 ... [详细]
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社区 版权所有