实战C++对象模型之成员函数调用
先说结论:C++的类成员函数和C函数实质是一样的,只是C++类成员函数多了隐藏参数this。
通过本文的演示,可以看见这背后的一切,完全可C函数方式调用C++类普通成员函数和C++类虚拟成员函数。
为了实现C函数方式调用C++类成员函数,准备两个文件:。
1) 被调用的C++类成员函数源代码文件aaa.cpp
|
#include <stdio.h> // fprintf class X { public: void xxx(); private: int m; int n; }; void X::xxx() // bbb.cpp完全以C函数方式调用类X的成员函数xxx { printf("m=%d, n=%d\n", m, n); } |
把aaa.cpp编译成共享库:
|
$ g++ -g -o libaaa.so aaa.cpp -fPIC -shared |
2) 调用的C++类成员函数源代码文件bbb.cpp
|
#include <dlfcn.h> // dlopen #include <libgen.h> // basename #include <stdio.h> // fprintf #include <stdlib.h> // exit #include <string.h> // strdup struct X // 对应于aaa.cpp中类X { int m; int n; }; // 定义C风格函数指针XXX,使用前让它指向类X的成员函数xxx typedef void (*XXX)(struct X*); // 参数实为aaa.cpp中类X的this指针 // 需要指定一个命令行参数argv[1], // 值为aaa.cpp中类X的成员函数xxx的名字, // 因为C++编译器会对类X的成员函数xxx名字编码,所以实际名字不会是xxx, // 本文测试环境xxx编码后的名为_ZN1X3xxxEv, // 不同环境可能有区别,特别是不同编译器通常不同,因为C++标准未对这个做规范。 int main(int argc, char* argv[]) { char* prog = strdup(argv[0]); if (argc != 2) { fprintf(stderr, "Usage: %s symbol-name\n", basename(prog)); exit(1); } // 为测试方便,两个文件放在同一目录下,省去设置LD_LIBRARY_PATH const char* so = "./libaaa.so"; void* h = dlopen(so, RTLD_NOW); // 加载类X所在共享库文件 if (NULL == h) { fprintf(stderr, "dlopen %s failed: %s\n", so, dlerror()); exit(1); } XXX xxx = (XXX)dlsym(h, argv[1]); // 取和类X的类成员函数xxx的函数地址,以便可以调用它 if (NULL == xxx) { fprintf(stderr, "dlsym %s failed: %s\n", argv[1], dlerror()); exit(1); } // 第1组测试数据 struct X x1; x1.m = 19; x1.n = 18; (*xxx)(&x1); // 这里完全以C函数方式调用类X的类成员函数xxx // 第2组测试数据 struct X x2; x2.m = 2019; x2.n = 2018; (*xxx)(&x2); // 这里完全以C函数方式调用类X的类成员函数xxx // 第3组测试数据 x2.m = 29; x2.n = 28; (*xxx)(&x2); // 这里完全以C函数方式调用类X的类成员函数xxx return 0; } |
把bbb.cpp编译成可执行程序:
|
$ g++ -g -o bbb bbb.cpp -ldl |
执行bbb,看看效果,运行结果和预计完全一致:
|
$ ./bbb _ZN1X3xxxEv m=19, n=18 m=2019, n=2018 m=29, n=28 |
以更优雅方式运行:
|
$ ./bbb `nm libaaa.so | awk /xxx/'{print $3}'` m=19, n=18 m=2019, n=2018 m=29, n=28 |
对于类虚拟成员函数,做法是一样的,只是bbb.cpp中的struct X定义得改一下,因为有虚拟函数的类的头一个指针大小为指向虚拟函数表的指针。
包含虚拟函数的aaa.cpp:
|
#include <stdio.h> // fprintf class X { public: virtual void xxx(); // 编码后的函数名和是否为虚拟函数无关 private: int m; int n; }; void X::xxx() // bbb.cpp完全以C函数方式调用类X的成员函数xxx { printf("m=%d, n=%d\n", m, n); } |
bbb.cpp只需修改struct X的定义:
|
struct X // 对应于aaa.cpp中类X { void* p; // 对应aaa.cpp中类X的虚拟函数表指针 int m; int n; }; |
其它操作步骤完全相同,运行同样可得到预期的结果。
实战C++对象模型之成员函数调用的更多相关文章
- C++对象模型的那些事儿之六:成员函数调用方式
前言 C++的成员函数分为静态函数.非静态函数和虚函数三种,在本系列文章中,多处提到static和non-static不影响对象占用的内存,而虚函数需要引入虚指针,所以需要调整对象的内存布局.既然已经 ...
- 【深度探索c++对象模型】Function语义学之成员函数调用方式
非静态成员函数 c++的设计准则之一就是:非静态成员函数至少和一般的非成员函数有相同的效率.编译器内部已将member函数实体转换为对等的nonmember函数实体. 转化步骤: 1.改写函数原型以安 ...
- C++对象模型:成员变量<一>非静态成员变量
非静态成员变量,分别两种可能,要么类自定义,要么继承而来.根据<深度探索C++对象模型>的解读. class X { private: int x,y,z; }; 在这个类中,有三个私有成 ...
- 第50 课C++对象模型分析——成员变量(上)
C++对象模型,其实就是C++中的对象在内存中是如何排布的.C++中的对象包含了成员变量和成员函数,其实就是研究C++中的类对象它的成员变量和成员函数在内存中是如何排布的. 回归本质class 是一种 ...
- 类成员函数调用delete this会发生什么呢?
有如下代码 class myClass { public: myClass(){}; ~myClass(){}; void foo() { delete this; } }; int main() { ...
- C++并发类成员函数调用(练习1)
一般类成员函数开线程格式 std::thread t1(&类名::函数,&实例化对象,参数....) ||std::thread t1(std::bind(&&类名:: ...
- C++对象模型——对象成员的效率 (Object Member Efficiency)(第三章)
3.5 对象成员的效率 (Object Mem ber Efficiency) 以下某个測试,目的在測试聚合(aggregation).封装(encapsulation),以及继承(Inheritan ...
- 第50 课C++对象模型分析——成员函数(上)
类中的成员函数位于代码段中调用成员函数时对象地址作为参数隐式传递成员函数通过对象地址访问成员变量C++语法规则隐藏了对象地址的传递过程 #include<iostream> #includ ...
- Java类的成员函数调用顺序
class A { public A() { System.out.println("----------A 构造-------------"); } static void sb ...
随机推荐
- IDEA查看类继承关系及生成类关系图
1.在想要查看的类上按 Ctrl + H -> Diagrams -> Show Diagrams -> Java Class Diagrams -> Show Impleme ...
- VUE图片懒加载-vue lazyload插件的简单使用
序:vue项目时候,我们要对图片进行懒加载处理,这个开发项目中就不需要自己去写了,因为比较方便使用vue lazyload进行处理,高效率开发 一. vue lazyload插件: 插件地址:http ...
- Java序列化机制原理
Java序列化就是将一个对象转化为一个二进制表示的字节数组,通过保存或则转移这些二进制数组达到持久化的目的.要实现序列化,需要实现java.io.Serializable接口.反序列化是和序列化相 ...
- 面向的phthon2+3 的场景,Anaconda 安装+环境配置+管理
standard procedure in pyCharm for creating environment when Anaconda installed Create a conda env vi ...
- 基于LNMP的Zabbix4.0.1部署
转:http://www.safecdn.cn/monitor/2018/12/lnmp-zabbix4-0-1-install/306.htmlZabbix4.0.1部署 一 安装源和Zabb ...
- postgresql批量备份和恢复数据表
备份数据库:pg_dump -h localhost -U root demo02 > /home/arno/dumps/demo02.bak 恢复数据库:psql -h localhost - ...
- threading join用法
join():在子线程完成运行之前,这个子线程的父线程将一直被阻塞 import threading #线程import time def Beijing(n): print('Beijing tim ...
- 关于https不支持http的解决方案
由于在写md的时候截图是用的微博的图床,上传到github才发现不让在其他网站使用,所有本文只有一张图片. 刚才进行网站测试的时候,微博秀这个插件不能显示出来,一直是空白, 然后我把本地域名改成了12 ...
- MFC笔记4
1.添加图片 1)静态加载图片,直接在resourceView中控件设置就可以以实现 2)动态加载时,按照鸡啄米的教程http://www.jizhuomi.com/software/193.html ...
- logrotate-日志切割示例
logrotate是linux系统自带的工具,它可以自动对日志进行截断(或轮循).压缩以及删除旧的日志文件. 1)配置文件示例# cat /wls/wls81/bin/weblogic/wls/app ...