ZC: C++的类函数指针 不像 Delphi的类函数指针,前者 需要规定死 是哪个类的函数的指针,后者就不需要 很灵活。

测试环境:
  Win7x64
  cn_visual_studio_2010_ultimate_x86_dvd_532347.iso
  qt-opensource-windows-x86-msvc2010_opengl-5.3.2.exe

暂时的约定(20170105)(基于现在的水平状况)
  (1)、所有可能成为函数指针的 类成员函数,去掉 编译器代码优化
  (2)、所有 类TClassFuncPtr的子类 的Call()函数 都去掉 编译器代码优化
  (3)、所有可能成为函数指针的 类成员函数,都使用 __stdcall 调用约定
  (4)、所有 类TClassFuncPtr的子类 的Call()函数 都使用 __stdcall 调用约定

1、测试代码 --> vs2010控制台程序:

  1.1、main.cpp

#include <stdio.h>
#include <windows.h> #include "ZZ.h" class TClassFuncPtr
{
public:
TClassFuncPtr()
{
FpObj = NULL;
FpFunc = NULL;
} protected:
void* FpObj; // ZC: 对象指针
void* FpFunc; // ZC: 类函数的 函数地址(统一使用stdcall调用约定,函数传参/调用的时候 方便一点) public:
void Set(void *_pObj, void *_pFunc)
{
FpObj = _pObj;
FpFunc = _pFunc;
} bool IsValid()
{
return ( (FpObj != NULL) && (FpFunc != NULL) );
}
};
// ZC: 子类 命名规则:“Tcfp_返回值类型_各个传入参数类型()” class Tcfp__ :public TClassFuncPtr
{
public:
void __stdcall Call()
{
if (IsValid())
{
void* pObj = FpObj;
void* pFunc = FpFunc;
_asm
{
push pObj // ZC: 直接用FpObj不行,push的值会是0...需要用pObj转一下
call pFunc
}
}
}
}; class Tcfp_I_IF :public TClassFuncPtr
{
public:
int __stdcall Call(int _i, float _f)
{
if (IsValid())
{
void* pObj = FpObj;
void* pFunc = FpFunc;
int iFloat = ;
memcpy(&iFloat, &_f, );
_asm
{
push iFloat //ZC: 直接“push _f”不行,需要转一下
push _i
push pObj
call pFunc
}
}
}
}; class Tcfp_CR_IF :public TClassFuncPtr
{
public:
CRtn __stdcall Tcfp_CR_IF::Call(int _i, int _j);
}; #pragma optimize( "", off )
CRtn __stdcall Tcfp_CR_IF::Call(int _i, int _j)
{
if (IsValid())
{
_asm
{
push _j
push _i
mov eax,dword ptr [ebp+0xC]
push eax
mov eax,[this] // ZC: 貌似和语句“mov eax,this”是一样的效果...
// [eax] ==> FpObj
// [eax+4] ==> FpFunc
push [eax]
call [eax+]
}
}
}
#pragma optimize( "", on ) // ZC: 获取 类函数指针(地址)
template<typename dst_type,typename src_type>
dst_type pointer_cast(src_type src)
{
return *static_cast<dst_type*>(static_cast<void*>(&src));
} //#pragma optimize( "gs", off ) void main()
{
Z01(, );
Z02(, );
Z03(, ); // *** void* p01 = pointer_cast<void*>(&CA::ReturnObj01);
void* p02 = pointer_cast<void*>(&CA::ReturnObj02);
void* p03 = pointer_cast<void*>(&CA::ReturnObj03);
printf("p01 : %08X\n", p01);
printf("p02 : %08X\n", p02);
printf("p03 : %08X\n", p03); CA a;
a.i = ;
printf("&a : %08X\n", &a); void* pA = pointer_cast<void*>(&CA::A);
Tcfp_I_IF cfpA;
cfpA.Set(&a, pA);
int iRtn = cfpA.Call(, );
printf("iRtn : %08X, %d\n", iRtn, iRtn); // *** *** *** CRtn cr;
printf("&cr : %08X, %d\n", &cr, &cr);
//#pragma OPTIMIZE OFF
cr = a.ReturnObj01(, ); // ZC: 看汇编可以见到,&cr 并不等于 传入的 CRtn指针 的值... 只有在需要用到cr的时候 才会对cr进行设置...
//#pragma OPTIMIZE ON
//cr.Fi = 3;
printf("&cr : %08X, %d\n", &cr, &cr); cr = a.ReturnObj02(, );
//#pragma OPTIMIZE ON
printf("&cr : %08X, %d\n", &cr, &cr); Tcfp_CR_IF cfp01;
cfp01.Set(&a, p03);
CRtn cr01;
printf("&cr01 : %08X, %d\n", &cr01, &cr01);
printf("&cfp01 : %08X, %d\n", &cfp01, &cfp01);
cr01 = cfp01.Call(, );
printf("cr01.Fi : %08X, %d\n", cr01.Fi, cr01.Fi); system("pause");
}

  1.2、ZZ.h

#ifndef ZZZ
#define ZZZ //#include <stdio.h>
//#include <windows.h> void Z01(int _i, int _j);
void Z02(int _i, int _j);
void Z03(int _i, int _j); class CRtn
{
public:
int Fi;
}; class CA
{
public:
int i;
int __stdcall A(int _i, float _j)
{
int ii = _i + _j * i;
return ii * i;
} int __stdcall B(int _i, double _j)
{
int ii = _i + _j * i;
return ii * i;
} CRtn __stdcall ReturnObj03(int _i, int _j)
{
CRtn *pRtn = new CRtn();
pRtn->Fi = _i+ _j;
return (*pRtn);
} // ZC: 这个函数的实现,写在了cpp文件里面,并且使用了编译开关 ==> 它的代码没有被优化
CRtn __stdcall ReturnObj01(int _i, int _j); // ZC: 函数ReturnObj02(...) 虽然加了编译开关,但是看汇编Release版 里面仍然是被优化了...
#pragma optimize( "", off )
CRtn __stdcall ReturnObj02(int _i, int _j)
{
CRtn *pRtn = new CRtn();
pRtn->Fi = ;
return (*pRtn);
}
#pragma optimize( "", on ) }; #endif // ZZZ

  1.3、ZZ.cpp

#include "ZZ.h"
#pragma optimize( "", off )
void Z01(int _i, int _j)
{
_asm
{
mov eax,eax
mov eax,eax
mov eax,eax
}
} #pragma optimize( "", on ) void Z02(int _i, int _j)
{
_asm
{
mov ebx,ebx
mov ebx,ebx
mov ebx,ebx
}
} void Z03(int _i, int _j)
{
_asm
{
mov ecx,ecx
mov ecx,ecx
mov ecx,ecx
}
} #pragma optimize( "", off )
CRtn __stdcall CA::ReturnObj01(int _i, int _j)
{
CRtn *pRtn = new CRtn();
pRtn->Fi = ;
return (*pRtn);
}
#pragma optimize( "", on )

Z

ZC_C++类函数指针_模拟_Delphi类函数指针的更多相关文章

  1. ZC_C++类函数指针_模拟_Delphi类函数指针_Qt例子

    qt-opensource-windows-x86-msvc2010_opengl-5.3.2.exe ZC: “const QString” 作传入参数的时候,不太会弄... 貌似 还是在进行构建等 ...

  2. C++第四篇--重载_指针_引用

    C++第四篇--重载_指针_引用 1. 基础知识 重载:函数名相同,根据参数不同(类型.数量.顺序不同)调用同名函数 指针和引用:引用就是别名,引用时必须初始化,引用你定义的变量. int a; in ...

  3. ca71a_c++_指向函数的指针_通过指针调用函数txwtech

    /*ca71a_c++_指向函数的指针_通过指针调用函数用typedef简化函数指针的定义简化前: bool(*pf)(const string&, const string &); ...

  4. c语言中较常见的由内存分配引起的错误_内存越界_内存未初始化_内存太小_结构体隐含指针

    1.指针没有指向一块合法的内存 定义了指针变量,但是没有为指针分配内存,即指针没有指向一块合法的内浅显的例子就不举了,这里举几个比较隐蔽的例子. 1.1结构体成员指针未初始化 struct stude ...

  5. C语言_初步了解一下指针

    指针的基本概念 在计算机中,所有的数据都是存放在存储器中的. 一般把存储器中的一个字节称为一个内存单元, 不同的数据类型所占用的内存单元数不等,如整型量占2个单元,字符量占1个单元等.为了正确地访问这 ...

  6. 零基础逆向工程24_C++_01_类_this指针_继承本质_多层继承

    1 类内的成员函数和普通函数的对比 1.1 主要是从参数传递.压栈顺序.堆栈平衡来总结. 1.参数传递:成员函数多传一个this指针 2.压栈顺序:成员函数会将this指针压栈,在函数调用取出 3.堆 ...

  7. 【c实现,vc6调试通过】给出一字符串指针,计算出字符串指针中单词数

    #include <stdio.h> /* 给出一字符串指针,计算出字符串指针中单词数, 单词不包括'.',',',';','?','_','"',由0-9数字或26个字母组成 ...

  8. Qt 智能指针学习(7种指针)

    Qt 智能指针学习 转载自:http://blog.csdn.net/dbzhang800/article/details/6403285 从内存泄露开始? 很简单的入门程序,应该比较熟悉吧 ^_^ ...

  9. 不可或缺 Windows Native (18) - C++: this 指针, 对象数组, 对象和指针, const 对象, const 指针和指向 const 对象的指针, const 对象的引用

    [源码下载] 不可或缺 Windows Native (18) - C++: this 指针, 对象数组, 对象和指针, const 对象,  const 指针和指向 const 对象的指针, con ...

随机推荐

  1. Directed Graph Loop detection and if not have, path to print all path.

    这里总结针对一个并不一定所有点都连通的general directed graph, 去判断graph里面是否有loop存在, 收到启发是因为做了[LeetCode] 207 Course Sched ...

  2. C#实现无标题栏窗体点击任务栏图标正常最小化或还原的解决方法

    对于无标题栏窗体,也就是FormBorderStyle等于System.Windows.Forms.FormBorderStyle.None的窗体,点击任务栏图标的时候,是不能象标准窗体那样最小化或还 ...

  3. html09

    1.Jquery的常用方法1)选择器2)操作节点以下的 obj 都是指 jQuery对象 1.操作样式 obj.css() :不加参数是获取节点的css样式 obj.css({"属性&quo ...

  4. php mysql_connect pmysql_connect区别

    <?php        mysql_pconnect('192.168.75.128', 'root', 'root');        mysql_close();        sleep ...

  5. redis环境搭建与配置

    通过初始化脚本启动redis 1.将redis源码的utils文件夹下面有的redis_init_script复制到/etc/init.d/redis_端口号下面. 带密码的实例 REQUIRED_P ...

  6. Linux虚拟机克隆后网卡UUID问题

    虚拟机中的Linux系统克隆后,网卡配置eth0中的UUID可被克隆的系统是一样的,这样UUID就失去了唯一性. 我参考了该篇博客: 有时我们不小心将/etc/sysconfig/network-sc ...

  7. java接口对接——别人调用我们接口获取数据

    java接口对接——别人调用我们接口获取数据,我们需要在我们系统中开发几个接口,给对方接口规范文档,包括访问我们的接口地址,以及入参名称和格式,还有我们的返回的状态的情况, 接口代码: package ...

  8. MySQL root账户密码设为“root”后执行命令提示ERROR 1820 (HY000): You must reset your password using ALTER USER statement before executing this statement.

    修改root账户密码为“root”后,提示ERROR 1820 (HY000): You must reset your password using ALTER USER statement bef ...

  9. python多进程打印字符,加锁(Lock加锁)

    先看不加锁的: #coding=utf-8from multiprocessing import Process,Lockimport time def l(num):    #lock.acquir ...

  10. (一)github之基础概念篇

    1.github: 一项为开发者提供git仓库的托管服务, 开发者间共享代码的场所.github上公开的软件源代码全都由git进行管理. 2.git: 开发者将源代码存入名为git仓库的资料库中,而g ...