是否在空的初始化程序列表(并明确指定类型)上调用 std::min 未定义的行为?

Posted

技术标签:

【中文标题】是否在空的初始化程序列表(并明确指定类型)上调用 std::min 未定义的行为?【英文标题】:Is calling std::min on an empty initializer list (and explicitly specifying the type) undefined behavior? 【发布时间】:2015-07-02 08:52:58 【问题描述】:

使用空的初始化列表调用std::min() 通常不会编译(所有问题都可以用与std::max() 相同的方式说明)。 这段代码:

#include <iostream>
#include <algorithm>

int main() 
   std::cout << std::min() << "\n";
   return 0;

用 clang 给出这个错误:

test.cpp:6:17: error: no matching function for call to 'min'
   std::cout << std::min() << "\n";
                ^~~~~~~~
algorithm:2599:1: note: 
      candidate template ignored: couldn't infer template argument '_Tp'
min(initializer_list<_Tp> __t)

我明白为什么不允许这种情况,因为在这种情况下很难就返回的合理值达成一致。

但是,从技术上讲,代码不能编译,只是因为模板参数无法推断。如果我强制参数编译代码但我会崩溃:

#include <iostream>
#include <algorithm>

int main() 

  std::cout << std::min<int>() << "\n";

  return 0;


$ clang++ -std=c++11 test.cpp -o test
$ ./test 
Segmentation fault: 11

似乎发生崩溃是因为std::min() 是根据std::min_element() 实现的,并且空的初始化列表会导致对无效end() 迭代器的取消引用。

那么这段代码在 C++11/C++14 下是未定义的行为吗? std::min() 是否声明在没有显式模板参数的情况下调用时不会编译? 是否指定std::min()std::min_element() 的形式实现?

【问题讨论】:

我没想到它会做任何特别的事情。 我同意原则上没有好的答案。我什至在问题中说明了这一点。但是由于调用它的一种方式可以编译而另一种不可以,所以我问编译的方式是否确实指定为 UB。 【参考方案1】:

是的,它是 UB。根据 C++14 (n4140) 25.4.7/4:

template <class T>
constexpr T min(initializer_list<T> t);

...

4 要求: TLessThanComparableCopyConstructiblet.size() &gt; 0

(强调我的)

相同的措辞也出现在 C++11 中。

【讨论】:

我的错...我查看了标准,但由于某种原因我完全错过了t.size() &gt; 0 的要求!

以上是关于是否在空的初始化程序列表(并明确指定类型)上调用 std::min 未定义的行为?的主要内容,如果未能解决你的问题,请参考以下文章

我可以在空的arraylist索引上替换或放置一个新字符串吗

在空的 tibble 中,列类型会根据添加第一行的方式而改变

在空对象引用上调用虚拟方法“double android.location.Location.getLatitude()”时出错

无法从列表视图中获取项目位置:尝试在空对象引用上调用虚拟方法...

原因:java.lang.NullPointerException:尝试在空对象引用上调用接口方法

为啥 ToPagedList 在空的 IQueryable 上需要半分钟,而底层 SQL 只需一秒钟?