- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
引自 The C++ standard library: a tutorial and handbook :
The only portable way of using templates at the moment is to implement them in header files by using inline functions.
这是为什么?
(澄清:头文件不是唯一的可移植解决方案。但它们是最方便的可移植解决方案。)
最佳答案
警告:不需要将实现放在头文件中,请参阅本答案末尾的替代解决方案。
无论如何,您的代码失败的原因是,在实例化模板时,编译器使用给定的模板参数创建了一个新类。例如:
template<typename T>
struct Foo
{
T bar;
void doSomething(T param) {/* do stuff using T */}
};
// somewhere in a .cpp
Foo<int> f;
当读到这一行时,编译器会创建一个新类(我们称它为FooInt
),它等同于以下内容:
struct FooInt
{
int bar;
void doSomething(int param) {/* do stuff using int */}
}
因此,编译器需要访问方法的实现,以使用模板参数(在本例中为 int
)实例化它们。如果这些实现不在 header 中,则无法访问它们,因此编译器将无法实例化模板。
一个常见的解决方案是在头文件中编写模板声明,然后在实现文件(例如 .tpp)中实现该类,并将此实现文件包含在头文件的末尾。
Foo.h
template <typename T>
struct Foo
{
void doSomething(T param);
};
#include "Foo.tpp"
Foo.tpp
template <typename T>
void Foo<T>::doSomething(T param)
{
//implementation
}
这样,实现仍然与声明分开,但编译器可以访问。
另一种解决方案是保持实现分离,并显式实例化您需要的所有模板实例:
Foo.h
// no implementation
template <typename T> struct Foo { ... };
Foo.cpp
// implementation of Foo's methods
// explicit instantiations
template class Foo<int>;
template class Foo<float>;
// You will only be able to use Foo with int or float
如果我解释的不够清楚,你可以看看C++ Super-FAQ on this subject .
关于c++ - 为什么模板只能在头文件中实现?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56483877/
有一条(相对)众所周知的 Perl 公理:“只有 Perl 可以解析 Perl”。我想知道 Perl 6 是否仍然如此? 扩大讨论...考虑到 PyPy 最近的更新,我想到了这个问题。 Perl 独特
这是设置。在上一个问题中,我发现我可以通过子组件中的状态传递对象属性,然后使用 componentDidUpdate 获取该对象属性。在这种情况下,状态和属性都称为到达。 这是基本代码... expo
我运行的是 10.5.2 社区版。我已经标记了 源/主要/资源 作为源目录。我可以右键单击并“编译”某些文件,据我所知,这意味着 IDE 将文件复制到与发送类文件的“com.mydomain.pack
我是一名优秀的程序员,十分优秀!