删除constexpr会改变链接吗?

Does removing constexpr change linkage?

本文关键字:链接 改变 constexpr 删除      更新时间:2023-10-16

我在头文件中定义了一个简单的配置结构体,其中包含一系列其他简单结构体,这些结构体只是静态变量的容器。下面是一个例子:

// Config.h    
struct Config {
    struct Server {
        static constexpr const char* url = "http://example.com";
        static constexpr float polling_interval = 1.0f;
    };
    struct Window {
        static constexpr int width = 1920;
        static constexpr int height = 1200;
    };    
};

我在需要它的地方包含头文件,并像这样访问变量:Config::Window::width

这工作得很好,但在某一点上,我需要从文件中加载值,所以我改变了变量声明,而不是常量(例如,static constexpr int width = 1920;成为static int width;)。现在链接器抱怨变量的符号未定义。static constexpr对象的链接与static对象的链接不一样吗?我还遗漏了什么吗?

与链接无关。

在声明中指定值的静态常量可作为常量表达式使用,不需要为其用法定义。

静态变量确实需要定义,如果它们不是常量(因此值需要在运行时存储在某个地方),或者如果它们odr使用(粗略地说,如果你取它们的地址或形成对它们的引用,这再次意味着它们需要在运行时存在于某个地方)。

您错过了非constexpr成员的定义。在.cpp文件中,你需要有例如

int Config::width;

constexpr成员是隐式内联的,因此不需要定义。一旦它们成为适当的变量,您需要在翻译单元(.cpp)中分配它们。