我有一个基类First
和一个派生类Second
。在基类中有成员函数create
和虚函数run
。在Second
的构造函数中,我想调用函数First::create
,该函数需要访问其子类“run()
函数的实现”。一位同事建议使用函数模板,因为First
无法明确地知道它的子类。听起来怪怪的?这是一些代码:
First.h
#pragma once
#include <boost/thread/thread.hpp>
#include <boost/chrono/chrono.hpp>
class First
{
public:
First();
~First();
virtual void run() = 0;
boost::thread* m_Thread;
void create();
template< class ChildClass >
void create()
{
First::m_Thread = new boost::thread(
boost::bind( &ChildClass::run , this ) );
}
};
First.cpp
#include "First.h"
First::First() {}
First::~First() {}
Second.h
#pragma once
#include "first.h"
class Second : public First
{
public:
Second();
~Second();
void run();
};
Second.cpp
#include "Second.h"
Second::Second()
{
First::create<Second>();
}
void Second::run()
{
doSomething();
}
我在First::create<Second>();
收到错误错误:不允许输入类型名称。那么这个错误的原因是什么?我假设我没有完全掌握模板的整体机制,但是 - 我对这个主题很新。
答案 0 :(得分:1)
首先您必须在标头中放置模板定义,否则链接器将无法找到实例化的模板。
第二:现在双关语已经完成,我假设您从Second
派生了First<Second>
。这被称为“奇怪的重复模板模式”(CRTP)。
由于基类已经是专用的并且派生类的类型被刻入其类型中,因此不需要将它的任何函数作为模板(您没有),并且基类的函数可以是刚刚调用,没有指定模板参数:
template <class SubClass>
struct First {
void create() {
SubClass& sub = static_cast<SubClass&>(*this);
mThread = make_unique<std::thread>( [&](){sub.run();} );
}
};
struct Second : First<Second> {
Second() { create(); }
void run();
};
一些注释:
First
create
,它是一个继承的方法,并且由于Second
不是模板,所以不需要添加限定条件或帮助编译器进行查找。std::thread
而不是boost::thread
,现在已经有一段时间了。std::unique_ptr
而不是拥有所有权的原始指针bind
- 对我来说更清楚。struct
,以便默认情况下将成员函数设置为public,并在示例代码中保存一行。 class
除了那个没有区别。答案 1 :(得分:1)
虽然您可以使用CRre作为Kerrek SB和Arne Mertz的“建议”,但这是使用成员函数模板的解决方案:
class First
{
public:
First();
~First();
virtual void run() = 0;
boost::thread* m_Thread;
// vvvvvvvvvvv this declares a non-template member function `create`
void create(); // (better remove it)
// vvvvvvvvvv this declares a member function template `create`
template< class ChildClass >
void create();
};
对于模板,您应该在头文件中提供定义,因为定义必须在每个需要它的TU中可用(而不仅仅在一个TU中)。
最简单的方法是在类本身内提供成员函数模板或类模板成员的定义:
class First
{
public:
/* ... */
template< class ChildClass >
void create()
{
// AFAIK, the `bind` is not required
First::m_Thread = new boost::thread(&ChildClass::run,
static_cast<ChildClass*>(this));
}
};
现在,如果要在Second
类中调用此成员函数模板,则必须显式提供模板参数:
void Second::any_function()
{
create<Second>();
}
您的代码存在其他一些(可能的)问题:
&ChildClass::run
的类型为void (Second::*)()
,因此需要进行转发。create
的ctor中调用Second
,它将调用Second::run
(或您提供的任何模板参数),类{em>中的最终重写 {1}} 的。如果您在从Second
派生的类中覆盖该函数,则不会在Second
的ctor中调用该覆盖。Second
的你的dtor应该是First
或detach
提升线程,否则如果在调用dtor时线程仍在运行,则会调用join
。< / LI>
std::terminate
在此示例中不必为虚拟,完全相同的代码可以在run
虚拟甚至声明run
时使用First
。