当前位置: 首页 > 知识库问答 >
问题:

用户定义的容器无法使用std::ranges

莘睿
2023-03-14

我无法让我的用户定义的容器使用std::ranges。如果迭代器只是一个< code>int*的话,我的容器可以工作,但是一旦我创建了自己的迭代器类,就会出现编译器错误。

这是有效的。

#include <iostream>
#include <ranges>
using namespace std;

struct sentinel {};

class my_iota_view_a : public std::ranges::view_interface< my_iota_view_a >{
public:
    using value_type        = size_t;
    using iterator          = value_type *;

    my_iota_view_a() = default;
    my_iota_view_a( size_t begin ) : mBegin( &mData[begin] ) { }

    iterator begin() { return mBegin; }
    sentinel end() { return sentinel{}; }
private:
    value_type mData[10] ={ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };
    value_type * mBegin;
};

bool operator==( my_iota_view_a::iterator lhs, sentinel rhs ) { return false; }
bool operator!=( my_iota_view_a::iterator lhs, sentinel rhs ) { return true; }

int main() {
    for( auto i :  my_iota_view_a( 1 ) | ranges::views::take(3) )  
        cout << i << ' ';
    cout << "\n";
}

https://godbolt.org/z/nxr1qe

使用我自己的迭代器,它不起作用。

#include <iostream>
#include <ranges>
using namespace std;

struct sentinel {};

class my_iota_view_b : public std::ranges::view_interface< my_iota_view_b > {
public:
    struct iterator_type {
        using iterator_category = std::output_iterator_tag;
        using value_type        = size_t;
        using reference         = value_type; // Intentional not 'value_type &' because 
                                              // this iterator modifies the value.
        using pointer           = value_type*;
        using difference_type   = std::ptrdiff_t;

        iterator_type() = default;
        iterator_type( value_type val ) : mVal( val ) {}
        iterator_type( const iterator_type & o ) : mVal(o.mVal) {}
        iterator_type( iterator_type && o ) noexcept : mVal( std::move( o.mVal ) ) {}

        iterator_type operator++( int ) /* Postfix a++ */ { auto it = *this; ++it; return it; }
        iterator_type operator++() /* Prefix ++a */{ mVal++; return *this; }

        reference operator*() { return mVal; }
    
        value_type mVal{};
    };

    struct sentinel {};

    friend bool operator==( iterator_type lhs, sentinel rhs ) { return lhs.mVal > 1000; }
    friend bool operator!=( iterator_type lhs, sentinel rhs ) { return !(lhs == rhs); }

    using value_type        = size_t;
    using iterator          = iterator_type;
    using const_iterator    = iterator_type;

    my_iota_view_b() = default;
    my_iota_view_b( value_type begin ) : mBegin( begin ) {}

    iterator begin() { return mBegin; }
    sentinel end() { return sentinel{}; }

    size_t size() { return 10; }
private:
    value_type mBegin{};
};

bool operator==( my_iota_view_b::iterator lhs, sentinel rhs ) { return false; }
bool operator!=( my_iota_view_b::iterator lhs, sentinel rhs ) { return true; }

int main() {
    for( auto i :  my_iota_view_b( 1 ) | ranges::views::take(3) )  // *1
        cout << i << ' ';

    for( auto i :  my_iota_view_b( 1 )  )  // An old school range based for works.
        cout << i << ' ';
}

// Line *1 error: no match for 'operator|' (operand types are 'my_iota_view_b' and ranges::views::take(3)

https://godbolt.org/z/o9rzhf

为了实现这一点,是否需要为<code>std::ranges</code>做一些新的工作?

我正在使用Visual Studio 16.8.4(编写时的最新版本)进行编译,编译错误为

error C2678: binary '|': no operator found which takes a left-hand operand of type 'my_iota_view_b' (or there is no acceptable conversion)

但海湾合作委员会给出了同样的错误。

我已经找到了如何创建范围视图的例子,但是没有一个向我展示如何创建自己的迭代器。

共有1个答案

阎丰
2023-03-14

问题是您的迭代器不是迭代器。回答这个问题的最佳方法是使用静态断言:

static_assert(std::input_iterator<my_iota_view_b::iterator>);

如果你这样做,你会看到你的迭代器无法满足的所有事情。第一个是它不是可移动的。这是因为您提供了一个移动构造函数 - 您不应该这样做,让编译器为您生成它们。

删除移动和复制构造函数后,您将看到下一个错误,即前缀增量不返回引用,它应该返回。

然后< code >运算符*()不是< code>const,它需要是。

然后你的iterator_categoryoutput_iterator_tag,而它应该是forward_iterator_tag(甚至完全不存在)。

一旦你解决了所有这些问题,那么一切都会按预期进行。

 类似资料:
  • 我正在尝试使用@modeldattribute绑定数据。我有两个名为Address和Student的用户定义类。这两个数据类的结构如下所示: 住址爪哇: 大学生爪哇: 使用@modeldattribute,我试图绑定Student对象,但仅绑定String和int属性,而地址依赖项仅为null。 网状物xml 学生控制员。JAVA 接纳表格。jsp

  • 我已经执行了操作员的命令 头文件: cpp文件: 这对像这样的操作很有效 文件 ,但当用作 库特 结果有: 错误:无法将'std::ostream{aka std::basic_ostream}'左值绑定到'std::basic_ostream 编辑:替换为std::ostream

  • 我使用SWIG类型映射包装一个类,使用SWIG提供的“std_vector.I”包装这个类的。目标语言是Python。除了我不能在向量上迭代之外,一切似乎都正常。 我创建了一个关于复数的最小示例(仅为方便起见,它与我的真实项目无关)。 这是我想包装的C类: 这是我的SWIG接口文件: 我可以使用以下Python代码简单地测试typecheck/in/out typemaps: 正如您在输出中看到的

  • 我的玻璃鱼在我尝试跑步时抛出一个异常。 [2018-02-22T17:07:04.135 0100][glassfish 5.0][Severy][javax.enterprise.system.core][tid:_ThreadID=46 _ThreadName=admin listener(4)][timeMillis:1519315624135][levelValue:1000][加载应用时

  • 问题内容: 我已经读过很多关于servlet容器之类的Java容器的信息,但是,我似乎找不到企业Java世界中容器的正确定义。 有谁知道企业Java容器的良好定义? 问题答案: M.Volter等人撰写的《服务器组件模式》一书更笼统地涉及了Container模式(企业Java容器可以看作是其专业化)。提供以下内容: [一个容器提供]一个执行环境,负责将技术关注点添加到组件中…从概念上讲,它包装了组

  • 找到价格最高和最低的信息: mysql> SELECT @min_price:=MIN(price),@max_price:=MAX(price) FROM shop; mysql> SELECT * FROM shop WHERE [email protected]_price OR [email protected]_price; +---------+--------+-------+ |