如何将虚函数表里的地址强制转化成函数指针?

zhang863211 2017-06-13 10:06:36
看到网上讲解虚函数表的博客,有点困惑就想自己试验一下,结果老是报错,粘贴别人的代码也是一样报错。我的环境是win10、vs2015.有没有哪位知道原因的,谢谢!!!
代码如下:

#include <iostream>

using namespace std;

class Base
{
public:
virtual void f() { cout << "f" << endl; }
virtual void g() { cout << "g" << endl; }

private:

};


int main()
{
typedef void(*Fun)(void);

Base b;

Fun pFun = NULL;

pFun = (Fun)*((long*)*(long*)(&b));
pFun();

system("pause");
return 0;
}
...全文
266 9 打赏 收藏 转发到动态 举报
写回复
用AI写文章
9 条回复
切换为时间正序
请发表友善的回复…
发表回复
赵4老师 2017-06-14
  • 打赏
  • 举报
回复
《深度探索C++对象模型》 《C++反汇编与逆向分析技术揭秘》
  • 打赏
  • 举报
回复
http://blog.csdn.net/what951006/article/details/71202443
zhang863211 2017-06-14
  • 打赏
  • 举报
回复
引用 8 楼 zhao4zhong1 的回复:
《深度探索C++对象模型》 《C++反汇编与逆向分析技术揭秘》
谢谢!
paschen 版主 2017-06-13
  • 打赏
  • 举报
回复
指向成员函数的指针与普通的函数指针是不同的,前者实际还会有一个参数为对象的地址
ztenv 版主 2017-06-13
  • 打赏
  • 举报
回复
#include <cstring>
#include <iostream>

using namespace std;

class Base
{
public:
    Base():i(100)
    {
        memset(j,'a',10);
        j[9]=0;
    }
    virtual ~Base()
    {
    }
    virtual void f()
    {
        cout <<i<< "f:"<<j << endl;
        j[8]='b';
    }
    virtual void g()
    {
        cout <<i<< "g:"<<j << endl;
    }
private:
    int i;
    char j[10];

};


int main()
{
    typedef void(*Fun)(Base *);

    Base b;

    Fun pFun = NULL;

    pFun = (Fun)*(((long*)*(long*)(&b))+2);//invoke f()
    pFun(&b);

    pFun = (Fun)*(((long*)*(long*)(&b))+3);//invoke g();
    pFun(&b);

    return 0;
}
Base类中的成员函数采用的是thiscall的调用约定,默认都要向f()、g()传一个this指针;具体地去搜一下thiscall吧; 上面的代码我在linux 使用g++4.8.5编译通过,运行正常
paschen 版主 2017-06-13
  • 打赏
  • 举报
回复
引用 1 楼 paschen 的回复:
指向成员函数的指针与普通的函数指针是不同的,前者实际还会有一个参数为对象的地址
http://blog.csdn.net/yue7603835/article/details/45054539
ztenv 版主 2017-06-13
  • 打赏
  • 举报
回复
这种代码本身就是有问题的,你访问一下类的数据成员就知道了
赵4老师 2017-06-13
  • 打赏
  • 举报
回复
//char (*(*x[3])())[5];//x是什么类型的变量?
//
//分析C语言声明,关键是搞清楚这个变量是个什么东西(函数、指针、数组),
//是函数那么剩下的就是他的参数和返回值,
//是指针那剩下部分是说明他指向什么,
//是数组剩下的部分就是说明数组的成员是什么类型。
//解析C语言声明规则:
//从左侧第一个标识符开始,按照优先级进行结合。*表示是..的指针,const表示只读的,volatile表示可变的,[]表示是数组,()表示是函数。
//
//x和[3]结合说明是一个大小为3的数组,该数组的每个元素为一类指针,该类指针指向一类函数,该类函数无参数,返回一类指针,该类指针指向一个大小为5的char型数组
#include <stdio.h>
#include <typeinfo.h>
char num[5];
char (*x00())[5] {
    return #
}
int main() {
    char (*(*x[3])())[5];//是个数组,大小为3
    char (*(*x0  )())[5];//数组的元素,是个函数指针
    char (*( x00 )())[5];//函数原型,参数为空,返回值为指针
    char (*  x000   )[5];//返回值

    x0 = x00;
    x[0] = x0;
    x[1] = x0;
    x[2] = x0;
    printf("typeid(x).name() is %s\n",typeid(x).name());
    return 0;
}
//typeid(x).name() is char (* (__cdecl**)(void))[5]
zhang863211 2017-06-13
  • 打赏
  • 举报
回复
引用 1 楼 paschen 的回复:
指向成员函数的指针与普通的函数指针是不同的,前者实际还会有一个参数为对象的地址
那这个应该怎么写呢?我看别人都是这样写的。。。

64,649

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧