首页 > 代码库 > 让类成员函数指针成为可调用对象

让类成员函数指针成为可调用对象

    类成员函数指针实践上是一个指针类型,不可直接通过调用运算符()作为可调用对象调用,一般调用该类成员函数指针需要指定该指针对应的对象。

    一般情况下调用类成员函数指针:

// a.h
#ifndef A_H
#define A_H
#include <iostream>
using std::cout;
using std::endl;

class A{
public:
    void print();
};
#endif

// a.cpp
#include "a.h"
void A::print()
{
    cout << "A::print" << endl;
}

// main.cpp
#include "a.h"
using pClassF = void (A::*)(); // 声明类A的成员函数指针类型

int main()
{
    pClassF pf= &A::print; // 定义类成员函数指针,不支持函数到指针的自动转换
    A a;
    (a.*pf)(); // .*、->*成员访问符,因为访问优先级则(a.*pf)的括号必须添加
    return 0;
}

其中A::*表示是类A的成员指针,接着的()表示是无参的函数类型;

如果直接是pf()则出错,因为pf不是可调用对象其未指定对象执行;


使用<functional>的std::function<T>模板类:

因为类的成员函数执行时,会在参数列表添加参数--隐式的this实参,在function模板类调用时可以传入对象实现this的功能(传入的对象不一定是指针类型),function<T>判断如果是类成员函数指针,则会将通过该对象使用成员访问运算符,实现类成员函数指针的调用功能(具体function如何判断是类成员函数指针还是普通函数指针,本人现在不清楚,如果有理解错误地方望指正):

// main.cpp ,头文件a.h与源文件a.cpp之前相同
#include <functional>
#include "a.h"
using std::function;
using pClassF = void (A::*)(); // 声明类A的成员函数指针类型

int main()
{
    auto pf= &A::print; // 定义类成员函数指针,不支持函数到指针的自动转换
    A a;
    
    // void 表示成员函数的返回值,A表示传入的参数类型为A,因为是模板类型则要求可以准确匹配,且A类型可以调用对应的成员函数,如果是const A类要调用const成员函数
    function<void (A)> fnt = pf;
    fnt(a);
    return 0;
}

通过fnt(a)传入对象a,在function<T>里通过a与成员访问符调用成员函数。


使用std::mem_fn标准库函数<functional>:

mem_fn函数可以通过成员函数指针的类型自动推断可调用对象类型,用户无须指定。在可调用对象里有接收对象与对象指针的一组调用运算符重载函数,可使用对象或对象指针调用该成员函数,使用方式与function<T>相同:

// main.cpp,头文件a.h与源文件a.cpp之前相同
#include <functional>
#include "a.h"
using std::mem_fn;
using pClassF = void (A::*)() const; // 声明类A的成员函数指针类型

int main()
{
    auto pf= &A::print; // 定义类成员函数指针,不支持函数到指针的自动转换
    A a;
    auto fnt = mem_fn(pf); // mem_fn通过成员函数指针自动推导可调用对象类型
    fnt(a); // 使用对象调用成员函数
    fnt(&a); // 使用对象指针调用成员函数
    return 0;
}

fnt(a)与fnt(&a)的结果一致。


使用通用的函数适配器bind生成可调用对象,需要命名空间std::placeholders表示在bind传给函数的参数:

与function<T>类似,将隐式传入this形参转为显示传入对象;与mem_fn类似,生成的可调用对象有接收对象与对象指针的一组重载调用运算符函数:

// main.cpp,头文件a.h与源文件a.cpp之前相同
#include <functional>
#include "a.h"
using namespace std::placeholders; // 用于表示bind传入指定函数的形参位置,即bind的_1、_2、...、_n等
using pClassF = void (A::*)() const; // 声明类A的成员函数指针类型

int main()
{
    auto pf= &A::print; // 定义类成员函数指针,不支持函数到指针的自动转换
    A a;
    auto fnt = bind(pf,_1); // _1表示在bind该位置的参数传给pf,并成为pf的第一个形参
    fnt(a); // 使用对象调用成员函数
    fnt(&a); // 使用对象指针调用成员函数
    return 0;
}

详细说明可查阅bind函数,fnt(a)与fnt(&a)的结果一致。

本文出自 “zmh009” 博客,请务必保留此出处http://zmh009.blog.51cto.com/11619347/1886049

让类成员函数指针成为可调用对象