面试:类模版中函数声明在.h,定义在.cpp中,其他cpp引用引入这个头文件,会有什么错误?

1、概述

类模版中函数声明在.h,定义在.cpp中,其他cpp引用引入这个头文件,会有什么错误?
报编译错误:error C2512: 'Demo<int>': no appropriate default constructor available

举例如下代码:
demo.h 声明模版类

cpp 复制代码
template <typename T>
class Demo
{
public:
    Demo(T value);

private:
    T m_value;
};

demo.cpp

cpp 复制代码
#include "demo.h"

template <typename T>
Demo<T>::Demo(T value) : m_value(value)
{
}

main.cpp

cpp 复制代码
#include "demo.h"
int main()
{
    Demo<int> demo;

    return 0;
}

2、原因

模版不是在编译时立即生成代码,而是在使用时(即实例化时)根据传入的参数,来生成具体的代码。

编译器要看到完整的模版定义,才能生成具体的函数代码。

如果模版的定义不在当前编译单元可见,编译器就不知道如何生成特定类型的实例化模版,就会报编译错误。

如上编译main.cpp文件时 :编译器会根据demo<int> 语句来生成Demo<int>的具体版本,但是模版的定义在demo.cpp文件中,编译main.cpp时是无法看到的,所以只有声明,没有实现,所以报错了"error C2512: 'Demo<int>': no appropriate default constructor available"

这里有一个问题:为什么普通函数 声明 和定义分开,编译就没有报错呢?
1、普通函数- 编译阶段

当编译器编译一个源文件时,会根据文件中函数的调用去查找函数声明,如果找到了匹配的声明,编译器就认为这个函数存在,并生成一个对外部符号的引用。

2、普通函数-链接阶段

(1)、在所有源文件被单独编译之后,链接器的任务是将所有生成的目标文件(.o 或 .obj 文件)合并成一个可执行文件或库。

(2)、链接器会解析每个目标文件中的外部符号引用,并尝试在其他目标文件中找到对应的定义。

(3)、如果所有被引用的符号都能找到定义,那么链接成功;否则,就会出现"未定义的引用"这样的链接错误。

3、解决办法

模版函数的声明和定义一般需要在一起

学习链接:https://github.com/0voice

相关推荐
路溪非溪29 分钟前
嵌入式wifi专家成长学习路线
学习
沛沛rh4531 分钟前
用 Rust 实现用户态调试器:mini-debugger项目原理剖析与工程复盘
开发语言·c++·后端·架构·rust·系统架构
mxwin38 分钟前
Unity Shader中如何学习阴影技术 产生阴影,接受阴影,联级阴影,软阴影
学习·unity·游戏引擎·shader
云栖梦泽39 分钟前
Linux内核与驱动:13.从设备树到Platform平台总线
linux·运维·c++·嵌入式硬件
qeen8743 分钟前
【算法笔记】模拟与高精度加减乘除
c++·笔记·算法·高精度·模拟
txinyu的博客1 小时前
高并发内存池 - 简化版 tcmalloc
c++
少司府1 小时前
C++基础入门:内存管理
c语言·开发语言·c++·内存管理·delete·new·malloc
鱼很腾apoc1 小时前
【学习篇】第17期 C++入门必看——类和对象全站最详篇
c语言·开发语言·学习·算法·青少年编程
土豆~1 小时前
Claude Code源码学习—— Agent Prompt 设计
学习·prompt·claude code
郝学胜-神的一滴1 小时前
从零起步:CMake基础入门与实战跨平台编译
c++·软件工程·软件构建·cmake