算法中的递归分析和分治法的原理

分析递归算法三种方法
替换法、迭代法、通用法(master method)
作用:分析递归算法的运行时间
分治算法
将一个问题分解为与原问题相似但规模更小的若干子问题,递归地解这些子问题,然后将这些子问题的解结合起来构成原问题的解。这种方法在每层递归上均包括三个步骤:
divide(分解):将问题划分为若干个子问题
conquer(求解):递归地解这些子问题;若子问题Size足够小,则直接解决之
Combine(组合):将子问题的解组合成原问题的解
其中的第二步很关键:递归调用或直接求解  (递归终结条件),且有的算法“分解”容易,有的则“组合”容易
分治法举例:
归并排序
①分解:把n个待排序元素划分为两个Size为n/2的子序列
②求解:递归调用归并排序将这两个子序列排序,若子序列长度为1时,已自然有序,无需做任何事情(直接求解)
③组合:将这两个已排序的子序列合并为一个有序的序列

显然,分解容易(一分为二),组合难。

快速排序 

刚刚分析过了,快速排序是枢轴记录划分,也就是分解难,但是组合易。   A[1…k-1] ≤ A[k] ≤ A[k+1…n]

分治算法分析
设T(n)是Size为n的执行时间,若Size足够小,如n ≤ C (常数),则直接求解的时间为θ(1)
①设完成划分的时间为D(n)
②设分解时,划分为a个子问题,每个子问题为原问题的1/b,则解各子问题的时间为aT(n/b)
③设组合时间C(n)
一般地,递归的求解划分,而解递归式时可忽略细节
①假定函数参数为整数,如
②边界条件可忽略,当n较小时,T(n)= θ(1)

因为这些细节一般只影响常数因子的大小,不改变量级。求解时,先忽略细节,然后再决定其是否重要!

分析的方法

替换法

猜测解,用数学归纳法确定常数C,证明解正确,关键步骤是用猜测的解代入到递归式中。
做出好的猜测(没有一般方法,只能凭经验)
①与见过的解类似,则猜测之。
②先证较宽松的上、下界,减小猜测范围。
细节修正
有时猜测解是正确的,但数学归纳法却不能直接证明其细节,这是因为数学归纳法不是强大到足以证明其细节。这时可从猜测解中减去一个低阶项以使数学归纳法得以满足
避免陷阱
与求和式的数学归纳法类似,证明时渐近记号的使用易产生错误。
变量变换
有时改动变量能使未知递归式变为熟悉的式子。例如:
迭代法
展开:无须猜测,展开递归式,使其成为仅依赖于n和边界条件的和式,然后用求和方法定界。需要关注:
1、达到边界条件所需的迭代次数
2、迭代过程中的和式。若在迭代过程中已估计出解的形式,亦可用替换法
3、当递归式中包含floor和ceiling函数时,常假定参数n为一个整数次幂,以简化问题。
递归树
使展开过程直观化
例: T(n)=2T(n/2)+n^2   (不妨设n=2k)

The master method(通用法,万能法)

可迅速求解

T(n)=aT(n/b)+f(n)   //常数a ≥1, b>1, f(n)渐近正
意义:将Size为n的问题划分为a个子问题,每个子问题Size为n/b。每个子问题的时间为T(n/b),划分和combine的时间为f(n)。
注意:n/b不一定为整数,应为⌊n/b⌋或⌈n/b⌉,不会影响渐近界。
 
关于递归和循环的理解与比较

递归通俗的说就是一个函数调用函数自己(本身),这个调用过程叫递归,递归是一把双刃剑(有时方便,有时不好),如果需要处理重复的需要多次计算的问题,通常可以选择用递归或者循环两种方式,但是递归的执行效率不如循环语句。

注意:必须设置终止递归的条件检测,否则慎用。

void up_and_down(int);//函数原型声明
int main() {up_and_down(1);//调用递归函数

system(
"pause");
return 0; }
void up_and_down(int n) {printf("level%d, n地址=%p\n", n, &n);if (n < 4){up_and_down(n + 1);}printf("LEVEL%d, n地址=%p\n", n, &n); }

首先,main函数用参数1调用递归函数,递归函数形参n=1,打印语句level1……然后,n<4,故函数本身使用参数n+1=2,第二次调用自己,这样就打印了level2……

以此类推,当执行到第四级调用,n=4,if失效,不再调用函数,而是执行了第二句打印,先输出LEVEL4……此时第四级调用结束,控制权返回给了主调函数,也就是第三级主调函数,此函数中上一句是if语句,已经执行完毕,然后继续执行第二句打印语句,输出LEVEL3……第三级调用结束,返回控制权给调用函数(也就是第二级主调函数),然后第二级函数开始继续执行,以此类推,打印LEVEL2,1……

递归的基本原理

每一级递归都使用自己这一级的私有变量n,同级调用时的地址和返回的地址是一样的。好好揣摩!

这是函数自己在一层层的往深度调用自己,然后一层层的往回返,每到一层,就继续执行接下来的语句(故调用开始的地址和返回的地址一样),而每一级递归都是用自己的局部变量。也就是第一级的n不同于第二级的n,这样子,函数逐步调用然后逐步返回直到main函数里。

递归函数里,递归语句之前的语句和各级被调的递归函数执行顺序一致,而递归语句之后的语句和被调的递归函数执行顺序相反(这一特点针对涉及反向顺序的编程问题很有用) 

递归函数必须包含可以终止的条件,因为递归可以替代循环,故必须有终止

尾递归

最简单的递归:递归语句放到函数末尾,恰在return语句前,叫做tail recursion(尾递归),因为出现在函数尾部,作用相当于一条循环语句。

//计算阶乘(递归和循环)
#include <stdio.h>
#include <stdlib.h>
//计算阶乘
int factorial(int);
int loopFactorial(int);int main()
{int num;printf("输入1-12的整数,q退出\n");while (scanf_s("%d", &num)){if (num < 0){printf("error!输入1-12的整数!");}else if (num > 12){printf("输入1-12的整数!");}else{printf("\n%d的阶乘=%d", num, factorial(num));printf("\n%d的阶乘=%d", num, loopFactorial(num));}printf("\n输入1-12的整数");}system("pause");return 0;
}
//循环计算阶乘
int factorial(int n)
{int temp;for (temp = 1; n > 1; n--){temp *= n;}return temp;
}
//使用递归计算阶乘
int loopFactorial(int n)
{int temp;if (n > 0){temp = n * loopFactorial(n - 1);//属于尾递归,如n>0那么这就是最后一句
    }else{temp = 1;//必须要有递归结束判断条件!
    }return temp;
}

注意:整型范围,32位机器,int类型最大到21多亿,再大的话,就要用long long或者double类型,一般来说,选择循环比较好些,递归每次调用都要有自己的变量集合,占据内存大,每次都要存储新的变量集合到堆栈,这样速度慢,但是递归(最简单的是尾递归)比较简单。一些情况还是要用。

 

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

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

相关文章

ps4连接r星服务器稳定,移植到不同平台的《GTA5》有什么变化?与八年前相比,差距这么大...

经历过多次移植的GTA5&#xff0c;这些年都产生了哪些变化呢&#xff1f;当年首次登上PS3平台的GTA5&#xff0c;展现出无数玩家羡慕的精彩时刻&#xff0c;但好景不长&#xff0c;PS4就发布了&#xff0c;而且成功了代替了PS3&#xff0c;成为了GTA5玩家的主力军主机&#xff…

关于pycharm+opencv没有代码提示的问题解决方法记录

代码可以看出实际我们引入的应该是cv2.cv2下面. 所以我们代码只需要import cv2.cv2 as cv 即可. 记着要重新启动下pycharm哦. 可以参考:https://blog.csdn.net/az9996/article/details/90546827 转载于:https://www.cnblogs.com/lovesKey/p/11135185.html

Web前端开发CSS基础(2)

CSS 层叠样式表(英文全称&#xff1a;Cascading Style Sheets),是一种用来表现HTML(标准通用标记语言的一个应用)或XML(标准通用标记语言的一个子集)等文件样式的计算机语言.CSS不仅可以静态地修饰网页,还可以配合各种脚本语言动态地对网页各元素进行格式化,并且能够对网页中元…

消息摘要算法-MAC算法系列

为什么80%的码农都做不了架构师&#xff1f;>>> 一、简述 mac&#xff08;Message Authentication Code&#xff0c;消息认证码算法&#xff09;是含有密钥散列函数算法&#xff0c;兼容了MD和SHA算法的特性&#xff0c;并在此基础上加上了密钥。因此MAC算法也经常…

软件开发七宗罪

导读&#xff1a;一起来看看世界各地的程序员们所共有的致命通病&#xff0c;从软件开发的地狱中拯救自己。这七宗罪你有几条&#xff1f; 想成为一名优秀的软件开发人员需要很长时间的培训和实践。但是如果不遵循合适的原则&#xff0c;即便是再好的程序员也会成为失败的牺牲品…

Spring Boot:快速入门教程

什么是Spring Boot? Spring Boot是由Pivotal团队提供的全新框架&#xff0c;其设计目的是用来简化新Spring应用的初始搭建以及开发过程。该框架使用了特定的方式来进行配置&#xff0c;从而使开发人员不再需要定义样板化的配置。简而言之&#xff0c;Spring Boot通过提供默认配…

移动微技(Mobile Widget)应用开发权威指南

移动微技&#xff08;Mobile Widget&#xff09;应用开发权威指南 基本信息 作者&#xff1a; 程宝平 杨晓华 朱春梅 丛书名&#xff1a; 中国移动创新系列丛书 出版社&#xff1a;电子工业出版社 ISBN&#xff1a;9787121104527 上架时间&#xff1a;2010-4-19 出版日…

Apache、Nginx、Tomcat的区别

Apache、Nginx、Tomcat的区别NginxTomcatApache&#xff0c;Nginx与Tomcat的区别Nginx -----是一个高性能的HTTP和反向代理服务器&#xff0c;也是一个IMAP/POP3/SMTP代理服务器。** Tomcat -----servlet(jsp)应用服务器 Tomcat是运行在JVM中的一个进程。它定义为“中间件”…

检索

【摘抄】基于线性表的检索一、检索的基本概念和算法分类1、检索概念&#xff1a; 可以形式化地定义基于关键码的检索。假定k1、k2…kn是互不相同的关键码值&#xff0c;有一个包含n条记录的集合C&#xff0c;形式如下&#xff1a; (k1, R1)&#xff0c;(k2, R2)&#xff0c;…

例说C#深拷贝与浅拷贝

一开始&#xff0c;先对C#深拷贝与浅拷贝知识做个简单的总结。 无论是浅拷贝与深拷贝&#xff0c;C#都将源对象中的所有字段复制到新的对象中。不过&#xff0c;对于值类型字段&#xff0c;引用类型字段以及字符串类型字段的处理&#xff0c;两种拷贝方式存在一定的区别&#x…

企业架构 - 开篇:TOGAF介绍

在《年度总结和计划&#xff1a;去年4个1&#xff0c;今年5个1》中说过今年我准备在项目组引入一个架构方TOGAF。工业和信息化部副部长杨学山在一次内部座谈时提到&#xff1a;与西方发达国家比&#xff0c;国内的信息化建设在硬件方面已经不相上下&#xff0c;在软件方面有5年…

C#(4) implicit explicit

最近几节课学的知识点的量有很大增长&#xff0c;加上上机和托福考试的临近&#xff0c;这里练习的所有代码就不走形式全贴出来了&#xff0c;找我自己觉得又不熟悉的点贴出来&#xff0c;争取简明扼要&#xff0c;恩恩 这次主要看接口的问题&#xff1a; 接口可以多继承&#…

【jquery】基于 jquery 实现 ie 浏览器兼容 placeholder 效果

placeholder 是 html5 新增加的属性&#xff0c;主要提供一种提示&#xff08;hint&#xff09;&#xff0c;用于描述输入域所期待的值。该提示会在输入字段为空时显示&#xff0c;并会在字段获得焦点时消失。placeholder 属性适用于以下类型的 input 标签&#xff1a;text, se…

连接远程电脑中的虚拟机---端口映射

连接远程电脑中的虚拟机---端口映射问题:A主机中运行有2台虚拟机&#xff0c;现在我们希望在B主机中通过ssh&#xff08;或其他方式&#xff09;访问A主机中的2台虚拟机。(前提&#xff1a;B主机可以ping通A主机&#xff0c;即A有公网IP或A、B在同一子网下) 解决方法 设置虚拟…

Unity 动画属性

在动画的使用上使用不当的设置往往会造成不可预料的结果。 首先&#xff0c;如果动画自身可以驱动物体移动&#xff0c;那么在Animator组件上必须选择apply root motion&#xff0c;物体的动画位移才能生效&#xff0c;否则动画只能在原地播放。 第二&#xff0c;Bake Into Pos…

Javascript正则匹配数字,中英文,中横线,下划线,utf-8中文

为什么80%的码农都做不了架构师&#xff1f;>>> function check_string(nickname) {var reg /^[A-Za-z0-9-_\u4e00-\u9fa5]{4,30}$/;if (!reg.test(nickname)) {return false;}return true; } 转载于:https://my.oschina.net/biezhi/blog/396989

基于linux的集群系统LVS

一 LVS &#xff08;Linux Virtual Server&#xff09;针对高可伸缩、高可用网络服务的需求&#xff0c;我们给出了基于IP层和基于内容请求分发的负载平衡调度解决方法&#xff0c;并在Linux内核中实现了这些方法&#xff0c;将一组服务器构成一个实现可伸缩的、高可用网络服务…

jQuery 开发环境搭配(转)

jQuery 作为一个轻量级的 JavaScript 库&#xff0c;至 06 年出来到现在&#xff0c; 也算是用得最火的 JavaScript 库了吧&#xff0c; 要开始学习 jQuery &#xff0c;首先还是必须得把环境搭配好吧&#xff0c; 这一篇博文呢&#xff0c;主要是想介绍一下在 Dreamweaver 上和…

Connection to node 0 (/192.168.204.131:9092) could not be established

解决连接虚拟机内kafka出现Connection to node 0 /192.168.204.131:9092 could not be established.Broker may not be available.问题&#xff1a; 在我尝试使用Springboot集成虚拟机内运行的kafka服务器时&#xff0c;出现了以下问题。 原因&#xff1a;可能出现的原因主要有…

获取本机MSSQL保存凭证

本文转自王子博客&#xff1a;http://www.cnblogs.com/killbit/p/4355950.html 首先要感谢哥们对我的指点&#xff0c;多谢。 当我们遇到类似情况下&#xff0c;如何获取保存在MSSQL工具里的凭证呢&#xff1f; //如果对方连接地址后面加了IP\sqlexpress 连接的时候你也记得加上…