3

似乎编译器应该能够捕捉到 std::variant 没有迭代器方法的事实,但似乎我的代码编译没有问题(即使我为变体随机组成方法或成员变量),但它在运行时崩溃(理所当然)。有人可以解释为什么这段代码会编译吗?

注意:这不会阻碍进度,因为现在我正在使用 std::visit,但很高兴知道为什么要编译。

我尝试过使用不同的变体模式,它们都可以编译。请参阅代码示例。您可以将其弹出到 cppreferences 或 Godbolt 中,它应该使用 C++17 标志或更高版本进行编译

#include <variant>
#include <string>
#include <cassert>
#include <iostream>
#include <list>
#include <map>

template<typename K, typename V>
//using var_maps = std::variant<std::map<K,V>, std::multimap<K,V> >;
//using var_maps = std::variant<std::list<int>, std::list<float> >;
using var_maps = std::variant<int, float>;

template <typename K, typename V>
void flat( const var_maps<K,V>& vmap)
{
    //for(auto bIter = vmap.bexxxgin(), eIter = vmap.end(); bIter != eIter;
    for(auto bIter = vmap.begin(), eIter = vmap.end(); bIter != eIter;
      bIter = vmap.upper_bound( bIter->first )  )
      {

      }
}

我最初的案例是使用地图,但它可以有效地与任何东西一起编译。此外,我可以随机将 begin() 替换为任何其他单词,它仍然可以编译。我知道这样做的正确方法是访问。我不可避免地试图拥有一个处理地图和多地图并将其转换为另一种数据结构的函数。

谢谢!

4

2 回答 2

2

您的代码可以编译,因为begin()end()是依赖名称 - 它们依赖于函数模板参数,因此它们的查找被推迟到flat模板实例化。但它永远不会被实例化!

如果您添加以下内容,您的代码将不再编译:

int main () {
     &flat<int, int>;
}
于 2019-08-10T17:52:55.507 回答
1

它“编译”是因为该函数是一个模板。此处不生成任何代码,并且在解析模板时,除了基本的语法检查之外,无法进行完整的检查。

这是因为编译器不知道是否var_maps<K,V>包含begin()。可能有专业。

当您实例化 时,您将收到错误消息var_maps,即使用var_maps具体类型KV.

于 2019-08-10T17:52:13.510 回答