为什么我在尝试创建 std::numpunct 衍生物的实例时出现链接错误<char16_t>?

Why do I get link errors when trying to create an instance of std::numpunct<char16_t> derivative?

本文关键字:错误 链接 lt gt char16 创建 std 实例 衍生物 numpunct 为什么      更新时间:2023-10-16

我正在尝试将std::basic_stringstream<char16_t>与基于UTF16的std::numpunct一起使用。但是,创建std::numpunct<char16_t>的最简单可能的导数的尝试在链接阶段失败了。这是代码:

#include <locale>
struct my_punct : public std::numpunct<char16_t> {};
int main()
{
my_punct n;
}

以下是链接错误(使用g++7.2.0)(实时测试):

$ g++-7 test0.cpp -o test
/tmp/cc0oWPhL.o: In function `std::numpunct<char16_t>::numpunct(unsigned int)':
test0.cpp:(.text._ZNSt8numpunctIDsEC2Ej[_ZNSt8numpunctIDsEC5Ej]+0x36): undefined reference to `std::numpunct<char16_t>::_M_initialize_numpunct(__locale_struct*)'
/tmp/cc0oWPhL.o: In function `my_punct::~my_punct()':
test0.cpp:(.text._ZN8my_punctD2Ev[_ZN8my_punctD5Ev]+0x18): undefined reference to `std::numpunct<char16_t>::~numpunct()'
/tmp/cc0oWPhL.o:(.rodata._ZTVSt8numpunctIDsE[_ZTVSt8numpunctIDsE]+0x8): undefined reference to `std::numpunct<char16_t>::~numpunct()'
/tmp/cc0oWPhL.o:(.rodata._ZTVSt8numpunctIDsE[_ZTVSt8numpunctIDsE]+0xc): undefined reference to `std::numpunct<char16_t>::~numpunct()'
collect2: error: ld returned 1 exit status

如果我将char16_t更改为char,则链接成功完成。那么,我在这里做错了什么?我需要一些额外的库来链接吗?

作为健全性检查,我尝试过为_ZTVSt8numpunctIDsE进行GCC安装,但没有结果。但对CCD_ 7的搜索确实找到了CCD_ 8。这是否意味着我在libstdc++中发现了一个错误?

这是因为没有专门类的支持实现。当您使用模板时,编译器会为特定用途生成一个符号。如果已经有一个预定义的、完全模板化的实现,那么编译器可以进行自己的定义,但前提是它在编译试图使用模板化实现的文件时看到了模板化实现。如果看不到,那么它只会创建符号,使其未定义,并期望稍后链接它。

举个简单的例子。。。

在Template_Add.h

// Generic template
template< class T >
T add(T first, T second);

在Template_Add.cpp中

// Specialized for int
template<>
int add<int>(int first, int second){

在main.cpp中

#include "Template_Add.h"
int main(){
int     ia, ib, ic;
double  da, db, dc;
ia = 3;
ib = 4;
da = 3.0;
db = 4.0;

// we expect ia = 7, and works
ic = add(ib, ic);
// we expect dc = 7.0, but doesn't work
// this will result in an unresolved symbol
dc = add(da, db);
return 0;
}

这种行为的原因是main.cpp可以看到的任何地方都不存在模板化实现。然而,当所有内容都被链接时,链接器会看到存在"int add(int,int)",因此这是允许的。

如果您拥有模板,那么您可以将(模板化的)实现添加到包含模板声明的.h文件中。这样编译器就知道如何在编译时生成符号和定义。为了修复上面的例子,我们可以将以下代码添加到Template_add.h中:

template< class T >
T add(T first, T second){
return first + second;
}

然后,包括"Template_Add.h"在内的任何东西都可以为任何试图使用该模板的东西生成自己的实现。

如果你没有模板,那么你能做的最好的事情就是找到一种方法来使用支持的模板专业化。在这种情况下,最好将char16_t强制转换(或typedef)为short,然后看看这是否有效。

相关文章: