(九)枚举器和迭代器(1)

news2024/11/27 12:33:33

一、枚举器和可枚举类型

复习完了数组之后,由于数组遍历的这个行为,跟枚举器有很大的相关性,所以接下来继续要学习与枚举器相关的内容。

1、使用 foreach 语句

  int[] arr1 = { 10, 11, 12, 13 };

            foreach (int item in arr1)//枚举元素
                Console.WriteLine($"Item value:{ item }");

问:为什么 foreach 语句能够遍历数组(或集合)中的元素?

答:

  • 1)使用 foreach 语句可以遍历数组中的元素,是因为 foreach 语句中使用了一个枚举器对象。
  • 2)枚举器通过获知数组中当前项的次序并跟踪(记录)它在序列中的位置。(枚举器中有 current 值来保存数组被访问到的当前项)

枚举器与可枚举类型的关系:

可枚举类型: 可以支持被遍历访问的类型,比如数组、集合或字符串等
枚举器: 在执行 foreach 语句中,可以通过调用可枚举类型对象(即可枚举类型对象)的 GetEnumertator 方法来获取枚举器。

关系: 可以通过枚举器来对可枚举类型中的元素进行遍历访问。

这些概念说起来可能有点绕,以下就举些例子来理解一下。

假如我要声明定义一个数组,并遍历其元素:

       //一维数组
            int[] arr = { 10, 11, 12, 13 };
         
            foreach(var item in arr)
            {
                Console.Write($"{item }  ");
            }

输出结果:

10 11 12 13

咱们先不理 foreach 语句这个代码块,要直接获取 GetEnumertator 方法来实现数组遍历,把代码改为:

 //一维数组
            int[] arr = { 10, 11, 12, 13 };

            //获取枚举器
            var arrEnumerator = arr.GetEnumerator();

            // arr 在没访问之前,枚举器里并没有记录 arr 的任何元素位置。
            arrEnumerator.MoveNext();//枚举器开始检索第1个元素次序(位置)
            Console.Write($"{arrEnumerator.Current }  ");

            arrEnumerator.MoveNext();//枚举器检索第2个元素次序(位置)
            Console.Write($"{arrEnumerator.Current }  ");

            arrEnumerator.MoveNext();//枚举器检索第3个元素次序(位置)
            Console.Write($"{arrEnumerator.Current }  ");

            arrEnumerator.MoveNext();//枚举器最后检索第4个元素次序(位置)
            Console.Write($"{arrEnumerator.Current }  ");

输出结果与 foreach 语句遍历的结果一样:

10 11 12 13

也就是说,执行 foreach 语句遍历的行为,实际上是通过获取枚举器,然后枚举器对可枚举类型进行反复操作处理的行为。

再回到上一个例子中的 foreach 语句来分析以下:

   int[] arr = { 10, 11, 12, 13 };
   
     //1、遍历之前,通过 arr 里的 GetEnumertator 方法获取枚举器
     //2、通过枚举器来移动访问元素的位置,并把元素值和位置都存在枚举器对象里。
     //3、每遍历一次,就重复第2个步骤,直到最后遍历结束
     //4、释放枚举器对象
     //跳出 foreach 语句
     
            foreach(var item in arr)//枚举元素,枚举的元素也称为迭代变量
            {
                Console.Write($"{item }  ");
            }

//arr:为可枚举类型
//item:为枚举的元素,迭代变量

因此,foreach 实际上相当于对可枚举类型给和枚举器具体操作的行为进行“封装”了起来,当然 foreach 语句中还使用到了迭代器。(下一篇文章会讲到迭代器)

2、获取枚举器以及其类型:

   class  MyClass
    {
        public string name = "";
    }

    class Program
    {
        static void Main(string[] args)
        {
            //字符串
            string str = "123";

            //一维数组
            int[] arr = { 10, 11, 12, 13 };
            string[] strArr = { "123", "123" };
            MyClass[] myClassArr = { new MyClass(), new MyClass(), new MyClass(), new MyClass() };
            //矩阵数组
            string[,] strMatrixArr = { { "123", "123" }, { "456", "456" } };

            //交错数组
            string[][] strJaggedArr = new string[2][];
            strJaggedArr[0] = new string[] {  "1" ,   "1"  };
            strJaggedArr[1] = new string[] { "2", "2" };

            //集合
            List<int> IntList = new List<int>();
            IntList.Add(1);
            IntList.Add(2);
            IntList.Add(3);
            IntList.Add(4);
            IntList.Add(10);

            //字符串,获取 CharEnumerator 类
            var strEnumerator = str.GetEnumerator();

            //一维数组,获取 IEnumerator 接口
           var arrEnumerator = arr.GetEnumerator();
            var strArrEnumerator = strArr.GetEnumerator();
            var myClassArrEnumerator = myClassArr.GetEnumerator();

            //矩阵数组,获取 IEnumerator 接口
            var strMatrixArrEnumerator = strMatrixArr.GetEnumerator();

            //交错数组,获取 IEnumerator 接口
            var strJaggedArrEnumerator = strJaggedArr.GetEnumerator();

            //关于 IEnumerator 接口被谁继承:
            //通过 GetType 方法获取到名称为:SZArrayEnumerator 类
            var allArrEnumerator = arrEnumerator.GetType();
            
            //集合,获取 Enumerator 结构
            var IntListEnumerator = IntList.GetEnumerator();

            Console.ReadKey();
        }
    }

从上面例子中,我们可以获取到枚举器有类、结构和接口的类型,分别是 CharEnumerator 类、IEnumerator 接口 和 Enumerator 结构。

由例子可知:

  • 字符串为可枚举类型时,获取到的枚举器类型为与字符相关的类
  • 一维数组、矩阵数组 和 交错数组,即数组类型为可枚举类型时,获取到的枚举器类型为IEnumerator接口被继承的 Arraay 集合枚举器类型:SZArrayEnumerator 类
  • 集合为枚举类型时,获取到的枚举器类型为结构

目前调试的代码有三种枚举器类型,由于我不是很了解枚举器,所以不清楚除了这三种类型外,还有哪些其他类型。

以下分析一下这三种枚举器类型的区别:

        //(一)CharEnumerator 类
        //在源码中声明:
        public sealed class CharEnumerator : ICloneable, IEnumerator<char>, IEnumerator, IDisposable {...}
        //主要的公有成员:Current、MoveNext()、Reset()
        //主要的私有成员:currentElement、index、str

        //(二)IEnumerator 接口
        //在源码中声明:
        public interface IEnumerator {...}
        //SZArrayEnumerator 成员:
        //主要的公有成员(实现接口的公有成员):Current、MoveNext()、Reset()
        //主要的私有成员:_array、_endIndex、_index

        //(三)Enumerator 结构
        //在源码中声明:
        public struct Enumerator : IEnumerator<T>, IDisposable, IEnumerator { ...}
        //主要的公有成员:Current、MoveNext()
        //主要的私有成员:current、index、list、version
        

分析这三种枚举器类型的内容:

  • 枚举器都有 Current 和 MoveNext() 这两个公有成员,主要用来对数组或集合中改变下一个指向其某枚举对象(元素)的次序和位置,并重新获取 当前元素的值。(Current值)

  • 数组或集合初始化后,需要通过枚举器调用一次MoveNext(),才能访问到第一个元素。因为初始化后,枚举数默认为第一个元素之前的一个位置。

  • 集合没有 Reset(),意味着集合不能初始化设置枚举数,不能回到第一个元素之前的一个位置。(可能是因为集合初始化后,还可以动态增加、插入、删除、清空元素的功能,并不像数组那样初始化后是固定了内容长度。所以,能支持调用 Reset() 必须有一个条件:可枚举类型对象的长度在初始化后,必须是固定的。)

  • Current 只有 get 访问器,没法修改 Current 的值。所以在 foreach 语句中使用到枚举器,就以为在 foreach 遍历过程中不能修改其元素的值。(for 可以修改,是因为 for 并没有枚举器)

分析枚举器的私有成员:

//CharEnumerator 类 私有成员:
currentElement:当前项,即 存储的是 char 类型的值
index:当前次序(位置)
str:可枚举类型,字符串内容

//IEnumerator 接口 私有成员:
_array :可枚举类型,数组的所有元素
_endIndex,即最后元素的下一个次序(位置)
_index:当前元素的次序(位置)

// Enumerator 结构 私有成员:
current:当前项
index:当前次序(位置)
list:可枚举类型,集合的所有元素
version:不太清楚,但是调试代码后发现,其值等于集合所包含元素的总数

从以上的分析可知:

  • 枚举器里的核心成员: Current、MoveNext()、Reset()、endIndex
    • Current: 当前项
    • MoveNext(): 执行可枚举类型对象中的下一个元素的次序(位置)
    • Reset(): 初始化设置枚举数,即第一个元素的上一个位置。
    • endIndex: 对固定长度的可枚举类型对象有用,最后一个元素的下一个位置。

二、图-枚举器和可枚举类型概览:

请添加图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/647028.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

尚硅谷大数据Flink1.17实战教程-笔记01【Flink概述、Flink快速上手】

尚硅谷大数据技术-教程-学习路线-笔记汇总表【课程资料下载】视频地址&#xff1a;尚硅谷大数据Flink1.17实战教程从入门到精通_哔哩哔哩_bilibili 尚硅谷大数据Flink1.17实战教程-笔记01【Flink概述、Flink快速上手】尚硅谷大数据Flink1.17实战教程-笔记02【Flink部署】尚硅谷…

【JVM 监控工具】性能诊断--JProfiler的使用

文章目录 背景一、Java 性能诊断工具简介二、简单命令行工具三、图形化综合诊断工具JVisualvmJProfiler 四、分布式应用性能诊断五、IDEA中设置JProfilerJProfiler是什么功能安装使用生成快照配置VM运行程序 背景 性能诊断是软件工程师在日常工作中需要经常面对和解决的问题&a…

公司新来的阿里p8,看了我做的APP和接口测试,甩给了我这份文档

移动应用App已经渗透到每个人的生活、娱乐、学习、工作当中&#xff0c;令人激动、兴奋且具有创造性的各种App犹如雨后春笋般交付到用户手中。各类智能终端也在快速发布&#xff0c;而开发者对于全球移动设备的质量和性能却掌握甚少&#xff0c;App与设备的兼容性问题常常导致用…

【状态估计】基于卡尔曼滤波器的传感器直流电机驱动研究(Matlab代码、Simulink实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

Navicat 受邀出席 PostgreSQL 技术峰会,欢迎莅临我们的展台了解 Navicat 工具包如何提升你的工作效能

Navicat 受邀出席 PostgreSQL 技术峰会成都站&#xff0c;欢迎童鞋们莅临我们的展台。你有机会与我们的专家面对面交流&#xff0c;并了解实用的 Navicat 工具包如何帮助PostgreSQL用户&#xff08;应用开发人员、DBA、运维人员以及数据分析师&#xff09;有效地提升日常的工作…

串口控制小车(二次开发)

0.资料 项目工程文件夹 分文件原理 之前的代码 1.L9110S电机驱动模块demo 2.串口通信&#xff08;习题4&#xff1a;PC发送字符串指令给单片机&#xff09; 3.wifi模块&#xff08;串口中断代码优化&#xff09; 3.蓝牙模块 1.串口指令控制小车_分文件 1、和单片机的接…

MVC、MVP、MVVM:详解2

概述 MVC、MVP、MVVM 都是在 Android 开发中经常用到的架构思想&#xff0c;它们都是为了更好地分离代码、提高代码可复用性、方便维护等目的而设计的。下面对这三种架构思想进行简单的介绍和比较。 MVC MVC 架构是最早被使用的一种架构&#xff0c;它把程序分成了三个部分&…

CVPR 2023 首届视觉异常检测(Visual Anomaly and Novelty Detection,VAND)挑战赛 Zero-shot 赛道冠军

这篇文章主要介绍一下我们在 CVPR 2023 VAND Workshop 的挑战赛中所采用的模型和方案。在 Zero-shot 赛道中我们获得了冠军&#xff08;Winner&#xff09;&#xff0c;在 Few-shot 赛道中&#xff0c;我们获得了第四名&#xff08;Honorable Mentions&#xff09;。 题目&…

怎么安装anaconda?anaconda安装详解!

Anaconda Navigator 是 Anaconda 的图形化管理界面&#xff0c;点击它即可进入 Anaconda 的图像化管理界面。许多小伙伴可能被朋友或者小编安利过Anaconda而跃跃欲试&#xff0c;今天小编就将Anaconda安装详解分享给大家。心动的小伙伴们赶紧安装起来吧&#xff01; 下载 官方…

1分钟教你配置好你的python环境

欢迎来到我们的系列博客《Python360全景》&#xff01;在这个系列中&#xff0c;我们将带领你从Python的基础知识开始&#xff0c;一步步深入到高级话题&#xff0c;帮助你掌握这门强大而灵活的编程语法。无论你是编程新手&#xff0c;还是有一定基础的开发者&#xff0c;这个系…

不能发现BUG的测试用例不是好的测试用例吗?

一般情况下技术岗面试都需要经历面试和笔试部分&#xff0c;面试过程中主要采用问答的形式&#xff0c;一般没有完全固定的回答&#xff0c;主要是根据自己的工作经验应答面试官的问题&#xff0c;而笔试部分更注重基础知识以及问题的常规解决方案。下面IT技术宅男为大家整理了…

C++案例

目录 一、while循环猜数组 二、 水仙花数 三、for循环敲桌子游戏 四、99乘法表 五、一维数组--元素逆置 六、冒泡排序 七、封装一个函数--利用冒泡排序&#xff0c;实现对整型数组的升序排序 八、结构体嵌套结构体 九、结构体排序 一、while循环猜数组 说明&#x…

经验总结:13 条自动化测试框架设计原则

1.代码规范 测试框架随着业务推进&#xff0c;必然会涉及代码的二次开发&#xff0c;所以代码编写应符合通用规范&#xff0c;代码命名符合业界标准&#xff0c;并且代码层次清晰。特别在大型项目、多人协作型项目中&#xff0c;如果代码没有良好的规范&#xff0c;那么整个框…

使用@Schedule注解实现定时任务,多线程执行定时任务,Cron表达式详解

Schedule注解实现定时任务&#xff0c;多线程执行定时任务&#xff0c;Cron表达式详解 使用Schedule注解实现定时任务Scheduled注解多线程执行定时任务Cron表达式Cron中的通配符 使用Schedule注解实现定时任务 1、首先&#xff0c;在项目启动类上添加 EnableScheduling 注解&am…

Vue CLI 全局事件总线 消息的订阅与发布

3.10. 全局事件总线&#xff08;GlobalEventBus&#xff09; 一种可以在任意组件间通信的方式&#xff0c;本质上就是一个对象&#xff0c;它必须满足以下条件 所有的组件对象都必须能看见他这个对象必须能够使用$on $emit $off方法去绑定、触发和解绑事件 使用步骤 定义全…

MySQL数据库基础 12

第十二章 MySQL数据类型 1. MySQL中的数据类型2. 整数类型2.1 类型介绍2.2 可选属性2.2.1 M2.2.2 UNSIGNED2.2.3 ZEROFILL 2.3 适用场景2.4 如何选择&#xff1f; 3. 浮点类型3.1 类型介绍3.2 数据精度说明3.3 精度误差说明 4. 定点数类型4.1 类型介绍 5. 位类型&#xff1a;BI…

CSS--Java EE

在前端的代码中&#xff0c;CSS 相关的代码写在什么位置呢&#xff1f; CSS 可以写在<style>标签中外部引入&#xff1a;输入 link: css写在 div 标签中 目录 一、选择器的种类 1 基础选择器 1.1 类选择器 1.2 id选择器 1.3 标签选择器 1.4 通用选择器 小结 2 …

Spring Security6 全新写法,大变样!

文章目录 1. WebSecurityConfigurerAdapter2. 使用 Lambda3. 自定义 JSON 登录3.1 自定义 JSON 登录3.1.1 自定义登录过滤器3.1.2 自定义登录接口 3.2 原因分析3.3 问题解决 Spring Security 在最近几个版本中配置的写法都有一些变化&#xff0c;很多常见的方法都废弃了&#x…

Java-IO流基础知识

目录 1.File类与路径知识 1.File类 2.Java中的路径知识 3.创建File类的实例 4.File类的方法使用 5.File类使用的注意点 2.IO流知识 1.IO流原理 2.文件的读入 3.read()的重载方法&#xff1a;难点 4.文件的写出 1.写出的说明 2.写出操作的具体步骤 5.文件的复制&am…

测试工程师如何有效的编写bug报告?

为什么要求有效的缺陷报告&#xff1f; 缺陷报告是测试过程中最重要的部分&#xff0c;对产品的质量有较大的影响&#xff0c;是测试人员价值的终极体现。好的缺陷报告可以减少研发部门的二次缺陷率、提高研发修改缺陷的速度、提高测试部门的信用度、增强测试和研发部门的协作…