RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 1001482
Accepted
Andrej Levkovitch
Andrej Levkovitch
Asked:2020-07-10 23:26:20 +0000 UTC2020-07-10 23:26:20 +0000 UTC 2020-07-10 23:26:20 +0000 UTC

类外的模板成员特化

  • 772

例如,有一个具有模板成员函数的类get不专门处理该类的标头。在另一个文件中(cpp在这个类声明之外),我创建了这个模板的两个特化,稍后将在代码中使用。所以,编译后,我得到一个很奇怪的结果:在调试版本(优化-O0)中,调用模板成员时,我调用了专门的函数,而在发布版本(优化)-O3中,调用了“空”的......

为什么?

这是一个例子:

结构体:

.
├── build.cmake
├── CMakeLists.txt
├── main.cpp
├── TemplateClass.hpp
└── utils.cpp

编码:

# cmake

cmake_minimum_required(VERSION 3.12)

project(template)

include(build.cmake)

set(PROJECT_SRC
  main.cpp
  utils.cpp
  )

add_executable(${PROJECT_NAME} ${PROJECT_SRC})

# build.cmake

set(CMAKE_EXPORT_COMPILE_COMMANDS ON)

if(NOT CMAKE_BUILD_TYPE)
  SET(CMAKE_BUILD_TYPE Release)
endif(NOT CMAKE_BUILD_TYPE)

if(NOT DEFINED BUILD_SHARED_LIBS)
  set(BUILD_SHARED_LIBS ON)
endif()

if(${CMAKE_BUILD_TYPE} STREQUAL Debug)
  enable_testing()
  include(CTest)
endif()


if("${CMAKE_CXX_COMPILER_ID}" STREQUAL "MSVC")
else()
  set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -Wall")
  set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -Wall -Wextra -Wshadow -Wnon-virtual-dtor -pedantic -g -O0")
endif()

set(CMAKE_RUNTIME_OUTPUT_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}/bin")
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}/lib")
set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY "${CMAKE_CURRENT_BINARY_DIR}/lib")


option(leak_check "set leak_check" 0)
option(profiling "set profiling" 0)
option(thread_check "set thread_check" 0)

if(${CMAKE_BUILD_TYPE} STREQUAL Debug AND leak_check)
  set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -fsanitize=address -fno-omit-frame-pointer")
endif()

if(${CMAKE_BUILD_TYPE} STREQUAL Debug AND thread_check)
  set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -fsanitize=thread")
endif()

if(${CMAKE_BUILD_TYPE} STREQUAL Debug AND profiling)
  set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -pg")
  set(CMAKE_EXE_LINKER_FLAGS_DEBUG "${CMAKE_EXE_LINKER_FLAGS_DEBUG} -pg")
  set(CMAKE_SHARED_LINKER_FLAGS_DEBUG "${CMAKE_SHARED_LINKER_FLAGS_DEBUG} -pg")
  set(CMAKE_STATIC_LINKER_FLAGS_DEBUG "${CMAKE_SHARED_LINKER_FLAGS_DEBUG} -pg")
endif()


if("${CMAKE_CXX_COMPILER_ID}" STREQUAL "MSVC")
  set(CMAKE_WINDOWS_EXPORT_ALL_SYMBOLS ON)
endif()


message(STATUS "buid type        " ${CMAKE_BUILD_TYPE})
message(STATUS "Project          " ${PROJECT_NAME})
message(STATUS "c compiler       " ${CMAKE_C_COMPILER})
message(STATUS "cxx compiler     " ${CMAKE_CXX_COMPILER})
message(STATUS "build tests      " ${BUILD_TESTING})
message(STATUS "build shared     " ${BUILD_SHARED_LIBS})
message(STATUS "leak   sanitizer " ${leak_check})
message(STATUS "thread sanitizer " ${thread_check})
message(STATUS "profiling        " ${profiling})
// main.cpp

#include "TemplateClass.hpp"
#include <cstdlib>
#include <iostream>

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

  TemplateClass a{"string", 0.5};

  std::cout << "first: " <<  a.get<std::string>() << std::endl;;
  std::cout << "second: " <<  a.get<double>() << std::endl;;

  return EXIT_SUCCESS;
}
// TemplateClass.hpp

#pragma once

#include <string>

class TemplateClass {
public:
  template <typename T>
  T get() const {
    return T{};
  }

  std::string string_;
  double      double_;
};
// utils.cpp

#include "TemplateClass.hpp"

template<>
double TemplateClass::get<double>() const {
  return double_;
}

template<>
std::string TemplateClass::get<std::string>() const {
  return string_;
}

PS恕我直言,我不喜欢类声明之外的模板专业化,但我必须使用它(意思是:没有声明的专业化,如示例所示)

c++
  • 1 1 个回答
  • 10 Views

1 个回答

  • Voted
  1. Best Answer
    AnT stands with Russia
    2020-07-10T23:29:55Z2020-07-10T23:29:55Z

    如果您希望将专业化定义放在单独的.cpp文件中 - 请。但是,所有成员特化必须在头文件中预先声明。

    // TemplateClass.hpp
    
    class TemplateClass {
      ...
      template <typename T> T get() const { ... }
      ...
    };
    
    template<> double TemplateClass::get<double>() const;
    template<> std::string TemplateClass::get<std::string>() const;
    

    没有这个,行为是不确定的。


    这种语言要求背后的逻辑很简单:

    在编译时没有前向声明的情况下,main.cpp编译器根本不知道在utils.cpp某处某处有这些函数的专用版本。在优化的代码中,编译时,main.cpp编译器只是悄悄地,不怀疑任何事情,根据类模板生成这些方法的“空”版本,并将它们的代码直接嵌入到调用点中。

    在未优化的版本中,main.cpp编译过程中不会发生内联,但会在链接阶段为后续解析生成一个诚实的调用。由于在传统实现中,专用版本是普通(“强”)符号,而从模板生成的版本是“弱”符号,链接器会选择您的特化,并且在未优化的版本中一切似乎都“工作”。但这只不过是巧合。


    但是,通过声明它们将这些琐碎的特化完全放在头文件中是有意义的inline。

    • 7

相关问题

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    根据浏览器窗口的大小调整背景图案的大小

    • 2 个回答
  • Marko Smith

    理解for循环的执行逻辑

    • 1 个回答
  • Marko Smith

    复制动态数组时出错(C++)

    • 1 个回答
  • Marko Smith

    Or and If,elif,else 构造[重复]

    • 1 个回答
  • Marko Smith

    如何构建支持 x64 的 APK

    • 1 个回答
  • Marko Smith

    如何使按钮的输入宽度?

    • 2 个回答
  • Marko Smith

    如何显示对象变量的名称?

    • 3 个回答
  • Marko Smith

    如何循环一个函数?

    • 1 个回答
  • Marko Smith

    LOWORD 宏有什么作用?

    • 2 个回答
  • Marko Smith

    从字符串的开头删除直到并包括一个字符

    • 2 个回答
  • Martin Hope
    Alexandr_TT 2020年新年大赛! 2020-12-20 18:20:21 +0000 UTC
  • Martin Hope
    Alexandr_TT 圣诞树动画 2020-12-23 00:38:08 +0000 UTC
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5