C# 迭代,递归,回调--13

目录

一.迭代

迭代器示例:

关键点:

优势:

二.递归

递归示例:

关键点:

优势:

注意:

三.回调

回调示例:

关键点:

优势:

应用场景:

4.三种模式的特点对比:

迭代:

递归:

回调:


一.迭代

在C#中迭代通常指重复执行一系列指令

在C#中,迭代器是一种特殊的结构,允许我们自定义遍历集合的方式,主要使用yield return关键字

yield return关键字的主要优点包括:

  • 简化了迭代器的实现:不需要手动维护状态或创建临时集合
  • 提高了代码的可读性和可维护性:使用 yield return 的方法显得更加直观
  • 延迟执行:元素在需要时才生成,适合处理大量数据或计算密集型操作

除了 yield return,还有一个相关的关键字 yield break,用于终止迭代器并退出迭代

迭代器示例:

class Program
{static void Main(string[] args){Console.WriteLine("迭代器示例:\n");// 1. 使用迭代器方法Console.WriteLine("1. 斐波那契数列(前10个数):");foreach (var num in Fibonacci(10)){Console.Write($"{num} ");}Console.WriteLine("\n");// 2. 自定义集合迭代var customCollection = new CustomCollection(5);Console.WriteLine("2. 自定义集合迭代:");foreach (var item in customCollection){Console.Write($"{item} ");}Console.WriteLine("\n");// 3. yield return 示例Console.WriteLine("3. 生成偶数序列(0-10):");foreach (var even in GetEvenNumbers(10)){Console.Write($"{even} ");}Console.WriteLine("\n");// 4. 带条件的迭代器Console.WriteLine("4. 按条件过滤的数字(1-20中能被3整除的数):");foreach (var num in GetNumbersDivisibleBy3(20)){Console.Write($"{num} ");}Console.WriteLine("\n");Console.ReadKey();}// 斐波那契数列迭代器static IEnumerable<int> Fibonacci(int count){int current = 0, next = 1;for (int i = 0; i < count; i++){yield return current;int temp = current + next;current = next;next = temp;}}// 生成偶数的迭代器static IEnumerable<int> GetEvenNumbers(int max){for (int i = 0; i <= max; i++){if (i % 2 == 0)yield return i;}}// 能被3整除的数的迭代器static IEnumerable<int> GetNumbersDivisibleBy3(int max){for (int i = 1; i <= max; i++){if (i % 3 == 0)yield return i;}}
}// 自定义可迭代集合
class CustomCollection : IEnumerable<int>
{private int[] array;public CustomCollection(int size){array = new int[size];for (int i = 0; i < size; i++){array[i] = i * i; // 存储数字的平方}}public IEnumerator<int> GetEnumerator(){for (int i = 0; i < array.Length; i++){yield return array[i];}}System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator(){return GetEnumerator();}
}

关键点:

  • 循环结构: 在C#中,迭代通常通过forwhiledo...whileforeach等循环结构实现

  • 用途: 用于在数量已知或者条件可判定的情况下重复执行某段代码,例如遍历数组、执行固定次数的计算等

  • 控制条件: 迭代需要有明确的开始条件、结束条件,以及每次迭代时的状态更新,避免出现无限循环

优势:

  • 可控性强: 通过明确的控制条件,可以精确地控制迭代次数
  • 效率高: 迭代通常比递归占用更少的内存,因为不涉及函数调用的堆栈开销

二.递归

递归是指函数在其定义中直接或间接调用自身的编程技巧

递归通常用于解决可以分解为相同子问题的问题,如阶乘,斐波那契数列等

递归示例:

class Program
{static void Main(string[] args){Console.WriteLine("递归示例演示:\n");// 1. 阶乘计算int n = 5;Console.WriteLine($"1. {n}的阶乘是: {CalculateFactorial(n)}");// 2. 斐波那契数列int position = 8;Console.WriteLine($"\n2. 斐波那契数列第{position}个数是: {Fibonacci(position)}");// 3. 数组求和int[] numbers = { 1, 2, 3, 4, 5 };Console.WriteLine($"\n3. 数组求和结果: {ArraySum(numbers, numbers.Length - 1)}");// 4. 字符串反转string text = "Hello, World!";Console.WriteLine($"\n4. 字符串 \"{text}\" 反转后: {ReverseString(text)}");// 5. 最大公约数int a = 48, b = 36;Console.WriteLine($"\n5. {a}和{b}的最大公约数是: {GCD(a, b)}");// 6. 汉诺塔问题Console.WriteLine("\n6. 汉诺塔移动步骤(3个盘子):");HanoiTower(3, 'A', 'B', 'C');// 7. 目录结构显示Console.WriteLine("\n7. 显示目录结构:");string path = @"C:\Example"; // 替换为实际路径try{ShowDirectoryStructure(path, 0);}catch (Exception ex){Console.WriteLine($"读取目录时出错: {ex.Message}");}Console.ReadKey();}// 1. 计算阶乘static int CalculateFactorial(int n){if (n <= 1) return 1;return n * CalculateFactorial(n - 1);}// 2. 斐波那契数列static int Fibonacci(int n){if (n <= 1) return n;return Fibonacci(n - 1) + Fibonacci(n - 2);}// 3. 数组求和static int ArraySum(int[] array, int index){if (index < 0) return 0;return array[index] + ArraySum(array, index - 1);}// 4. 字符串反转static string ReverseString(string str){if (string.IsNullOrEmpty(str) || str.Length <= 1)return str;return ReverseString(str.Substring(1)) + str[0];}// 5. 最大公约数(欧几里得算法)static int GCD(int a, int b){if (b == 0) return a;return GCD(b, a % b);}// 6. 汉诺塔问题static void HanoiTower(int n, char from, char auxiliary, char to){if (n == 1){Console.WriteLine($"将盘子 1 从 {from} 移动到 {to}");return;}HanoiTower(n - 1, from, to, auxiliary);Console.WriteLine($"将盘子 {n} 从 {from} 移动到 {to}");HanoiTower(n - 1, auxiliary, from, to);}// 7. 显示目录结构static void ShowDirectoryStructure(string path, int level){// 添加缩进string indent = new string(' ', level * 2);try{// 显示当前目录DirectoryInfo dir = new DirectoryInfo(path);Console.WriteLine($"{indent}[{dir.Name}]");// 显示文件foreach (FileInfo file in dir.GetFiles()){Console.WriteLine($"{indent}  {file.Name}");}// 递归显示子目录foreach (DirectoryInfo subDir in dir.GetDirectories()){ShowDirectoryStructure(subDir.FullName, level + 1);}}catch (UnauthorizedAccessException){Console.WriteLine($"{indent}访问被拒绝");}}
}

关键点:

  • 基例(终止条件): 递归必须要有一个明确的终止条件,即基例,否则会导致无限递归

  • 递归关系: 问题的解可以表示为其子问题的解的组合

  • 堆栈开销: 每一次递归调用都会在调用栈中存储当前函数的状态,深入太深可能导致栈溢出

优势:

  • 代码简洁: 对于某些问题,递归能使代码更易读、更简洁
  • 自然适应某些算法: 如树的遍历、分治算法等,递归实现更符合逻辑

注意:

  • 效率问题: 递归可能导致大量的函数调用,增加开销。
  • 栈溢出风险: 如果递归深度太大,可能导致栈溢出错误。

三.回调

回调是指将一个函数作为参数传递给另一个函数,当特定事件发生或条件满足时调用该函数

在C#中,回调通常通过委托事件来实现

回调示例:

class Program
{// 定义委托类型public delegate void ProcessCompleted(string result);public delegate int CalculateDelegate(int x, int y);static void Main(string[] args){Console.WriteLine("回调示例:\n");// 1. 使用委托回调Console.WriteLine("1. 委托回调示例:");ProcessCompleted callback = ShowResult;ProcessWithCallback("任务1", callback);// 2. 使用Action/FuncConsole.WriteLine("\n2. Action/Func回调示例:");ProcessWithAction("任务2", (result) => {Console.WriteLine($"Lambda回调结果: {result}");});// 3. 事件回调var processor = new TaskProcessor();processor.OnCompleted += (sender, result) => {Console.WriteLine($"事件回调结果: {result}");};processor.StartProcess("任务3");// 4. 异步回调Console.WriteLine("\n4. 异步回调示例:");AsyncProcessDemo().Wait();// 5. 计算器回调示例Console.WriteLine("\n5. 计算器回调示例:");CalculateWithCallback(10, 5, Add);CalculateWithCallback(10, 5, Subtract);Console.ReadKey();}// 基本回调方法static void ShowResult(string result){Console.WriteLine($"普通回调结果: {result}");}// 使用委托的处理方法static void ProcessWithCallback(string input, ProcessCompleted callback){// 模拟处理过程Thread.Sleep(100);string result = $"处理完成: {input}";callback(result);}// 使用Action的处理方法static void ProcessWithAction(string input, Action<string> callback){Thread.Sleep(100);string result = $"处理完成: {input}";callback(result);}// 异步回调示例static async Task AsyncProcessDemo(){await ProcessAsync("异步任务", (result) =>{Console.WriteLine($"异步回调结果: {result}");});}static Task ProcessAsync(string input, Action<string> callback){return Task.Run(() =>{Thread.Sleep(100);string result = $"处理完成: {input}";callback(result);});}// 计算器回调方法static int Add(int x, int y) => x + y;static int Subtract(int x, int y) => x - y;static void CalculateWithCallback(int x, int y, CalculateDelegate callback){int result = callback(x, y);Console.WriteLine($"计算结果: {result}");}
}// 使用事件的处理器类
class TaskProcessor
{public event EventHandler<string> OnCompleted;public void StartProcess(string input){// 模拟处理过程Thread.Sleep(100);string result = $"处理完成: {input}";// 触发事件OnCompleted?.Invoke(this, result);}
}

关键点:

  • 委托(Delegate): 在C#中,回调通常通过委托实现,委托是对函数的引用,可以作为参数传递

  • 事件(Event): 事件是委托的特殊形式,用于发布和订阅机制,常用于GUI编程、异步操作等

  • 异步编程: 回调在异步编程中非常重要,可以在异步操作完成后执行后续处理

优势:

  • 解耦: 回调机制使得调用者和被调用者之间的依赖减少,代码更加灵活
  • 灵活性: 可以在运行时决定调用哪个函数,增强了代码的可扩展性

应用场景:

  • 事件处理: 例如按钮点击事件,网络请求完成事件等
  • 异步操作: 在异步编程中,回调函数用于在任务完成后继续执行后续逻辑

4.三种模式的特点对比:

迭代:

        特点:

  • 逐个处理元素
  • 内存效率高
  • 代码直观

        适用场景:

  • 集合遍历
  • 简单重复操作
  • 需要延迟计算

递归:

        特点:

  • 自调用
  • 代码简洁
  • 需要注意栈溢出

        适用场景:

  • 树形结构处理
  • 分治算法
  • 自然递归问题

回调:

        特点:

  • 灵活可配置
  • 解耦操作
  • 支持异步

        适用场景:

  • 事件处理
  • 异步操作
  • 策略模式

策略模式(Strategy Pattern)是一种行为设计模式,它定义了一系列算法,并将每个算法封装起来,使它们可以互相替换。策略模式使得算法可以在不影响客户端的情况下发生变化。它通过定义一系列算法,把它们一个个封装起来,并且使它们可以相互替换,从而让算法的变化不会影响到使用算法的客户。

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

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

相关文章

炸砖块游戏的最终图案

描述 小红正在玩一个“炸砖块”游戏,游戏的规则如下:初始有一个 n * m 的砖块矩阵。小红会炸 k 次,每次会向一个位置投炸弹,如果这个位置有一个砖块,则砖块消失,上方的砖块向下落。小红希望你画出最终砖块的图案。 输入描述 第一行输入三个正整数 n, m, k,代表矩阵的行…

【Python项目】图像信息隐藏技术的实现

【Python项目】图像信息隐藏技术的实现 技术简介&#xff1a;采用Python技术、MYSQL数据库等实现。 系统简介&#xff1a;系统主要的功能有登录模块、对个人信息的维护模块、图片信息的隐藏模块涉及图片的上传和信息的验证。 背景&#xff1a; 在当今时代&#xff0c;信息技术…

游戏市场成果及趋势

2024 年的游戏行业发展情况如何&#xff1f;这是一个既关系到开发商&#xff0c;又关系到玩家的问题&#xff0c;而市场分析师可以为我们揭晓答案。下面&#xff0c;就让我们来看看分析师给出的结论以及他们对未来趋势的预测。 玩家 自 2021 年起&#xff0c;全球平均游戏时间…

【算法学习】——整数划分问题详解(动态规划)

&#x1f9ee;整数划分问题是一个较为常见的算法题&#xff0c;很多问题从整数划分这里出发&#xff0c;进行包装&#xff0c;形成新的题目&#xff0c;所以完全理解整数划分的解决思路对于之后的进一步学习算法是很有帮助的。 「整数划分」通常使用「动态规划」解决&#xff0…

深入理解计算机系统阅读笔记-第十二章

第12章 网络编程 12.1 客户端-服务器编程模型 每个网络应用都是基于客户端-服务器模型的。根据这个模型&#xff0c;一个应用时由一个服务器进程和一个或者多个客户端进程组成。服务器管理某种资源&#xff0c;并且通过操作这种资源来为它的客户端提供某种服务。例如&#xf…

Redis 实战篇 ——《黑马点评》(中)

《引言》 &#xff08;中&#xff09;篇将接着记录 Redis 实战篇 ——《黑马点评》&#xff08;上&#xff09;篇之后的学习内容与笔记&#xff0c;希望大家能够点赞、收藏支持一下 ᕦ(&#xff65;ㅂ&#xff65;)ᕤ&#xff0c;谢谢大家。 传送门&#xff08;上&#xff09;&…

【2024年华为OD机试】(A卷,100分)- 猜字谜(Java JS PythonC/C++)

一、问题描述 小王设计了一个简单的猜字谜游戏&#xff0c;游戏的谜面是一个错误的单词&#xff0c;比如 nesw&#xff0c;玩家需要猜出谜底库中正确的单词。猜中的要求如下&#xff1a; 对于某个谜面和谜底单词&#xff0c;满足下面任一条件都表示猜中&#xff1a; 变换顺序…

ip属地是根据手机号还是位置

在数字化时代&#xff0c;IP地址作为网络世界中的“门牌号”&#xff0c;其属地信息往往与用户的地理位置紧密相关。然而&#xff0c;关于IP属地是如何确定的&#xff0c;是否依赖于手机号还是实际位置&#xff0c;这一话题时常引发讨论。本文将深入探讨IP属地的确定方式&#…

nginx 实现 正向代理、反向代理 、SSL(证书配置)、负载均衡 、虚拟域名 ,使用其他中间件监控

我们可以详细地配置 Nginx 来实现正向代理、反向代理、SSL、负载均衡和虚拟域名。同时&#xff0c;我会介绍如何使用一些中间件来监控 Nginx 的状态和性能。 1. 安装 Nginx 如果你还没有安装 Nginx&#xff0c;可以通过以下命令进行安装&#xff08;以 Ubuntu 为例&#xff0…

React性能优化: 使用React.lazy与Suspense提高加载效率

## 1. React.lazy与Suspense简介 在开发React应用程序时&#xff0c;我们经常会遇到需要加载大型组件或者数据的情况。为了提高页面加载的效率&#xff0c;React引入了React.lazy和Suspense这两个特性。 什么是React.lazy&#xff1f; 是React 16.6版本引入的新特性&#xff0c…

Golang笔记——Interface类型

大家好&#xff0c;这里是Good Note&#xff0c;关注 公主号&#xff1a;Goodnote&#xff0c;专栏文章私信限时Free。本文详细介绍Golang的interface数据结构类型&#xff0c;包括基本实现和使用等。 文章目录 Go 语言中的 interface 详解接口定义实现接口空接口 interface{}示…

轨迹优化 | 基于贝塞尔曲线的无约束路径平滑与粗轨迹生成(附ROS C++/Python仿真)

目录 0 专栏介绍1 从路径到轨迹2 基于贝塞尔曲线的粗轨迹生成2.1 路径关键点提取2.2 路径点航向角计算2.3 贝塞尔曲线轨迹生成 3 算法仿真3.1 ROS C仿真3.2 Python仿真 0 专栏介绍 &#x1f525;课设、毕设、创新竞赛必备&#xff01;&#x1f525;本专栏涉及更高阶的运动规划…

理解STC15F2K60S2单片机的最小电路

一、STC15F2K60S2与51单片机的区别 STC15F2K60S2和51单片机虽然都基于8051内核&#xff0c;但在多个方面存在显著区别&#xff1a; 1. CPU性能&#xff1a; - STC15F2K60S2&#xff1a;采用增强型8051 CPU&#xff0c;1T单时钟/机器周期&#xff0c;速度比普通8051快8-12倍…

VSCode 搜索 搜不到

VSCode 搜索 搜不到 今天打开一个新的工作目录之后 ctrl P 搜文件 搜不到 经观察后发现 当我搜索时候&#xff0c; 右侧搜索按钮有一个时钟标识&#xff0c;疑似 搜索的范围 是最近打开内容。 经过和全局搜索的编辑器对比设置后发现&#xff0c;把设置中 下图中 选项去掉勾选…

软件测试 —— Selenium常用函数

软件测试 —— Selenium常用函数 操作测试对象点击/提交对象 click()模拟按键输入 send_keys("")清除文本内容 clear() 模拟用户键盘行为 Keys包示例用法 获取文本信息 textget_attribute("属性名称") 获取当前页面标题 title获取当前页面的 url current_u…

Vue 学习之旅:核心技术学习总结与实战案例分享(vue指令下+计算属性+侦听器)

Vue 学习之旅&#xff1a;核心技术学习总结与实战案例分享 文章目录 Vue 学习之旅&#xff1a;核心技术学习总结与实战案例分享一、指令补充&#xff08;一&#xff09;指令修饰符&#xff08;二&#xff09;v-bind 对样式操作的增强&#xff08;三&#xff09;v-model 应用于其…

UE5 打包项目

UE5 打包项目 flyfish 通过 “文件”->“打开项目”&#xff0c;然后在弹出的对话框中选择项目文件&#xff08;通常是以.uproject为后缀的文件&#xff09; 选择目标平台&#xff1a; 在 UE5 主界面中&#xff0c;找到 “平台”&#xff08;Platforms&#xff09;。根据…

1. Doris分布式环境搭建

一. 环境准备 本次测试集群采用3台机器hadoop1、hadoop2、hadoop3, Frontend和Backend部署在同一台机器上&#xff0c;Frontend部署3台组成高可用&#xff0c;Backend部署3个节点&#xff0c;组成3副本存储。 主机IP操作系统FrontendBackendhadoop1192.168.47.128Centos7Foll…

win10电脑 定时关机

win10电脑 定时关机 https://weibo.com/ttarticle/p/show?id2309405110707766296723 二、使用任务计划程序设置定时关机打开任务计划程序&#xff1a; 按下“Win S”组合键&#xff0c;打开搜索框。 在搜索框中输入“任务计划程序”&#xff0c;然后点击搜索结果中的“任务…

day07_Spark SQL

文章目录 day07_Spark SQL课程笔记一、今日课程内容二、Spark SQL函数定义&#xff08;掌握&#xff09;1、窗口函数2、自定义函数背景2.1 回顾函数分类标准:SQL最开始是_内置函数&自定义函数_两种 2.2 自定义函数背景 3、Spark原生自定义UDF函数3.1 自定义函数流程&#x…