首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

为什么自由函数指针总是类型指针,而成员函数指针实际上不是指针?

自由函数指针总是类型指针,而成员函数指针实际上不是指针的原因是因为它们在语法和语义上有所不同。

自由函数指针是指向自由函数(非成员函数)的指针,它可以直接指向函数的入口地址,并通过该指针调用函数。由于自由函数没有所属的类,因此它的类型可以直接表示为函数的返回类型和参数列表,因此自由函数指针可以被视为函数类型的指针。在C++中,自由函数指针的类型可以使用typedef或using关键字定义,例如:

代码语言:cpp
复制
typedef void (*FreeFunctionPtr)(int);
using FreeFunctionPtr = void (*)(int);

成员函数指针是指向类的成员函数的指针,它需要额外的信息来确定函数所属的类以及函数的签名(返回类型和参数列表)。成员函数指针实际上是一个复合类型,包含了函数的入口地址以及一个隐藏的指向类的指针(this指针)。由于成员函数指针需要额外的信息来确定函数的调用方式,因此它的类型不能直接表示为函数的返回类型和参数列表,而是需要使用特殊的语法来声明,例如:

代码语言:cpp
复制
typedef void (MyClass::*MemberFunctionPtr)(int);
using MemberFunctionPtr = void (MyClass::*)(int);

需要注意的是,成员函数指针只能通过类的实例来调用,因为它需要一个有效的this指针来访问类的成员变量和其他成员函数。而自由函数指针可以直接通过函数指针调用函数,不需要依赖于类的实例。

总结起来,自由函数指针是类型指针,因为它可以直接表示为函数类型的指针,而成员函数指针不是指针,因为它是一个复合类型,包含了函数的入口地址和类的指针。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

【C++】函数指针 ④ ( 函数指针函数参数 | 使用函数指针间接调用函数 | 函数指针做参数 | 函数指针类型的本质 | 函数指针做参数意义 )

一、函数指针函数参数 1、使用函数指针间接调用函数 在上一篇博客 【C++】函数指针 ③ ( 函数指针语法 | 函数名直接调用函数 | 定义函数指针变量 | 使用 typedef 定义函数类型 | 使用...typedef 定义函数指针类型 ) 中 , 最后一个示例 , 使用 typedef 定义函数指针类型 如下 : // int (int, int) 函数指针类型重命名为 pFun_add // 该类型变量可以直接接收..., 调用的函数可以动态指定 ; 2、函数指针做参数 定义了 如下 函数指针类型 pFun_add , 其类型为 int (*)(int, int) , 该指针指向一个 类型为 int (int, int..., int); 定义函数 接收 pFun_add 类型的形参作为参数 , 该类型函数指针类型 , 也就是 函数接收一个 函数指针类型参数 , 在该函数中调用 函数指针 指向的 函数 ; // 传入函数指针...函数名 ( 函数地址 ) 作为 函数指针 参数 // 传递给函数 caculate(add, 11, 12); 3、函数指针类型的本质 函数指针类型 本质 : 提前对任务 格式 进行约定 ; 函数参数类型

70650

【C++】函数重载 ④ ( 函数指针定义的三种方式 | 直接定义函数指针 | 通过 函数类型 定义 函数指针 | 通过 函数指针类型 定义 函数指针 )

的 返回值 不是 " 函数重载 " 的 判断标准 ; 二义性 : 如果 函数重载 与 默认参数 结合使用 , 出现了二义性 , 编译直接失败 ; 函数指针赋值重载函数 : 根据 函数指针 类型中的 参数列表类型..., 定义函数指针 , 直接根据指针的定义语法 指针类型* 指针名称 定义函数指针 , 同时将 add 函数 的 地址 赋值给 函数指针 ; // 根据 函数类型 定义 函数指针 func* func1...= add; 3、通过 函数指针类型 定义 函数指针 首先 , 通过 typedef 关键字, 定义 函数指针 类型 , 类型名称为 func_ptr , 对应的函数的 参数列表是 2 个 int 参数...); 然后 , 通过 func_ptr 函数指针类型 , 定义 函数指针 , 直接使用 变量类型 变量名称 的方式定义 函数指针 ; // 根据 函数指针类型 定义 函数指针 func_ptr func2...函数类型 定义 函数指针 func* func1 = add; // 根据 函数指针类型 定义 函数指针 func_ptr func2 = add; // 直接定义 函数指针 int (*

17930
  • 【C++】函数重载 ④ ( 函数指针定义的三种方式 | 直接定义函数指针 | 通过 函数类型 定义 函数指针 | 通过 函数指针类型 定义 函数指针 )

    的 返回值 不是 " 函数重载 " 的 判断标准 ; 二义性 : 如果 函数重载 与 默认参数 结合使用 , 出现了二义性 , 编译直接失败 ; 函数指针赋值重载函数 : 根据 函数指针 类型中的 参数列表类型..., 定义函数指针 , 直接根据指针的定义语法 指针类型* 指针名称 定义函数指针 , 同时将 add 函数 的 地址 赋值给 函数指针 ; // 根据 函数类型 定义 函数指针 func* func1...= add; 3、通过 函数指针类型 定义 函数指针 首先 , 通过 typedef 关键字, 定义 函数指针 类型 , 类型名称为 func_ptr , 对应的函数的 参数列表是 2 个 int 参数...); 然后 , 通过 func_ptr 函数指针类型 , 定义 函数指针 , 直接使用 变量类型 变量名称 的方式定义 函数指针 ; // 根据 函数指针类型 定义 函数指针 func_ptr func2...函数类型 定义 函数指针 func* func1 = add; // 根据 函数指针类型 定义 函数指针 func_ptr func2 = add; // 直接定义 函数指针 int (*

    15930

    【C++】函数指针 ③ ( 函数指针语法 | 函数名直接调用函数 | 定义函数指针变量 | 使用 typedef 定义函数类型 | 使用 typedef 定义函数指针类型 )

    , int) 函数类型指针 , 将 add 函数名 ( 函数地址 ) 直接 赋值给 函数指针 , 可以直接通过函数指针调用函数 ; 下面的代码中 , 定义了函数指针 p , 指向 int(int,...int) 类型函数 , 将 add 函数地址 赋值给 函数指针 p , 之后通过 函数指针 p 调用 add 函数 ; 代码如下 : // 声明一个 int(int, int) 类型指针变量 /...func_add // 使用时需要使用该类型指针调用函数 typedef int fun_add(int, int); 使用定义的 函数类型 调用函数 : 定义函数类型指针类型 func_add...pFun_add)(int, int); 使用定义的 函数指针类型 调用函数 : 定义函数指针类型 pFun_add 的 变量 , 然后 将 add 函数地址 赋值给 该变量 , 然后 通过 函数指针变量..., int) 类型函数类型 // 该指针 指向 函数地址 int(*p)(int, int) = add; // 通过函数指针调用 函数 p(3, 4); // 定义函数指针类型变量

    43660

    【C++】函数指针 ① ( 函数三要素 | 函数类型 | 函数指针类型 | 函数类型重命名 )

    和 个数 共同决定 ; 函数类型示例 : 下面的函数类型 表示的函数 , 返回值为空 , 参数为 int 类型的 ; void (int) 3、函数指针类型 函数指针类型 是指指向函数指针变量的类型...; 每个函数都有一个入口地址 , 这个地址是一个指针 , 指向函数的代码块在内存中的位置 ; 函数指针变量就是用来存储这个入口地址的变量 , 函数指针变量的类型需要与被调用的函数类型匹配 , 即函数指针类型应该与被调用的函数的返回值类型和参数列表类型一致...; 函数指针类型示例 : 下面的函数指针类型 表示的函数 , 返回值为空 , 参数为 int 类型的 ; void (*)(int) 4、函数类型重命名 在 C 语言中 , 可以使用 typedef...; newfunname 是 函数重命名的新名称 ; parameterlist 是 函数的参数列表 ; 为 函数类型函数指针 类型 重命名 : // void (int) 函数类型重命名, 可以使用该类型指针接收函数地址..., 可以使用该类型指针接收函数地址 typedef void func(int); // void (*)(int) 函数指针类型重命名, 可以直接使用该类型变量接收函数地址 typedef void

    37350

    C语言指针函数参数,指针函数返回类型

    指针变量作函数参数可以将函数外部的地址传递到函数内部,使得在函数内部可以访问到函数外部的数据,并且这些数据不会随着函数的结束被销毁。...,因为 intArr 仅仅是一个指针不是真正的数组,所以必须要额外增加一个参数来传递数组长度。...b); C语言为什么不允许直接传递数组的所有元素,而必须传递数组指针呢?...指针函数返回类型: 程序编译后,每个函数都有执行第一条指令的地址即首地址,称[函数指针函数指针即指向函数指针变量,要间接调用函数可以使用指针变量来实现。...数组指针变量加减一个整数可使指针移动指向后面或数组元素,函数指针的移动是毫无意义的。函数调用中“(* 指针变量名)”的两边的括号不可少,其中的“*”不应该理解为求值运算,在此处只是一种表示符号。

    2.6K20

    为什么我应该使用指针不是对象本身

    我发现使用 C++ 的人经常用指针表示对象,比如像下面这样: Object *myObject = new Object; 不是, Object myObject; 或者在调用成员函数的时候,都会这样...: myObject->testFunc(); 不是, myObject.testFunc(); 我有点想不明白为什么这么做?...引用语义(reference semantics): 有的时候,你希望函数传递进来的参数不是一份副本(copy),因为创建副本的代价很大。这个时候,你就可以通过指针。...多态(polymorphic): 对于多态类型指针和引用可以避免对象被切片(slice)。...切片的意思就是说:在函数传参处理多态变量时,如果一个派生类对象在向上转换(upcast),用的是传值的方式,不是指针和引用,那么,这个派生类对象在 upcast 以后,将会被 slice 成基类对象,

    1.4K10

    【C语言笔记】函数指针作为结构体成员

    函数指针在C语言中的意义 在C语言程序中,数据结构和算法是两个基本的元素。C语言的基本数据类型、结构体、数组和联合体是数据结构的代表;C语言中的函数则是算法的代表。...函数指针的应用 函数指针在嵌入式中的应用非常广泛,常常把函数指针作为结构体的成员、作为函数的参数等。如在物联网操作系统RT-Thread内核源码中,有如下代码: 1、函数指针作为结构体成员 ?...2、函数指针作为函数的参数 举例说明 建立一个结构体,用于四则运算(根据函数指针的指向可以选择加法运算、减法运算、乘法运算、除法运算),如 typedef int (*fun_t)(int, int);...add2; result = data.operation(data.a, data.b); printf("加法运算: %d+%d = %d\n",data.a, data.b, result); 函数指针...以上就是函数指针作为结构体成员的笔记,如有错误欢迎指出!

    3.5K30

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

    成员函数指针实践上是一个指针类型,不可直接通过调用运算符()作为可调用对象调用,一般调用该类成员函数指针需要指定该指针对应的对象。     ...*pf)的括号必须添加     return 0; } 其中A::*表示是类A的成员指针,接着的()表示是无参的函数类型; 如果直接是pf()则出错,因为pf不是可调用对象其未指定对象执行; 使用: mem_fn函数可以通过成员函数指针类型自动推断可调用对象类型,用户无须指定。... // mem_fn通过成员函数指针自动推导可调用对象类型     fnt(a); // 使用对象调用成员函数     fnt(&a); // 使用对象指针调用成员函数     return 0; }...、_n等 using pClassF = void (A::*)() const; // 声明类A的成员函数指针类型 int main() {     auto pf= &A::print; // 定义类成员函数指针

    1.1K40

    VC和GCC成员函数指针实现的研究(二)

    image.png 图十:VC多重继承的虚表 (和前面不是同一批的截图,可能地址会对不上) 可以看到c和b的foo_b类型的虚表的info函数指向的地址是不同的。...跳过去后的代码是这样: image.png 图十一:VC多重继承的vcall (和前面不是同一批的截图,可能地址会对不上) 可以看到cx又减了8,把this指针由foo_b*类型转回了foo_c*。...GCC多重继承成员函数指针实现 image.png 图十二:GCC多重继承的函数指针赋值 哈,GCC的多重继承的赋值部分也和单继承一样,那么调用呢?...image.png 图十三:GCC多重继承的函数指针调用 如上图所示,比单继承多了两行,第一行是调整虚表地址到foo_b,这点和VC一样。后面增加的add指令是调整成员函数的this指针地址。...下一篇 VC和GCC成员函数指针实现的研究(三)

    68620

    VC和GCC成员函数指针实现的研究(三)

    ;如果虚函数被覆盖,则再在虚表指向的函数里多一层跳转,并把偏移加回来到子类地址;进了函数之后,变量的地址偏移就好算了。...用foo_c的指针去调用foo_b的成员函数的时候是需要对指针值做些offset修正的。 然而 获取成员函数指针成员函数调用是分开的场景。...可以把基类成员函数当成子类成员函数来取地址(&foo_c::print)。 在2发生之后,如果用子类指针去调用这个“伪”子类成员函数指针时,子类的指针值需要转换成基类的指针值。...所以必须要靠成员函数指针把这个offset存> 下来。 你可以试下把子类成员函数指针转换为基类成员函数指针,如果这个基类不是子类的第一父类,转换过程必然会导致修正这个offset值。...如果非虚非静态成员函数的类签名和指针类型的类签名不一样,并且这两个类型转换的时候地址会变化(不是第一个基类)的时候。要有offset来记录this的偏移。这上面的sample都过于简单了。

    81010

    VC和GCC内成员函数指针实现的研究(一)

    那么,VC和GCC是怎么实现这一功能的呢 VC单继承成员函数指针实现 image.png 图一: VC指针赋值 可以看到,对非虚函数指针,直接把函数地址赋值过去了,但是对于虚函数,赋值的并不是foo_a...image.png 图五:GCC单继承下成员函数指针赋值操作 很明显就和VC不一样啦,GCC在给成员函数指针赋值的时候,给了两个值,第二个暂时没用到后面讨论。...往下看函数调用: image.png 图六:GCC单继承下成员函数指针调用 可以看到,GCC的成员函数指针和VC的设计明显不同。...GCC正好利用了字节对齐的特点,用最后一位来标识是不是函数。如果是虚函数则查找虚表,计算实际函数地址;否则就直接跳转到该函数。...下一篇 VC和GCC成员函数指针实现的研究(二)

    84230

    VC和GCC内成员函数指针实现的研究(一)

    那么,VC和GCC是怎么实现这一功能的呢 VC单继承成员函数指针实现 图片 图一: VC指针赋值 可以看到,对非虚函数指针,直接把函数地址赋值过去了,但是对于虚函数,赋值的并不是foo_a::info...图片 图五:GCC单继承下成员函数指针赋值操作 很明显就和VC不一样啦,GCC在给成员函数指针赋值的时候,给了两个值,第二个暂时没用到后面讨论。...往下看函数调用: 图片 图六:GCC单继承下成员函数指针调用 可以看到,GCC的成员函数指针和VC的设计明显不同。...GCC正好利用了字节对齐的特点,用最后一位来标识是不是函数。如果是虚函数则查找虚表,计算实际函数地址;否则就直接跳转到该函数。...下一篇 VC和GCC成员函数指针实现的研究(二)

    52020

    C++ this指针:用于在成员函数中指向调用该函数的对象

    C++中this指针是一个指向当前对象的指针。在成员函数中,可以使用this指针来访问调用该函数的对象的成员变量和成员函数。...一、定义和使用this指针 this指针是在成员函数内部定义的一个常量指针。它存储了当前对象的地址,可以通过它访问当前对象的成员变量和成员函数。...在成员函数内,无需显式地传入this指针,编译器会自动将当前对象的地址赋给this指针。...二、作为返回值的this指针 this指针可以作为返回值返回。这种情况下,返回的是指向调用该函数的对象的指针。为了实现这个功能,需要将返回类型设置为类的引用或指针类型。...这里使用了*this来访问调用该函数的对象。 三、作为函数参数的this指针 this指针也可以作为函数参数传递。这种情况下,可以在函数内部访问其他对象的成员变量和成员函数

    23340

    c++系列之二 指向成员函数指针(烧脑)

    所以,静态成员函数不是类的一部分,成员函数指针的语法对常规函数指针并不成立,例如上面例子中的静态成员函数指针。...这个例子证明了成员函数指针不是常规指针。另外,为什么C++如此费心地去发明这样的语法?很简单,因为它和常规指针是不同的东西,而且这样的类型转换也是违反直觉的。...C++类型转换规则 非虚函数情形 我们在前面一节看到,成员函数指针不是常规指针,所以,成员函数指针(非静态)不能被转换成常规指针(当然,如果哪个脑残真想这么做的话,可以使用汇编技术来暴力解决),因为成员函数指针代表了...偏移量 不是 绝对地址 。...(翻译君去看了一下 mem_fun() 的源代码,原来是用成员函数来构造仿函数functor的。) 成员函数调用和 this 指针 现在我们回到文章最开始的地方。为什么一个空指针也能调用成员函数

    3K20

    【C++】函数指针 ② ( 数组类型基本语法 | 数组语法 | 数组首元素地址 和 数组地址 | 定义数组类型 | 定义指针数组类型语法 | 直接定义指针数组类型变量语法 )

    三、指针数组类型 1、定义指针数组类型语法 定义指针数组类型语法 : typedef type (*pArrayType)[size]; typedef 是 重命名关键字 , 这里是将数组类型重命名为...arrayType 数组类型名称 ; type 是 数组元素类型 ; * 表示 该定义的类型指针类型 , 该指针指向一个 type[size] 数组类型 的 数组 ; pArrayType 是 定义的指针数组类型名称...; size 是 数组大小 ; 定义指针数组类型示例 : 下面是定义了 指针数组类型 , 指向 有 5 个 int 类型的数组 , 数组类型名称为 pInt5ArrType; // 定义指针数组类型...该定义的类型指针类型 , 该指针指向一个 type[size] 数组类型 的 数组 ; p 是 定义的指针数组类型变量名称 ; size 是 数组大小 ; 直接定义指针数组类型变量示例 : 下面定义了变量...p , 该变量是一个指针类型的变量 , 指向 int[5] 类型数组的指针 ; // int(*)[5] 是 指向 int[5] 类型数组的指针 // 这里不使用 typedef 定义的

    12230

    【C++干货基地】六大默认成员函数: This指针 | 构造函数 | 析构函数

    其实在C++这里虽然我们没有给print 函数参数,但实际上是有一个默认的this 指针来自动调用的,实际的代码可能是下面这样但是编译器给自动化了大大简化了用户操作。...1.2 this 指针的特性 this指针类型:类类型 const,即成员函数中,不能给this指针赋值。...并不是,任何类在什么都不写时,编译器会自动生成以下6个默认成员 函数。 默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数。...这里很多人就觉得为什么对内置类型不处理,自动处理了不是更好吗?...因为析构函数一般是用来释放我们申请的空间,内置类型在程序结束会自动释放空间所以没有申请资源可以不写析构函数

    8200
    领券