如果函数是在.cpp中实现的,则出现c++链接错误

c++ link error if function is implemented in .cpp

本文关键字:c++ 错误 链接 函数 cpp 实现 如果      更新时间:2023-10-16

如果我在.cpp中实现类的创建方法,我会得到

错误LNK2019:未解析的外部符号"protected:__thiscall Singleton::Singleton(void)"(??0Singleton@@IAE@XZ)在函数"public:static void __cdecl Singleton::create(void)"(?create@Singleton@@saxz

然而,如果我在头文件中实现该方法,它编译时没有任何错误:S

头文件

#pragma once
#include <iostream>
class Singleton
{
public:
static Singleton * getInstance()
{
return s_instance;
}
static void create();
static void destroy();
void help();
protected:
static Singleton * s_instance;
Singleton();
};

源文件:

#include "Singleton.h"
Singleton * Singleton::s_instance = NULL;
void Singleton::create()
{
if (!s_instance)
{
s_instance = new Singleton;
}
}

void Singleton::destroy()
{
delete s_instance;
s_instance = NULL;
}

然而,如果我在头中实现create方法,它不会抛出任何错误

在中实现创建方法的头文件

#pragma once
#include <iostream>
class Singleton
{
public:
static Singleton * getInstance()
{
return s_instance;
}
static void create(){
if (!s_instance)
{
s_instance = new Singleton;
}
}
static void destroy();

protected:
static Singleton * s_instance;
Singleton();
};

在cpp中,您的create函数试图通过使用新运算符初始化Singleton,但您没有给它一个构造函数。尝试给出Singleton()的实现。即:

protected:
static Singleton * s_instance;
Singleton() {}
};

问题

您已经声明为默认构造函数,并且正在使用它(在new表达式中),但尚未实现它。

修复

只需删除构造函数声明:

protected:
static Singleton * s_instance;
// Singleton();  -- don't have this. Remove it.
};

其他事项

有了protected特性,类是为继承而设计的,那么如何确保派生类只能通过单例机制实例化呢?

好吧,您对派生类没有太多控制,所以最简单的就是记录每个派生类应该声明和定义一个非public默认构造函数。

然而,有一个技巧可以用来强制执行这一点,因为虚拟基必须由派生最多的类初始化。这可以用来强制客户端代码在底部添加最终的类派生。其中派生最多的类是一个模板实例化,它定义了一个非公共构造函数。

一个更实际的选择是把事情颠倒过来。

也就是说,与其设计Singleton类进行派生(由受保护的东西发出信号),不如将其设计为从客户端代码类继承。同样,这意味着使用模板。Andrei Alexandrescu在他的经典著作《;现代C++设计";。