递归和分治思想及其应用

目录

  • 递归和分治思想
  • 一些实例
    • 逆序输出字符串
    • 查找数组元祖是否存在
    • 汉诺塔问题
    • 八皇后问题
  • 更多:

递归和分治思想

如果可以使用迭代,尽量别使用递归。由编译原理可以知道,每次自调用的时候,计算机都需要保存在调用,浪费时间空间。当然,迭代是当我们知道循环次数的时候。而当我们不知道循环次数,比如说对于文件夹和文件进行遍历,不知道深度的情况下,我们就需要递归来实现。

显然,递归是先解决小的问题,这种思想是分治思想。根据具体需求,来决定是否使用递归。

递归要注意:

  • 结构是选择结构,而迭代是循环结构
  • 必须有基线条件和递归条件,防止出现死循环
  • 如果知道循环次数的话,尽量使用递归
  • 对于某些编程式函数,有对于尾递归的迭代优化
  • 递归逻辑更容易理解

一些实例

逆序输出字符串

#include<iostream>
using namespace std;void print(){char a;cin>>a;if(a!='#') print(); // 不是停止符,先自调用 if(a!='#') cout<<a; //在回来的时候,打印自己的字符 
}
int main(){print();return 0;
}

查找数组元祖是否存在

很多时候我们需要查找一个数组中是否有一个元素。如果使用迭代,肯定十分简单,时间复杂度为O(n)。

此时,如果使用分而治之的思想,我们可以使用二分法来进行查找。不论多大的数据,时间复杂度显著降低为O(log_2 n)。也就是说一个大小为123456789的数组,使用迭代,我们需要123456789个时间单位。但是二分法只需要27次。

实现思路:

  1. 首先转化的思想对数组进行排序。如果不排序,那么low和high就没有意义了。
  2. 再用迭代进行二分
#include<iostream>
#include<algorithm>
using namespace std;
const int SIZE = 5;
const int NONE = -1;
//二分查找并且返回element的位置,没查找到则返回NONE
template<class T>
int BinFind(T *arr,int low,int high,T elem){  int mid;if(low>high)return NONE;else{mid = (low+high)/2;if(arr[mid] == elem)return mid;else if(elem>arr[mid])return BinFind(arr,mid+1,high,elem);elsereturn BinFind(arr,low,mid-1,elem);}
}
int main(){int *arr = new int [SIZE];cout<<"请输入"<<SIZE<<"个数据:"; for(int i=0;i<SIZE;++i)cin>>arr[i];sort(arr,arr+SIZE);int elem;cout<<"输入您要查找的数据:"<<endl;cin>>elem; int index = BinFind(arr,0,SIZE-1,elem);if (index+1)cout<<"含有这个数据\n";elsecout<<"不含有这个数据";return 0;
}

汉诺塔问题

首先我们假设需要移动64层,那么思路如下(附截图):

大问题

此时,需要解决两个问题(附截图):

总问题下的子问题

一直这样类推,知道最后从begin(开始柱子)->end(目标柱子)。

按照第一张截图,我们可以写出来函数里面else的递归部分。并且,每次输出的时候,就对应着思路里面的移动(而不是分治),此时step步数需要加+。

#include<iostream>
#include<algorithm>
using namespace std;
void Hanoi(int num,char begin,char by,char end,int &step){if(num==1){cout<<begin<<"-->"<<end<<endl;++step;}else{Hanoi(num-1,begin,end,by,step);cout<<begin<<"-->"<<end<<endl;++step;Hanoi(num-1,by,begin,end,step);}
}
int main(){int step = 0;int num;cout<<"汉诺塔层数是: ";cin>>num;Hanoi(num,'X','Y','Z',step);cout<<"一共有:"<<step<<"步数"<<endl; return 0;
}

八皇后问题

在8×8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行、同一列或同一斜线上,问有多少种摆法。正规的方法是递归,如果不考虑效率,这里采用递归实现。假设从第一行开始,每一行都找到符合条件的一个位置,而条件就是新的一行的新位置符合要求,以此类推,就可以写出来递归函数。

#include<iostream>
using namespace std;const int Q_NUM = 8;
int queens[Q_NUM][Q_NUM] = {0};
int RESULT = 0;void print(){for(int i=0;i<Q_NUM;++i){for (int j=0;j<Q_NUM;++j)cout<<queens[i][j]<<" ";cout<<endl;}cout<<endl<<endl;
}
bool IfQueen(int row,int col){if(row==0){//当第一行时候,随便摆放 queens[row][col] = 1;return true;}/**************其他时候,需要考虑上面的同一列、左上角斜线、右上角斜线,以下分别实现*****/ for(int i=0;i<row;++i)if(queens[i][col]==1)return false;for (int i=row-1,j = col-1;i>=0 && j>=0;--i,--j)if(queens[i][j]==1)return false;for(int i=row-1,j=col+1;i>=0 && j<Q_NUM;--i,++j)if(queens[i][j]==1)return false;/******当所有情况都满足********/queens[row][col] = 1;return true;
}
void Queen(int row){if(row==Q_NUM){ //注意row是从0开始到Q_NUM-1结束。这样当row==Q_NUM时,已经排完所有情况 ++RESULT;   //这样当row==Q_NUM时,已经排完所有情况,进行输出就可以了。 print();return ;} for(int i=0;i<Q_NUM;++i){ //i代表列数 if(IfQueen(row,i)) //如果row行i列可以放得话,判断下一行 Queen(row+1);queens[row][i] = 0; //重置为0,防止下次结果干扰 }
}int main(){Queen(0);cout<<"一共"<<RESULT<<"种摆法\n";return 0;
}

更多:

毫无疑问,递归以及分治思想还有很多用法:斐波那契数列、快速排序、文件查找、字典树的建立等等。理论上递归可以解决任何问题,而作为我们只需要提供思路,其他的交给计算机解决。所以听人说过计算机最适合解决递归问题。但是,有利有弊,递归同样会消耗更多的内存。在初步实现阶段,将大问题分而治之,分装成递归函数,还是逻辑代码化的最佳表达。


欢迎进一步交流本博文相关内容:

博客园地址 : http://www.cnblogs.com/AsuraDong/

CSDN地址 : http://blog.csdn.net/asuradong

也可以致信进行交流 : xiaochiyijiu@163.com

欢迎转载 , 但请指明出处  :  )


转载于:https://www.cnblogs.com/AsuraDong/p/7045141.html

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

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

相关文章

AM+PM+FM基本调制原理及相关理论

总论&#xff1a; 调制信号&#xff1a; 模拟信号m(t)&#xff0c;可以是正弦波信号、方波信号等任意信号&#xff0c;又称基带信号 载波信号&#xff1a;一般为正弦波信号 已调信号&#xff1a; 幅度调制AM---A(t)随m(t)成比例变化----线性调制 相位调制PM---随m(t)成比…

win32 api 文件操作!

CreateFile打开文件要对文件进行读写等操作&#xff0c;首先必须获得文件句柄&#xff0c;通过该函数可以获得文件句柄&#xff0c;该函数是通向文件世界的大门。ReadFile从文件中读取字节信息。在打开文件获得了文件句柄之后&#xff0c;则可以通过该函数读取数据。WriteFile向…

小说里的lt什么意思_游戏cpdd网络用语是什么意思 王者荣耀里很常见

[闽南网]随着互联网的发展&#xff0c;越来越多的流行语横空出世&#xff0c;在网络上得到广泛使用。当一个网络语流行的时候&#xff0c;不管在微博上还是贴吧里&#xff0c;都会看见和流行语有关的句子和表情包。眼下在各种游戏里&#xff0c;总是能看到游戏玩家们说“cpdd”…

linux系统 硬链接和软链接

背景&#xff1a; 当几个用户同在一个项目里工作时。经常须要共享文件。假设一个共享文件同一时候出如今属于不同用户的不同文件夹下。工作起来就非常方便。比如B和C文件夹下有一文件D是两者都能够訪问和改动的共享文件&#xff0c;这样是非常方便&#xff0c;但也会有一些问题…

openssl 学习之从证书中提取RSA公钥N 和 E

原文链接: http://blog.csdn.net/kkxgx/article/details/19850509 通常数字证书包含很多信息&#xff0c;其中N和E值即我们称为的公钥。如何从PEM 或者DER格式的证书中提出证书呢&#xff1f;下面给出代码实现从PEM和DER编码的证书中提出N、E。 [cpp] view plaincopy #include …

Web browser的发展演变

我们每天都在使用着浏览器&#xff0c;每个人使用的浏览器各不一样。在这个科技飞速发展的时代&#xff0c;一个游览器能否站住脚跟取决于使用者的数量&#xff0c;看用户是否喜欢这个产品&#xff0c;听取用户们的意见来改善。 我们这个年龄的人最初用到的浏览器肯定是IE浏览器…

matlab 散点图 线性回归图_线性回归思路梳理

作者&#xff1a;夏雨骄阳 封面&#xff1a;自己想吧1简单线性回归1根据研究目的确定因变量和自变量。2判断有无异常值。通过绘制散点图直观观察&#xff1b;亦可通过线性回归的【统计】→【个案诊断】→【所有个案】进行分析&#xff0c;若标准残差超过[-3,3]&#xff0c;则…

物联网云端设计分析

物联网是世界信息产业发展的新浪潮&#xff0c;智能手表、智能手环、智能灯等物联网产品不断的改变着人们的生活方式。那这些产品是怎么设计出来的呢&#xff1f;其实物联网操作系统不光由本地物联网设备上的操作系统组成&#xff0c;还包括提供物联网终端设备支持的云端架构。…

Redis和Memcached的区别

2019独角兽企业重金招聘Python工程师标准>>> Redis的作者Salvatore Sanfilippo曾经对这两种基于内存的数据存储系统进行过比较&#xff1a; Redis支持服务器端的数据操作&#xff1a;Redis相比Memcached来说&#xff0c;拥有更多的数据结构和并支持更丰富的数据操作…

hbase hmaster一会就没了_浅析HBase

一、HBase简介1、Apache HBase™是Hadoop数据库&#xff0c;是一个分布式&#xff0c;可扩展的大数据存储。2、当您需要对大数据进行随机&#xff0c;实时读/写访问时&#xff0c;请使用Apache HBase™。 该项目的目标是托管非常大的表&#xff08; 数十亿的行*百万的列 &#…

MATLAB 迭代法解方程

MATLAB 迭代法解方程 1、代码如下&#xff1a; %%牛顿迭代法解方程 function xnewton_interation(fun,dfun,x0,EPS) %简单牛顿迭代法%fun即迭代函数&#xff0c;dfun即迭代函数的一阶导数&#xff0c;x0为迭代初值&#xff0c;EPS为精度x1x0-fun(x0)/dfun(x0); %牛顿迭代公…

超微服务器电源短接启动图解_教你一招,让你的电脑启动速度秒杀别人

win10快速启动其实是电脑的一种休眠模式&#xff0c;它将电脑中的一些本该关闭的文件保存到hiberfil.sys的磁盘文件中&#xff0c;这样打开电脑时就达到了快速开机的目的。接下来&#xff0c;我就将win10设置快速启动的方法分享给你们win10系统功能非常强大&#xff0c;最让大家…

MATLAB 求离散信号卷积

MATLAB 求离散信号卷积 代码如下&#xff1a; function [C,Ck] dt_convolution_advance(A,B,Ak,Bk) % dt_convolution_advance 计算离散信号卷积 % A 输入信号 % B 输入信号 % Ak 输入信号A下标 % Bk 输入信号B下标 % C 输出信号 % Ck 输出信号C下标 % 计算输入信号A&…

发布 项目_第十八期科创基金项目发布会圆满结束

第十八期科创基金项目发布会圆满结束贺电&#xff01;贺电!电子信息工程学院发来贺电&#xff1a;第十八届科创基金项目发布会圆满结束啦&#xff01;感谢导师们的支持、现场同学的热情参与和科协朋友后勤服务&#xff01;NUAA即使是寒冷的冬天也无法阻挡同学们对科创的热情,那…

Fisher线性判别算法原理及实现 MATLAB

Fisher线性判别算法原理及实现 MATLAB 一、Fisher判别器原理 二、代码实现 clc; close all; clear; %% 生成数据 rng(2020); %指定一个种子 mu1 [0 3]; sigma1 [0.5 0; 0 0.5]; data1 mvnrnd(mu1,sigma1,300); %生成一个300*2的矩阵&#xff0c;每一列的数据分别以0&…

大话设计模式之外观模式

年年作品展&#xff0c;岁岁不相同&#xff0c;鹅黄新绿涟漪泛起思想的火花却不尽相同。十期的作品展&#xff0c;从13年3月20号開始到完美落幕&#xff0c;时至今日&#xff0c;她已经在我的记忆中成为过去。这朵小小的浪花激起的涟漪渐渐褪去&#xff0c;或许已没有或许&…

HK算法原理及MATLAB实现

HK算法原理及MATLAB实现 一、编程原理 说明&#xff1a; 1、绿色框图中&#xff0c;当alpha取为最优权向量时&#xff0c;会使 取为最小值。arg的意思是当后面那个函数满足时&#xff0c;取出它的自变量的值。 2、注意后面那个式子是对b求偏导 3、e(t)大于0时&#xff0c;下…

python列反过来_xlwings 教程:使用Python更快速地处理Excel

Excel在当今商业中的使用非常普遍。在Dataquest&#xff0c;出于很多原因&#xff0c;我们通常推荐使用代码处理数据&#xff0c;并且我们的许多数据科学课程的目标是教授数据分析和数据科学的高效编码。但是&#xff0c;无论您多么喜欢使用Python&#xff0c;在一天结束时&…

多类线性分类器算法原理及代码实现 MATLAB

多类线性分类器算法原理及代码实现 MATLAB 一、算法原理 下面举例说明为何蓝圈部分在case2中是确定的而在case1中不确定&#xff1a; 二、代码实现 1、HK函数 function [] HK(w1_data,w2_data) %w1_data为第一类数据集 w2_data为第二类数据集 %此函数的作用为用HK算法对输…

(转) C#如何使用异步编程

怎么使用异步&#xff0c;就是用委托进行处理&#xff0c;如果委托对象在调用列表中只有一个方法&#xff0c;它就可以异步执行这个方法。委托类有两个方法&#xff0c;叫做BeginInvoke和EndInvoke&#xff0c;它们是用来异步执行使用。 异步有三种模式 等待模式&#xff0c;在…