当前位置: 首页 > news >正文

C++类成员指针的实现示例

一、类成员指针的核心定位

类成员指针是C++中专门指向“类的成员”(而非具体对象的成员)的特殊指针,和普通指针的核心区别:

  • 普通指针:直接指向内存中的某个地址(如变量、函数的入口地址);
  • 成员指针:不直接指向内存地址,而是存储“成员在类中的偏移量”,必须绑定具体对象/对象指针后才能访问(因为类的成员属于对象,而非类本身)。

二、数据成员指针

1. 定义语法

指向类的非静态数据成员的指针,语法格式:

// 格式:类名::* 指针变量名
类型 类名::* 数据成员指针名 = &类名::数据成员名;

2. 完整示例(可直接运行)

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

#include <iostream>

usingnamespacestd;

classPerson {

public:

string name;

intage;

staticintcount;// 静态数据成员(特殊情况)

};

intPerson::count = 0;// 静态成员初始化

intmain() {

// 1. 定义并赋值:指向Person的age成员

intPerson::*p_age = &Person::age;

// 指向Person的name成员

string Person::*p_name = &Person::name;

// 2. 绑定对象访问(用.*运算符)

Person p1{"Alice", 20};

cout <<"Name: "<< p1.*p_name << endl;// 输出Alice

cout <<"Age: "<< p1.*p_age << endl;// 输出20

// 3. 绑定对象指针访问(用->*运算符)

Person* p2 = &p1;

p2->*p_age = 21;// 修改age

cout <<"New Age: "<< p2->*p_age << endl;// 输出21

// 4. 静态数据成员指针(特殊:和普通指针一样)

int* p_count = &Person::count;// 无需Person::*,直接用普通指针

*p_count = 100;

cout <<"Count: "<< Person::count << endl;// 输出100

return0;

}

3. 关键注意点

  • 数据成员指针的类型必须严格匹配(如int Person::*不能指向string Person::*);
  • 静态数据成员不属于对象,因此指向静态数据成员的指针是普通指针,而非成员指针;
  • 空类/无数据成员的类,其数据成员指针的偏移量为0,但依然是合法的成员指针。

三、成员函数指针

1. 定义语法

指向类的非静态成员函数的指针,语法格式(需匹配返回值、参数列表、const/volatile限定):

// 格式:返回值 (类名::*)(参数列表) [const/volatile]
返回值 (类名::* 函数指针名)(参数列表) [const] = &类名::成员函数名;

2. 完整示例(含const成员函数)

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

#include <iostream>

#include <string>

usingnamespacestd;

classCalculator {

public:

intadd(inta,intb) {returna + b; }

intsub(inta,intb)const{returna - b; }// const成员函数

staticintmul(inta,intb) {returna * b; }// 静态成员函数

};

intmain() {

// 1. 指向非const成员函数

int(Calculator::*p_add)(int,int) = &Calculator::add;

// 2. 指向const成员函数(必须加const限定)

int(Calculator::*p_sub)(int,int)const= &Calculator::sub;

// 3. 绑定对象调用(.*运算符)

Calculator calc;

cout <<"Add: "<< (calc.*p_add)(10, 5) << endl;// 输出15

cout <<"Sub: "<< (calc.*p_sub)(10, 5) << endl;// 输出5

// 4. 绑定对象指针调用(->*运算符)

Calculator* p_calc = &calc;

cout <<"Add via ptr: "<< (p_calc->*p_add)(20, 3) << endl;// 输出23

// 5. 静态成员函数指针(特殊:和普通函数指针兼容)

int(*p_mul)(int,int) = &Calculator::mul;

cout <<"Mul: "<< p_mul(4, 5) << endl;// 输出20

return0;

}

3. 关键注意点

  • 成员函数指针的括号不可省略(calc.*p_add)(10,5)不能写成calc.*p_add(10,5)(运算符优先级问题,.->*优先级低于());
  • const成员函数的指针必须加const限定,否则不匹配(如int (Calculator::*)(int,int) const不能赋值给int (Calculator::*)(int,int));
  • 成员函数隐含this指针,因此必须绑定对象才能调用(静态成员函数无this,故无需绑定);
  • 成员函数指针的大小可能大于普通指针(如64位系统可能占16字节),因为需要存储函数地址+this调整信息(多继承场景)。

四、将成员函数用作可调用对象

成员函数指针本身不能直接作为“无上下文的可调用对象”(如传给std::threadstd::functionstd::for_each等),必须绑定对象/this指针,常用3种方式:

1. 方式1:std::bind(C++11及以上)

最经典的方式,将成员函数与对象绑定,生成可调用的函数对象:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

#include <iostream>

#include <functional> // 必须包含

usingnamespacestd;

classPrinter {

public:

voidprint(string msg,intnum) {

cout << msg <<": "<< num << endl;

}

};

intmain() {

Printer p;

// 绑定成员函数+对象,固定部分参数(也可留空参数用placeholders)

auto func = bind(&Printer::print, &p,"Number", placeholders::_1);

// 调用:只需传未绑定的参数

func(100);// 输出Number: 100

return0;

}

2. 方式2:std::function适配

将成员函数指针+对象封装为std::function,适配通用可调用接口:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

#include <iostream>

#include <functional>

usingnamespacestd;

classMath {

public:

intsquare(intx) {returnx * x; }

};

intmain() {

Math m;

// 封装为std::function:参数列表要匹配成员函数(隐含this已绑定)

function<int(int)> f = bind(&Math::square, &m, placeholders::_1);

cout << f(5) << endl;// 输出25

// 也可直接用lambda(更简洁,见方式3)

function<int(int)> f2 = [&m](intx) {returnm.square(x); };

cout << f2(6) << endl;// 输出36

return0;

}

3. 方式3:lambda表达式包裹(推荐,C++11+)

最简洁、易读的方式,用lambda捕获对象/this,包裹成员函数调用:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

#include <iostream>

#include <thread> // 用于演示线程调用成员函数

usingnamespacestd;

classWorker {

public:

voidwork(intid) {

cout <<"Worker "<< id <<" is running"<< endl;

}

};

intmain() {

Worker w;

// 1. 普通调用:lambda捕获对象

auto func = [&w]() { w.work(1); };

func();// 输出Worker 1 is running

// 2. 线程调用成员函数(核心场景)

threadt([&w]() { w.work(2); });

t.join();// 输出Worker 2 is running

// 3. 捕获this(成员函数内调用)

classTest {

public:

voidcall_self() {

auto f = [this]() {this->work(3); };

f();// 输出Worker 3 is running

}

voidwork(intid) { cout <<"Test Worker "<< id << endl; }

};

Test tst;

tst.call_self();

return0;

}

4. 典型场景:成员函数作为算法的回调

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

#include <iostream>

#include <vector>

#include <algorithm>

usingnamespacestd;

classFilter {

public:

boolis_even(intx) {returnx % 2 == 0; }

};

intmain() {

vector<int> nums = {1,2,3,4,5};

Filter f;

// 用lambda包裹成员函数,作为find_if的谓词

auto it = find_if(nums.begin(), nums.end(), [&f](intx) {

returnf.is_even(x);

});

if(it != nums.end()) {

cout <<"First even number: "<< *it << endl;// 输出2

}

return0;

}

五、核心对比:成员指针 vs 普通指针

特性普通指针(如int*、void(*)(int))类成员指针(如int A::、void (A::)(int))
指向目标内存地址(变量/函数入口)类成员的偏移量(需绑定对象)
大小固定(64位=8字节)可能更大(如多继承场景=16字节)
调用/访问方式直接解引用(*p、p())需绑定对象,用.* / ->* 运算符
静态成员适配完全兼容静态成员的指针等价于普通指针
this指针非静态成员函数指针隐含this,需绑定对象

总结

  1. 成员指针本质:存储类成员的偏移量,而非直接内存地址,必须绑定对象才能访问/调用;
  2. 语法关键:数据成员指针用类名::*,成员函数指针需匹配返回值/参数/const限定,调用用.*/->*
  3. 可调用对象适配:优先用lambda表达式包裹成员函数(简洁无坑),其次用std::bind,避免直接传递成员函数指针;
  4. 特殊情况:静态成员的指针等价于普通指针,无需绑定对象即可使用。
http://www.jsqmd.com/news/843785/

相关文章:

  • (高温)投入式液位计的产品优势是什么?该如何选型? - 仪表人小余
  • NotebookLM摘要可信度遭挑战?用LLM可解释性框架XAI-Note验证摘要溯源路径(附开源验证脚本)
  • 3月聚酯尼龙袋生产厂家口碑推荐,谁是口碑王者?聚酯尼龙袋厂商墨胶红塑业引领行业标杆 - 品牌推荐师
  • 磁盘使用率监控与告警脚本
  • 3分钟掌握:ncmdumpGUI免费转换网易云音乐ncm文件的完整指南
  • LeetCode 按摩师题解
  • transformer 中文本版代码
  • 电磁流量计全面产品介绍:工作原理与测量优势深度解析 - 陈工日常
  • 番茄小说下载器:一款跨平台智能小说下载与有声书生成工具
  • 地理空间智能和相似性分析在数据映射中的应用
  • 收的顶青岛黄金回收,实体可查,专业鉴定 + 秒到账 - 奢侈品回收测评
  • 【程序源代码】校园论坛仿知乎贴吧微信小程序系统(含源码)
  • 如何快速安装Winget:PowerShell一键部署Windows包管理器终极指南
  • 四足机器人动态运动控制与并联柔顺性优化
  • 使用 Python 和 Taotoken SDK 五分钟内完成第一个大模型调用
  • 基于LangBot框架构建智能对话机器人:从架构解析到生产部署
  • 2026 年 4 月超声波热量表品牌排名,国产替代趋势加速 - 陈工日常
  • Claude Code × DeepSeek V4:从零开始配置与调用实战
  • 旧改门禁落地避坑指南:从上海1700小区失效事件学到的五大教训
  • 杭州四维彩超诊疗机构排行:西湖花生医院领衔第一梯队 - 奔跑123
  • CodeTree:多Git仓库管理工具的设计原理与工程实践
  • 魔兽争霸III终极优化指南:5分钟让你的经典游戏重获新生 [特殊字符]
  • 终极Unity游戏翻译指南:XUnity.AutoTranslator完全手册
  • 品牌设计公司核心方法论全景:从视觉执行到品牌治理的范式演进 - 2026品牌推荐官
  • 石家庄全屋整装厂家技术实力拆解与适配场景推荐 - 奔跑123
  • AlwaysOnTop终极指南:3分钟掌握Windows窗口置顶神器,工作效率提升300%[特殊字符]
  • JetBrains IDE无限期使用终极指南:简单三步永久重置30天试用期
  • 茉莉花插件:5分钟解决Zotero中文文献管理的三大痛点
  • (带显示)单晶硅双法兰液位变送器的产品优势是什么?该如何选型? - 仪表人小余
  • 从原理到实战:晶体管开关电路设计与常见问题解析