- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我有一个在 CLion 2016.3.4 上开发的 C++14 项目,一段代码给我检查错误。我创建了一个最低限度的代码来重现该问题:
#include <iostream>
#include <type_traits>
#include <system_error>
using error_id_type = int;
template <typename T> using enable_if_condition_enum_t =
typename std::enable_if<std::is_error_condition_enum<T>::value, T>::type;
// Declaration
template <typename T, typename = enable_if_condition_enum_t<T>>
constexpr error_id_type error_enum_to_int(T elem) noexcept;
// Definition
template <typename T, typename = enable_if_condition_enum_t<T>>
constexpr error_id_type error_enum_to_int(T elem) noexcept {
return static_cast<error_id_type>(elem);
};
int main(void) {
error_id_type condition = error_enum_to_int(std::errc::owner_dead); // inspection error here
switch (condition) {
case error_enum_to_int(std::errc::address_in_use): break; // inspection error here
default: break;
}
std::cout << condition << std::endl;
return 0;
}
CLion 给我 Call to "error_enum_to_int" is ambiguous
对于 error_enum_to_int
的每次调用.这种用法真的有问题吗?
一些我尝试过的东西,但恕我直言,这并不能真正解决问题:
T
enable_if_condition_enum_t<T>
的争论(不起作用,给我一整套新的编译和检查错误)。此外,代码在 g++ (GCC) 6.3.1 20170306
上编译和运行得很好,没有任何修复。 .不幸的是,我现在无法访问另一个编译器来测试它,但我猜这是标准的、可移植的 C++11。
当然,始终可以选择 static_cast<some_enum_class>(some_int)
,但我想具体了解这段代码可能有什么问题。
所以,我的问题是:这是我的 IDE 的错误吗,是否真的存在我不知道的语言的一些特殊情况,或者我真的在这里做了一些愚蠢的事情 (:D)?
如有错误请指正
声明本身不是定义,即使它是一个函数模板。函数模板本身在被实例化之前并不是真正的函数。即便如此,编译器应该能够看到同一个函数模板有两次不同的出现,并且不会将一个与另一个混淆,只要两者都存在于同一个编译/翻译单元中。我的看法是 CLion(或者可能是 clang 的静态分析器)以某种方式将声明视为定义并将它们解释为不同的东西。由于两者都是模板,因此对于实例化哪一个变得很困惑。
请注意,声明和定义都出现在同一个编译单元中。另外,constexpr
暗示 inline
,所以单一定义规则仍然适用。此外,我使用 enable_if_condition_enum_t
对于基于 SFINAE 的消除,如果用作参数的枚举未声明为 std::error_condition
枚举。
正在关注 @Angew's answer , 这是 error_enum_to_int
的正确声明和定义.
// Declaration
template <typename T, typename = enable_if_condition_enum_t<T>>
constexpr error_id_type error_enum_to_int(T elem) noexcept;
// Definition
template <typename T, typename>
constexpr error_id_type error_enum_to_int(T elem) noexcept {
return static_cast<error_id_type>(elem);
};
最佳答案
在 C++ 中,您不能为同一参数/模板参数多次提供默认参数或默认模板参数。来自函数所有声明(包括定义)的默认 [template] 参数被组合(级联)。您应该从模板的定义中删除默认模板参数。
关于c++ - SFINAE 消除、Constexpr 和函数模板 : Can I keep declaration and definition separate?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42891055/
我有一个模板类 展览.h: template class ExpOf{ ... } 我在整个代码中反复使用,例如T = double [和其他类ExpOf应该一无所知]。 所以我认为一次性编译它是个
如果你有一个名为“Rock”的类,你会做类似的事情:- int main() { Rock; } 为什么会出现“声明未声明任何内容”错误? 它不应该只是调用默认构造函数并在那一刻创建对象吗?
这是一个非常业余的问题,我确信这将是一个非常简单的答案,但我似乎无法弄清楚问题所在。我有一个带有相应 .cpp 文件的头文件,但出于某种原因,每当我尝试使用 g++ 进行编译时,我都会收到错误消息:
我正在使用 MinGW 将我的 Linux 项目转换为在 Windows 上编译。它在 Linux 上编译和运行都很好,但是当我尝试用 MinGW 编译它时,它会出现以下错误消息: camera.h:
我收到“decleration does not declare anything [-fpermissive] 错误”;这是我的代码; #ifndef CAMERA_H #define CAMERA
我正在编写一些 cython 代码,但遇到了一个奇怪的问题。当我尝试将对象作为结构直接从 python 传递到 C 时,cython 生成的代码很好,但 gcc 不喜欢代码输出并给我以下错误:erro
typedef struct BO2Offsets { struct Prestige { u32 offset = 0x000000; char da
我不明白 C++ 中的某些东西,gcc 不喜欢我如何进行。 我做到了: if (!fModeMdi) MyFirstClass* main = (MyFirstClas
在 switch-case 语句中,declaration-with-initialization 是无效的,但允许 declaration-and-then-assignment。如以下代码片段所示
我在我的界面文件中收到一条奇怪的警告。 这也出现在我为此声明属性的那一行。 谁能帮帮我? 最佳答案 在您的项目中的某处,您有一个 #define 将 xOffset 定义为空(除了注释)。像这样: #
declare +x 下面做了什么? (特定于 Bash。)我理解 declare -x,但不理解 declare +x: function the_func { declare +x MY_VA
由于我是 Symfony 的新手,我尝试使用 Doctrine 创建实体关系。我收到错误 “[bundle/entity/file_location”中的属性“report”已经声明,但在我尝试更新架
关闭。这个问题是not reproducible or was caused by typos .它目前不接受答案。 想改进这个问题?将问题更新为 on-topic对于堆栈溢出。 2年前关闭。 Imp
这是给我错误的代码: TAdvSmoothDockItems = class(TCollection) private FOwner: TAdvSmoothDock; FOnChange: T
我对 python 很陌生,我尝试制作一个简单的 GUI 程序。但是,我遇到了一个“问题”,确切地说是一个警告,上面写着:“m”未在全局范围内定义(Python(变量未定义全局))。 我知道如果你想在
当我用 GCC 编译程序时,它会显示“警告:声明未声明任何内容 [-fpermissive]”。 有问题的代码如下: typedef int BOOL; 如何清除警告? 最佳答案 您可以尝试以下操作。
我正在编写一个包含键值对集合的重要类,在编译期间我收到一个我无法弄清楚的非常奇怪的错误。在一个与这里的函数非常相似的函数中,但由于所需代码的复杂性而没有上下文,我收到错误: TValue& opera
这个问题很简单。为了进一步阐明,下面代码中的 Foo1 和 Foo2 在它们的声明方式方面到底有什么区别(例如,使用 class Foo1 { 。 .. }; 而另一个使用 typedef class
我正在开发 Web 项目,并且在从 Oracle 数据库迁移到 mysql 数据库时遇到一些问题。我想用这段代码创建函数: DROP FUNCTION IF EXISTS F_MANIFEST_GAB
是否有一个标志可以传递给 gcc 以禁用此警告?我知道它的作用,但这对我的程序来说无关紧要。 编辑:我只想禁用警告,保持代码不变。编译以下代码会生成警告: struct post{ unsigne
我是一名优秀的程序员,十分优秀!