命名空间
变体
操作

std::ranges::views::concat, std::ranges::concat_view

来自 cppreference.cn
< cpp‎ | ranges
 
 
范围库
范围适配器
 
 
定义于头文件 <ranges>
template< ranges::input_range... Views >

    requires (ranges::view<Views> && ...) && (sizeof...(Views) > 0) &&
              /*concatable*/<Views...>
class concat_view

    : public ranges::view_interface<concat_view<Views...>>
(1) (自 C++26 起)
namespace views {

    inline constexpr /* unspecified */ concat = /* unspecified */;

}
(2) (自 C++26 起)
调用签名
template< ranges::viewable_range... Rs >

    requires /* see below */

constexpr ranges::view auto concat( Rs&&... rs );
(自 C++26 起)
辅助类型别名
template< class... Rs >

using /*concat-reference-t*/ =

    ranges::common_reference_t<ranges::range_reference_t<Rs>...>;
(3) (仅为说明目的*)
template< class... Rs >
using /*concat-value-t*/ = std::common_type_t<ranges::range_value_t<Rs>...>;
(4) (仅为说明目的*)
template< class... Rs >

using /*concat-rvalue-reference-t*/ =

    ranges::common_reference_t<ranges::range_rvalue_reference_t<Rs>...>;
(5) (仅为说明目的*)
辅助概念
template< class Ref, class RRef, class It >
concept /*concat-indirectly-readable-impl*/ = /* see description */;
(6) (仅为说明目的*)
template< class... Rs >
concept /*concatable*/ = /* see description */;
(7) (仅为说明目的*)

concat_view 提供一个 view 工厂,它接受任意数量的范围作为参数列表,并提供一个视图,该视图从第一个范围的第一个元素开始,到最后一个范围的最后一个元素结束,所有范围元素按参数中给出的顺序依次排列,有效地连接或链接在一起参数范围。

1) 类模板,其模板参数是非空的 view 包,每个视图都至少是 input_range 的模型,并且是 concatable (7)
2) views::concat 是一个定制点对象。

给定子表达式包 exprs,表达式 views::concat(exprs...) 表达式等价于

  • views::all(exprs...) 如果 exprs 是一个只包含一个元素的包,且该元素的类型是 input_range 的模型,
  • concat_view(exprs...) 否则。
3) 表示引用类型。需要额外的约束来确保每个底层范围的 ranges::range_reference_t 可转换为 ranges::common_reference_t
4) iterator::value_type,它还考虑了底层范围的 value_type,以支持底层范围具有代理迭代器的情况。
5) rvalue 引用,它也正确地支持了底层迭代器自定义 iter_move 的情况。
6)iterator 定义了 indirectly-readable 概念,以便 concat_view 可以作为 input_range 的模型。
等价于
template< class... Rs >
concept /*concat-indirectly-readable*/ = // exposition only
    std::common_reference_with</*concat-reference-t*/<Rs...>&&,
                               /*concat-value-t*/<Rs...>&> &&
    std::common_reference_with</*concat-reference-t*/<Rs...>&&,
                               /*concat-rvalue-reference-t*/<Rs...>&&> &&
    std::common_reference_with</*concat-rvalue-reference-t*/<Rs...>&&,
                               /*concat-value-t*/<Rs...> const&> &&
    (/*concat-indirectly-readable-impl*/</*concat-reference-t*/<Rs...>,
                                         /*concat-rvalue-reference-t*/<Rs...>,
                                         ranges::iterator_t<Rs>> && ...);
其中仅用于说明的概念 /*concat-indirectly-readable-impl*/
template< class Ref, class RRef, class It >
concept /*concat-indirectly-readable-impl*/ = // exposition only
    requires(const It it) {
        { *it } -> std::convertible_to<Ref>;
        { ranges::iter_move(it)} -> std::convertible_to<RRef>;
    };
7) 确定是否可以将任意两个或多个不同的范围适配成一个本身是范围的序列。等价于
template< class... Rs >
concept /*concatable*/ = requires { // exposition only
        typename /*concat-reference-t*/<Rs...>;
        typename /*concat-value-t*/<Rs...>;
        typename /*concat-rvalue-reference-t*/<Rs...>;
    } && /*concat-indirectly-readable*/<Rs...>;

concat_view 始终作为 input_range 的模型,并且如果每个适配的 view 类型都作为相应的概念的模型,则作为 forward_rangebidirectional_rangerandom_access_rangesized_range 的模型。

如果最后一个底层范围作为 common_range 的模型,则 concat_view 可以是 common_range

目录

定制点对象

名称 views::concat 表示一个定制点对象,它是一个 const 函数对象,类型为 字面 semiregular 类。为了说明目的,其类型的 cv 非限定版本表示为 __concat_fn

__concat_fn 的所有实例均相等。在相同参数上调用 __concat_fn 类型的不同实例的效果是等效的,无论表示实例的表达式是左值还是右值,以及是否具有 const 限定符(但是,不要求可调用 volatile 限定的实例)。因此,views::concat 可以自由复制,并且其副本可以互换使用。

给定一组类型 Args...,如果 std::declval<Args>()... 满足上述 views::concat 参数的要求,则 __concat_fn 作为以下模型的模型:

否则,__concat_fn 的任何函数调用运算符都不参与重载解析。

[编辑] 数据成员

成员 描述
std::tuple<Views...> views_ 所有适配的视图对象
(仅为说明目的的成员对象*)

[编辑] 成员函数

构造一个 concat_view
(公有成员函数) [编辑]
返回指向开头的迭代器
(公有成员函数) [编辑]
返回指向结尾的迭代器或哨位
(公有成员函数) [编辑]
返回元素的数量,仅当底层(适配的)范围满足 sized_range 时提供
(公有成员函数) [编辑]
继承自 std::ranges::view_interface
返回导出的视图是否为空,仅当它满足 sized_rangeforward_range 时提供
(std::ranges::view_interface<D> 的公有成员函数) [编辑]
(C++23)
返回指向范围开头的常量迭代器
(std::ranges::view_interface<D> 的公有成员函数) [编辑]
(C++23)
返回范围的常量迭代器的哨位
(std::ranges::view_interface<D> 的公有成员函数) [编辑]
返回导出的视图是否非空,仅当 ranges::empty 适用于它时提供
(std::ranges::view_interface<D> 的公有成员函数) [编辑]
返回导出视图中的第一个元素,如果它满足 forward_range 则提供
(std::ranges::view_interface<D> 的公有成员函数) [编辑]
返回导出视图中的最后一个元素,仅当它满足 bidirectional_rangecommon_range 时提供
(std::ranges::view_interface<D> 的公有成员函数) [编辑]
返回导出视图中的第 n 个元素,仅当它满足 random_access_range 时提供
(std::ranges::view_interface<D> 的公有成员函数) [编辑]

[编辑] 推导指引

[编辑] 嵌套类

类名 定义
迭代器类型
(仅为说明目的的成员类模板*)

[编辑] 辅助模板

concat_view 没有 ranges::enable_borrowed_range 的特化,因为这将要求迭代器实现始终包含所有底层范围的所有迭代器和哨位的副本。

[编辑] 注解

无参数 views::concat() 是非良构的,因为没有合理的方法来确定元素类型 T。单参数 views::concat(r) 表达式等价于 views::all(r)

特性测试 Std 特性
__cpp_lib_ranges_concat 202403L (C++26) std::ranges::concat_view

[编辑] 示例

可以在 Compiler Explorer 上查看初步版本。

#include <cassert>
#include <list>
#include <print>
#include <ranges>
#include <vector>
 
int main()
{
    std::vector<int> v0{1, 2, 3}, v1{4, 5};
    int a[]{6, 7};
    int i{8};
    auto ie{std::views::single(i)};
 
    auto con = std::views::concat(v0, v1, a, ie);
    assert(con.size() == v0.size() + v1.size() + std::size(a) + ie.size());
    std::println("con.size(): {}", con.size());
    std::println("con: {}", con);
    con[6] = 42; // con is random_access_range, operator[] returns a reference
    assert(a[1] == 42); // a[1] was modified via con[6]
    std::println("con: {}", con);
 
    std::list<int> l{7, 8}; // list is bidirectional range
    auto cat = std::views::concat(v0, l);
    std::println("cat: {}", cat);
    // cat[0] = 13; // compile-time error: cat is bidirectional => no operator[]
}

输出

con.size(): 8
con: [1, 2, 3, 4, 5, 6, 7, 8]
con: [1, 2, 3, 4, 5, 6, 42, 8]
cat: [1, 2, 3, 7, 8]

[编辑] 参考

  • C++26 标准 (ISO/IEC 14882:2026)
  • 26.7.18 Concat view [range.concat]

[编辑] 参见

一个 view,由展平 viewrange 序列获得
(类模板) (范围适配器对象)[编辑]
一个 view,由展平 range 的 view 获得的序列组成,元素之间带有分隔符
(类模板) (范围适配器对象)[编辑]
一个 view,由指向适配视图的相应元素的引用元组组成
(类模板) (定制点对象)[编辑]
一个 view,由通过适配视图的 n 元笛卡尔积计算的结果元组组成
(类模板) (定制点对象)[编辑]