C++中的类成员函数当线程函数

 更新时间:2022年11月05日 11:00:06   作者:Ryan-S  
这篇文章主要介绍了C++中的类成员函数当线程函数,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

类成员函数当线程函数

C++类成员函数使用时,都会隐式传递一个this指针给该函数,this指针指向该类的对象。函数体可以通过显示调用该指针或直接访问类内成员。

回调函数是通过指针调用的函数,最常使用的回调函数就是在创建线程时,以一个函数指针以及传递给这个函数多个参数来调用线程函数来创建线程。

那么一般的类成员函数是不能用作回调函数的,因为库函数在使用回调函数时,都会传递指定的符合回调函数声明的的参数给回调函数,而类成员函数隐式包含一个this指针参数,所以把类成员函数当作回调函数编译时因为参数不匹配会出错。

std::thread,它的第一个参数为函数指针,在c++中这样是获取不到其成员函数的指针,所以会报错。

解决方法一

把成员函数设成静态成员函数,不属于某个对象,属于整个类,没有this指针。但是静态成员函数并不能使用非静态的成员变量(因为它没有某个具体对象的this指针),可通过对象或者类指针调用。

解决方法二

把成员函数声明为友元函数,没有this指针,但是能够访问类的成员变量。

解决方法三

假设需要在单独的线程中调用类Hack的非静态成员函数func2()。不用直接传递成员函数的地址到thr_create(),声明一个带 void* 参数的普通函数 intermediary(void*),然后调用它:

void intermediary(void);

接着创建一个结构,结构定义如下:

struct A
{
Hack * p; //类对象指针
void (Hack::*pmf)(); // 成员函数指针
};

创建一个结构实例,用希望的对象地址和成员函数地址填充结构:

A a; // 结构实例
Hack h; // 创建对象
//填充结构
a.p = & h;
a.pmf = &Hack::func2; // 取成员函数地址

现在回过头来实现intermediary()函数:

void *intermediary(void* ptr)
{
 A* pa=static_cast < A* > (ptr); // 强制转换 p 为 A*
 Hack* ph=pa->p; // 从A中析取Hack对象地址
 void (Hack::*pmf)()=pa->pmf; // 析取 ptr 到成员函数
 (ph->*pmf)(); // 调用成员函数
}

最后将intermediary()的地址传递到thr_create():

pthread_create (&ptid, NULL, intermediary, (void *)&a );

pthread_create()调用函数intermediary()并将A的地址传递给它。intermediary()再从其指针参数中展开结构A并调用希望的成员函数。

这种间接方式的处理可以安全地在单独线程中启动成员函数,即便是线程库不支持成员函数。

如果需要调用不同类的不同成员函数,可以将结构A转换成类模板,将函数intermediary()转换成函数模板。从而编译器便会自动产生大多数样板文件代码。

类成员函数作为多线程的入口

搜了一圈答案,基本上都是启动线程的时候传入this指针,在线程函数内部再强转的解决方案。可能显得有些别扭。

编译器不允许强制转换,那就用union来实现。

union
{
  void *(*trfunc)(void *);
  void *(lock_client_cache::*memfunc)();
} func;
 
func.memfunc = &lock_client_cache::do_thread;
pthread_t pid;
pthread_create(pid, 0, func.trfunc, this);
pthread_detach(pid);

do_thread是非静态类成员函数,没有参数。posix库的情况下返回一个void*,win32的线程的情况下返回void。

*该方法适用于只需要传递this指针的情况,如果需要传递多个参数,还要按老方法。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 一文详解C语言char类型中的存储

    一文详解C语言char类型中的存储

    C语言中的char是用于声明单个字符的关键字,这篇文章主要给大家介绍了关于C语言char类型中存储的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-01-01
  • C语言const关键字的用法详解

    C语言const关键字的用法详解

    今天探讨const,首先来说是将变量常量化。为什么要将变量常量化,原因有诸多好处有诸多。比如可以使数据更加安全不会被修改
    2022-08-08
  • C++聚合关系类的构造函数的调用顺序详解

    C++聚合关系类的构造函数的调用顺序详解

    下面小编就为大家带来一篇C++聚合关系类的构造函数的调用顺序详解。小编觉得挺不错的,现在就分享给大家,也给大家做个参考,一起跟随小编过来看看吧
    2016-05-05
  • C++类与对象之运算符重载详解

    C++类与对象之运算符重载详解

    运算符重载的方法是定义一个重载运算符的函数,在需要执行被重载的运算符时,系统就自动调用该函数,以实现相应的运算。也就是说,运算符重载是通过定义函数实现的
    2021-10-10
  • 关于C++类的成员初始化列表的相关问题

    关于C++类的成员初始化列表的相关问题

    下面小编就为大家带来一篇关于C++类的成员初始化列表的相关问题。小编觉得挺
    2016-05-05
  • C语言实现直方图均衡化

    C语言实现直方图均衡化

    这篇文章主要为大家详细介绍了C语言实现直方图均衡化,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-10-10
  • 详解C语言中的getgrgid()函数和getgrnam()函数

    详解C语言中的getgrgid()函数和getgrnam()函数

    这篇文章主要介绍了详解C语言中的getgrgid()函数和getgrnam()函数,是C语言入门学习中的基础知识,需要的朋友可以参考下
    2015-08-08
  • C语言实现线索二叉树的定义与遍历示例

    C语言实现线索二叉树的定义与遍历示例

    这篇文章主要介绍了C语言实现线索二叉树的定义与遍历,结合具体实例形式分析了基于C语言的线索二叉树定义及遍历操作相关实现技巧与注意事项,需要的朋友可以参考下
    2017-06-06
  • C++炸弹小游戏示例代码

    C++炸弹小游戏示例代码

    这篇文章主要介绍了C++炸弹小游戏,本文给大家分享游戏代码,代码简单易懂通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-08-08
  • C语言详细分析常见字符串函数与模拟实现

    C语言详细分析常见字符串函数与模拟实现

    字符串函数(String processing function)也叫字符串处理函数,指的是编程语言中用来进行字符串处理的函数,如C,pascal,Visual以及LotusScript中进行字符串拷贝,计算长度,字符查找等的函数
    2022-03-03

最新评论