- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
最近我试图修复一个非常困难的 const-correctness 编译器错误。它最初表现为 Boost.Python 深处的多段模板呕吐错误。
但这无关紧要:这一切都归结为以下事实:C++11 std::begin
和 std::end
迭代器函数没有重载到取 R 值。
std::begin
的定义是:
template< class C >
auto begin( C& c ) -> decltype(c.begin());
template< class C >
auto begin( const C& c ) -> decltype(c.begin());
因此,由于没有 R 值/通用引用重载,如果您将 R 值传递给它,您将获得一个 const 迭代器。
那我为什么要关心呢?好吧,如果你有某种“范围”容器类型,即像“ View ”、“代理”或“切片”或一些呈现另一个容器的子迭代器范围的容器类型,通常很方便使用 R 值语义并从临时切片/范围对象中获取非常量迭代器。但是对于 std::begin
,您就不走运了,因为 std::begin
将始终为 R 值返回一个常量迭代器。这是一个老问题,在 C++11 给我们 R 值之前的一天,C++03 程序员经常对它感到沮丧——即临时对象总是绑定(bind)为 const
的问题。
那么,为什么 std::begin
没有定义为:
template <class C>
auto begin(C&& c) -> decltype(c.begin());
这样,如果 c
是常量,我们将得到一个 C::const_iterator
,否则将得到一个 C::iterator
。
一开始我以为是为了安全。如果您将临时变量传递给 std::begin
,如下所示:
auto it = std::begin(std::string("temporary string")); // never do this
...你会得到一个无效的迭代器。但后来我意识到这个问题在当前的实现中仍然存在。上面的代码只会返回一个无效的 const 迭代器,在取消引用时可能会出现段错误。
那么,为什么 std::begin
不 被定义为采用 R 值(或更准确地说,一个 Universal Reference )?为什么有两个重载(一个用于 const
,一个用于 non-const
)?
最佳答案
The above code would simply return an invalid const-iterator
不完全是。迭代器将一直有效,直到迭代器引用的临时对象是在词法上创建的完整表达式的末尾。所以像
std::copy_n( std::begin(std::string("Hallo")), 2,
std::ostreambuf_iterator<char>(std::cout) );
仍然是有效代码。当然,在您的示例中,it
在语句末尾无效。
修改临时值或 xvalue 有什么意义?这可能是范围访问器的设计者在提出声明时考虑的问题之一。他们没有考虑 .begin()
和 .end()
返回的迭代器在其生命周期结束后仍然有效的“代理”范围;也许正因为如此,在模板代码中,它们无法与正常范围区分开来 - 我们当然不想修改临时的非代理范围,因为那毫无意义并且可能会导致混淆。
但是,您不需要首先使用 std::begin
,而是可以使用 using 声明来声明它们:
using std::begin;
using std::end;
并使用 ADL。通过这种方式,您可以为 Boost.Python(o.s.)使用的类型声明 namespace 范围的 begin
和 end
重载,并规避 std::begin< 的限制
。例如
iterator begin(boost_slice&& s) { return s.begin(); }
iterator end (boost_slice&& s) { return s.end() ; }
// […]
begin(some_slice) // Calls the global overload, returns non-const iterator
Why have two overloads (one for const and one for non-const)?
因为我们仍然希望支持右值对象(它们不能被 T&
形式的函数参数接受)。
关于c++ - std::begin 和 R 值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26959953/
我正在开发一个小型图书馆,我需要做的一件事是让访问者访问一些数据并返回结果。 在一些较旧的 C++ 代码中,访问者需要声明一个 typedef return_type .例如,boost::stati
我正在尝试使用std:map类型的键和值制作std::any Visual Studio 2017 std::map m("lastname", "Ivanov"); std::cout (m["la
我已经在 C++ 的 map 中声明了一个集合为 std::map> .如何循环访问或打印设定值? 最佳答案 如果你知道如何迭代 std::map或 std::set单独地,您应该可以毫无问题地组合迭
如何循环? 我已经试过了: //----- code std::vector >::iterator it; for ( it = users.begin(); it != users.end();
我有两个用例。 A.我想同步访问两个线程的队列。 B.我想同步两个线程对队列的访问并使用条件变量,因为其中一个线程将等待另一个线程将内容存储到队列中。 对于用例 A,我看到了使用 std::lock_
我正在查看这两种类型特征的文档,但不确定有什么区别。我不是语言律师,但据我所知,它们都适用于“memcpy-able”类型。 它们可以互换使用吗? 最佳答案 不,这些术语不能互换使用。这两个术语都表示
我有以下测试代码,其中有一个参数 fS,它是 ofstream 的容器: #include #include #include #include int
这是这个问题的延续 c++ function ptr in unorderer_map, compile time error 我试图使用 std::function 而不是函数指针,并且只有当函数是
std::unordered_map str_bool_map = { {"a", true}, {"b", false}, {"c", true} }; 我们可以在此映射上使
我有以下对象 std::vector> vectorList; 然后我添加到这个使用 std::vector vec_tmp; vec_tmp.push_back(strDRG); vec_tmp.p
为什么 std::initializer_list不支持std::get<> , std::tuple_size和 std::tuple_element ?在constexpr中用得很多现在的表达式,
我有一个像这样定义的变量 auto drum = std::make_tuple ( std::make_tuple ( 0.3f , Ex
假设我有一个私有(private)std::map在我的类(class)里std::map 。我怎样才能将其转换为std::map返回给用户?我想要下面的原型(prototype) const std
假设我有一个私有(private)std::map在我的类(class)里std::map 。我怎样才能将其转换为std::map返回给用户?我想要下面的原型(prototype) const std
问题 我正在尝试将 lambda 闭包传递给 std::thread,它使用任意封闭参数调用任意封闭函数。 template std::thread timed_thread(Function&& f
我想创建一个模板类,可以容纳容器和容器的任意组合。例如,std::vector或 std::map ,例如。 我尝试了很多组合,但我必须承认模板的复杂性让我不知所措。我编译的关闭是这样的: templ
我有一个 std::vector>我将其分配给相同类型的第二个 vector 。 我收到这个编译器错误: /opt/gcc-8.2.0/include/c++/8.2.0/bits/stl_algob
有时候,我们有一个工厂可以生成一个 std::unique_ptr vector ,后来我们想在类/线程/你命名的之间共享这些指针。因此,最好改用 std::shared_ptr 。当然有一种方法可以
这个问题在这里已经有了答案: Sorting a vector of custom objects (14 个答案) 关闭 6 年前。 我创建了一个 vector vector ,我想根据我定义的参
我有三个类(class)成员: public: std::vector > getObjects(); std::vector > getObjects() const; privat
我是一名优秀的程序员,十分优秀!