gpt4 book ai didi

c++ - 将错误与模板函数中的局部派生对象链接起来

转载 作者:太空狗 更新时间:2023-10-29 20:46:22 28 4
gpt4 key购买 nike

我有一个模板函数,它使用派生自另一个基类的本地类。当此函数在不同的编译单元中实例化时,链接器会为默认构造函数和析构函数抛出“多重定义”错误。

以下是一些给我带来麻烦的代码的简化版本。它由三个文件组成。它应该是有效的(?)C++ 代码:

嗯:

struct foo {
template <typename T>
void f(const T&);
};

struct base {
virtual ~base(){};
};

template <typename T>
void foo::f(const T&) {
struct derived: public base {
// derived(){}
// virtual ~derived(){}
};
derived x;
}

a.cpp:

#include "a.h"
void fa() {
foo a;
a.f(1);
}

int main(int argc, char *argv[]){}

b.cpp:

#include "a.h"
void fb() {
foo a;
a.f(1);
}

编译它会产生链接器错误,因为 derived 的构造函数和析构函数存在两次:

$ g++ a.cpp b.cpp
/tmp/ccvPK1l5.o: In function `void foo::f<int>(int const&)::derived::derived()':
b.cpp:(.text+0x24): multiple definition of `void foo::f<int>(int const&)::derived::derived()'
/tmp/ccRb6RYO.o:a.cpp:(.text+0x36): first defined here
[...]

有趣的是,如果您手动定义 derived 的构造函数和析构函数(通过取消注释这两行),一切正常。

我的代码中有什么无效的地方吗?还是 gcc 中的错误?我试过 gcc 4.3 和 4.4,两者都有同样的问题。

对于我的真实代码,我通过将“派生”声明为全局类而不是 f 中的本地类来解决这种情况。但我仍然很想知道出了什么问题以及原因,这样我就可以在将来避免它。

最佳答案

规范说 本地类 (9.8) 的成员函数没有链接。 (C++0x 9.3p3),所以这可能是 gcc 问题。

但是,它似乎在 g++4.5 中得到了解决,因为你的示例成功地通过了 g++4.5.2 的编译和链接(有或没有构造函数和析构函数注释):

$ cat a.h
struct foo {
template <typename T>
void f(const T&);
};

struct base {
virtual ~base(){};
};

template <typename T>
void foo::f(const T&) {
struct derived: public base {
//derived(){}
//virtual ~derived(){}
};
derived x;
}

$ cat a.cpp
#include "a.h"
void fa() {
foo a;
a.f(1);
}

int main(int argc, char *argv[]){}

$ cat b.cpp
#include "a.h"
void fb() {
foo a;
a.f(1);
}

$ g++ --std=c++0x --pedantic a.cpp b.cpp -o a
$ g++ -v
Using built-in specs.
COLLECT_GCC=/usr/bin/g++
COLLECT_LTO_WRAPPER=/usr/lib/x86_64-linux-gnu/gcc/x86_64-linux-gnu/4.5.2/lto-wrapper
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu/Linaro 4.5.2-8ubuntu4' --with-bugurl=file:///usr/share/doc/gcc-4.5/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.5 --enable-shared --enable-multiarch --with-multiarch-defaults=x86_64-linux-gnu --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib/x86_64-linux-gnu --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.5 --libdir=/usr/lib/x86_64-linux-gnu --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --enable-plugin --enable-gold --enable-ld=default --with-plugin-ld=ld.gold --enable-objc-gc --disable-werror --with-arch-32=i686 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.5.2 (Ubuntu/Linaro 4.5.2-8ubuntu4)

关于c++ - 将错误与模板函数中的局部派生对象链接起来,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8380904/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com