草庐IT

【OOP】C++自定义类型的迭代器

bljw-02 2023-03-28 原文


1. 什么时候需要使用自定义的迭代器?

常见、基本的数组类型

  • 对于常见、基本的数组类型,如:intchar ,我们可以简单地使用下标来进行遍历:

    int array[5] = {1,2,3,4,5};
    
    // 方法1:使用下标遍历
    for (int ind = 0; ind < 5; ++ind) {
        cout << array[ind] << " ";
    }
    

    也可以使用范围来进行遍历,达到和使用下标同样的遍历效果:

    // 方法2:使用范围遍历
    for (int &n: array) {
        cout << n << " ";
    }
    

回到顶部

STL 容器

  • 对于 STL 容器来说,如:vectorlist ,我们同样也可以使用下标和范围来进行遍历:

    vector<int> vec = {1,2,3,4,5};
    
    // 方法1:使用下标遍历
    for (int ind = 0; ind < vec.size(); ++ind) {
        cout << vec[ind] << " ";
    }
    
    // 方法2:使用范围遍历
    for (int &n: vec) {
        cout << n << " ";
    }
    

    除了以上两种方法,STL 容器还可以使用迭代器(iterator)进行遍历:

    vector<int>::iterator iter;
    
    // 方法3:使用迭代器遍历
    // .begin() 是指向 vector 首元素的位置的迭代器
    // .end() 是指向 vector 最后一个元素的位置的下一个位置的迭代器
    for (iter = vec.begin(); iter != vec.end(); ++iter) {
        cout << *iter << " ";
    }
    

回到顶部

自定义数据类型

  • 下面给出一个简单的自定义类:

    class Group {
    private:
    	vector<vector<int>> students_marks;
        ......
    }
    

    其中,students_marks 是一个用来记录学生若干次的分数的 vector,students_marks[0] 也是一个 vector,用来记录第一个学生的分数,其中 students_marks[0][0] 表示第一个学生的第一个分数。

  • 如果要对 students_marks 进行遍历,由于不存在 vector<int> 类型的迭代器,因此方法2和方法3无法使用,只能使用下标遍历法:

    // 定义一个函数,用来输出 vector 中所有元素
    void printvec(vector<int> &vec) {
        for(int ind = 0; ind < vec.size(); ++ind) {
            cout << vec[ind] << " ";
        }
    }
    
    // 使用下标对 vector<int> 类型的元素进行遍历
    for (int ind = 0; ind < students_marks.size(); ++ind) {
        printvec(students_marks[ind]);
    }
    
  • 如果要使用以下方法对 students_marks 进行遍历:

    // 定义 Group 类型的 G
    Group G;
    
    // 使用范围遍历
    for(auto v: G) {
        printvec(v);
    }
    

    则需要自己手动编写指向 vector<int> 数据类型的迭代器。

回到顶部


2. 开始编写自定义迭代器之前需要思考的问题

  1. 迭代器进行遍历的对象是什么?
  2. 迭代器遍历的范围是什么?
  3. 迭代器指向的数据是什么类型?

只要想明白这三个问题,就不难实现自定义类型的迭代器。

以上面的类 Group 为例,

我们想遍历的是 students_marks ,将每个学生的分数输出。

对应问题1,迭代器遍历的对象是 vector<vector<int>> students_marks

对应问题2,假设总共有 k 个学生,则 students_marks.size() = k ,遍历的范围是从 students_marks[0]students_marks[k-1]

对应问题3,迭代器指向的数据是 students_marks[x] ,类型是 vector<int>

回到顶部


3. 如何编写及实现自定义类型的迭代器?

在后面的步骤中,一些常用的名词将会以以下标识符代称:

代替的标识符 对应上面的例子是
遍历的对象 OBJECT students_marks
遍历对象的数据类型 OBJECT_type vector<vector<int>>
遍历对象所处类 OBJECT_class Group
迭代器指向的数据 VALUE students_marks[x]
迭代器指向的数据类型 VALUE_type vector<int>

回到顶部

  1. object 所处的 class 中定义一个迭代器(iterator)的结构体(struct):

    class OBJECT_class {
    private:
    	OBJECT_type OBJECT;
        
    public:
        struct Iterator {
            
        }
        
        ......
    }
    

  2. 设定迭代器的属性:

    struct Iterator {
        using iterator_category = std::forward_iterator_tag;
        using difference_type = std::ptrdiff_t;
        using value_type = VALUE_type;
        using reference = const VALUE_type&;
        using pointer = VALUE_type*;
    }
    
    • iterator_category 是迭代器的类型,如果是正常、基础的情况下,选择 forward_iterator_tag 就可以了
    • difference_type 选择 ptrdiff_t
    • 其余的 value_typereferencepointer 都是自定义的,根据实际情况填写

  3. 定义迭代器的成员变量、构造函数、基础函数

    struct Iterator {
        using iterator_category = std::forward_iterator_tag;
        using difference_type = std::ptrdiff_t;
        using value_type = VALUE_type;
        using reference = const VALUE_type&;
        using pointer = VALUE_type*;
        
        // 构造函数
        Iterator(pointer p) :ptr(p) {}
        
        // 拷贝赋值函数
        Iterator& operator=(const Iterator& it) {
            ptr = it.ptr;
        }
        
        // 等于运算符
        bool operator==(const Iterator& it) const {
            return ptr == it.ptr;
        }
        
        // 不等于运算符
        bool operator!=(const Iterator& it) const {
            return ptr != it.ptr;
        }
        
        // 前缀自加
        Iterator& operator++() {
            ptr++;
            return *this;
        }
        
        // 后缀自加
        Iterator operator ++(int) {
            Iterator tmp = *this;
            ++(*this);
            return tmp;
        }
        
        // 前缀自减
        Iterator& operator--() {
            ptr--;
            return *this;
        }
        
        // 后缀自减
        Iterator operator --(int) {
            Iterator tmp = *this;
            --(*this);
            return tmp;
        }
        
        // 取值运算
        VALUE_type& operator*() {
            return *ptr;
        }
        
    private:
        // 定义一个指针
        pointer ptr; 
    }
    
    • 以上的函数不是必要也不是完整的,如果不需要用到的可以不写,额外需要用到的函数可以自行编写。
    • 但是在大多数情况下,上面这些函数基本上足矣。

  4. 设定遍历的范围:

    class OBJECT_class {
    private:
    	OBJECT_type OBJECT;
        
    public:
        struct Iterator {
            
            ......
            ......
                
        private:
            pointer ptr;
            
        }
        
        // 遍历的第一个元素的位置
        Iterator begin() {
            VALUE_type* head = &OBJECT[0];
            return Iterator(head);
        }
        
        // 遍历的最后一个元素的下一个位置
        Iterator end() {
            VALUE_type* head = &OBJECT[0];
            return Iterator(head + OBJECT.size());
        }
        
        ......
    }
    
    • head 是指向第一个 VALUE 的指针

回到顶部

有关【OOP】C++自定义类型的迭代器的更多相关文章

  1. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  2. ruby-on-rails - Rails 3.2.1 中 ActionMailer 中的未定义方法 'default_content_type=' - 2

    我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer

  3. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

    我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

  4. ruby - 主要 :Object when running build from sublime 的未定义方法 `require_relative' - 2

    我已经从我的命令行中获得了一切,所以我可以运行rubymyfile并且它可以正常工作。但是当我尝试从sublime中运行它时,我得到了undefinedmethod`require_relative'formain:Object有人知道我的sublime设置中缺少什么吗?我正在使用OSX并安装了rvm。 最佳答案 或者,您可以只使用“require”,它应该可以正常工作。我认为“require_relative”仅适用于ruby​​1.9+ 关于ruby-主要:Objectwhenrun

  5. ruby - Infinity 和 NaN 的类型是什么? - 2

    我可以得到Infinity和NaNn=9.0/0#=>Infinityn.class#=>Floatm=0/0.0#=>NaNm.class#=>Float但是当我想直接访问Infinity或NaN时:Infinity#=>uninitializedconstantInfinity(NameError)NaN#=>uninitializedconstantNaN(NameError)什么是Infinity和NaN?它们是对象、关键字还是其他东西? 最佳答案 您看到打印为Infinity和NaN的只是Float类的两个特殊实例的字符串

  6. ruby - 检查方法参数的类型 - 2

    我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)

  7. ruby - 在 Ruby 中有条件地定义函数 - 2

    我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin

  8. ruby - 定义方法参数的条件 - 2

    我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano

  9. ruby - 如何在 Grape 中定义哈希数组? - 2

    我使用Ember作为我的前端和GrapeAPI来为我的API提供服务。前端发送类似:{"service"=>{"name"=>"Name","duration"=>"30","user"=>nil,"organization"=>"org","category"=>nil,"description"=>"description","disabled"=>true,"color"=>nil,"availabilities"=>[{"day"=>"Saturday","enabled"=>false,"timeSlots"=>[{"startAt"=>"09:00AM","endAt"=>

  10. ruby - Ruby 有 `Pair` 数据类型吗? - 2

    有时我需要处理键/值数据。我不喜欢使用数组,因为它们在大小上没有限制(很容易不小心添加超过2个项目,而且您最终需要稍后验证大小)。此外,0和1的索引变成了魔数(MagicNumber),并且在传达含义方面做得很差(“当我说0时,我的意思是head...”)。散列也不合适,因为可能会不小心添加额外的条目。我写了下面的类来解决这个问题:classPairattr_accessor:head,:taildefinitialize(h,t)@head,@tail=h,tendend它工作得很好并且解决了问题,但我很想知道:Ruby标准库是否已经带有这样一个类? 最佳

随机推荐