JAVA算法之递归
Ⅰ、三角数字
首先我们来看一组数字:1,3,6,10,15,21.....,在这个数列中第n项是由n-1项加n得到的,这个序列中的数字称为三角数字因为他们可以形象化地表示成一个三角形排列。如下图
通过上面的图首先我们就可以想到使用循环来查找第n项的数值,下面代码就是从高度为n的列加到高度为1的列
int triangle(int n){
int total = 0;
while(n>0){
total = total + n;
--n;
}
return total;
}
上面方法循环了n次,第一次加n,第二次加n-1,一直加到1从而可以算出第n项的值。
使用递归的思想查找第n项的值,此时我们将三角数字看做是第一列和剩余所有列的和,如下图所示,可以写出triangle()方法
int triangle(int n){
//1. return (n + sumAllColums(n-1)) //这里我们可以发现sumAllColums方法做的事情和triangle做的事情一模一样
return (n + triangle(n-1)) //这里我们可以将上面步骤换为此步骤,从而得到求三角数字的递归算法
}
此时上面的递归代码是不会终止的,所以我们的每一个递归代码要有一个我们称之为基值(base case)以防止无限递归导致程序崩溃,所以上面求三角函数代码的基值就是1,
int triangle(int n){
if(n==1){
return 1; //求三角数字的基值
}else{
return (n+triangle(n-1)) ; //进行递归逻辑
}
}
下图表示了递归triangle方法的执行过程,假设传入n=5,从5开始每次减1,不断进入方法自身,直到减到1时,方法进行返回,直到返回到最外层。注意,在方法返回1之前,实际上同时有5个不同的triangle()方法实例存在,最外层传入的参数是5,最内层传入的参数是1
综上我们可以总结出递归方法的特征:
- 调用自身
- 调用自身为了解决更小的问题
- 存在足够简单的层次,即上面说的基值
递归的效率:递归的过程中控制必须从调用的位置转移到方法的开始处,除此之外这个方法的参数以及返回值会压入到一个内部栈中,从而知道访问的参数值和返回到哪里。所以递归效率较低,我们常常采用递归,是因为他从概念上简化了问题,而不是它更有效率。
II、归并排序
接下来我们讲讲与递归相关的排序算法,归并排序:
首先这种排序在时间上更有效,时间复杂度为O(N*logN),如果排序数据项N为10000,那么一般的简单排序N2就是100000000,而N*logN只是40000,意思就是若使用归并排序需要40s,那么使用插入排序需要近28小时。归并排序的缺点是需要在存储器中有另一个大小等于被排序的数据项数目的数组,所以排序对数组的大小有一定的限制。
归并算法的核心是归并两个已经有序的数组,我们假设数组A有4个数据项,数组B有6个数据项,他们要被归并到C中,开始的时候C有10个空的存储空间。下表显示了归并进行的必要的比较,每一次比较将较小的数据项复制到数组C中,表中B数组在第八步之后是空的,所以不需要再进行比较,直接将A数组复制到C中去即可。
接下来我们给出归并的java代码:
public int[] merge(int[] arrayA,int sizeA,int[] arrayB,int sizeB,int[] arrayC){
int aDex=0,bDex=0,cDex=0; //记录三个数组当前的脚标
while(aDex<sizeA&&bDex<sizeB){ //此时还需进行比较
if(arrayA[aDex] < arrayB[bdex])
arrayC[cDex++] = arrayA[aDex++];
else
arrayC[cDex++] = arrayA[bDex++];
}
while(aDex<sizeA) //说明数组a中还有剩余数据,拷贝到数组c
arrayC[cDex++] = arrayA[aDex++];
while(bDex<sizeB) //说明数组b中还有剩余数据,拷贝到数组c
arrayC[cDex++] = arrayB[bDex++];
return arrayC;
}
归并排序简单来讲就是反复地将数组进行分割,利用递归的思想,直到子数组只含有一个数据项(基值),在归并排序方法中每一次调用自身的时候排列都会被分成两部分,并且每一次返回时都会把两个较小的排列合并成一个更大的排列,接下来我们给出归并排序的代码:
public class DArray {
private long[] theArray; //存储数据的数组
private int nElems; //填充数据个数索引 public DArray(int max) {
theArray = new long[max];
nElems = 0;
}
/***插入数据*/
public void insert(long value){
theArray[nElems++] = value;
}
/***归并排序的方法*/
public void mergeSort() {
long[] workSpace = new long[nElems];
recMergeSort(workSpace, 0, nElems - 1);
} /**
* 利用递归进行归并排序
* @param workSpace 工作区
* @param lowerBound 归并区域的起始索引
* @param upperBound 归并区域的结束索引
*/
private void recMergeSort(long[] workSpace, int lowerBound, int upperBound) {
if (lowerBound == upperBound) {
return;
} else {
//找到中间分界点
int mid = (lowerBound + upperBound) / 2;
//首先递归调用自己将前半部分的数据归并为有序
recMergeSort(workSpace, lowerBound, mid);
//然后递归调用自己将后半部分的数据归并为有序
recMergeSort(workSpace, mid + 1, upperBound);
//调用归并算法将上面归并有序后的数据进行归并
merge(workSpace, lowerBound, mid + 1, upperBound);
}
} /**
* 归并算法
* @param workSpace 工作区
* @param lowPtr 首段归并区域的初始索引
* @param highPtr 末端归并区域的初始索引
* @param upperBound 末端归并区域的结束索引
*/
private void merge(long[] workSpace, int lowPtr, int highPtr, int upperBound) {
int j = 0; //工作区的index
int lowerBound = lowPtr; //首段归并区域的初始索引(复制到theArray的初始索引)
int mid = highPtr - 1; //对应索引较小区域的结束位置索引
int n = upperBound - lowerBound + 1; //归并的此段区域所含有的数据项的个数 //将对应范围lowPtr到upperBound的数据复制到工作区
while (lowPtr <= mid && highPtr <= upperBound) {
if (theArray[lowPtr] < theArray[highPtr]) {
workSpace[j++] = theArray[lowPtr++];
} else {
workSpace[j++] = theArray[highPtr++];
}
}
//将对应还未复制完的数据复制到工作区中
while (lowPtr <= mid) {
workSpace[j++] = theArray[lowPtr++];
}
while (highPtr <= upperBound) {
workSpace[j++] = theArray[highPtr++];
}
//此步骤相当于是将对象内的theArray数组变成lowPtr到upperBound局部有序,将归并到工作区的数据放入theArray的对应位置
for (j = 0; j < n; j++) {
theArray[lowerBound + j] = workSpace[j];
}
}
}
归并的效率:(假设复制和比较是最耗时的操作)
复制次数:
上表中可以看出来当N为2的乘方的时候的操作次数,我们可以这样来理解需要的复制次数,log2N表示将N对半分解为我们归并的基值1的时候需要的步数,然后每一步我们都需要将N个数据项复制到我们的工作区,所以复制到工作区的次数就应该是N*log2N,这些数据复制到工作区之后还需要复制到原数组中所以复制次数会增加一倍
比较次数:
上图中我们可以前面表示进行归并的时候进行的最多和最少的比较次数,后表列举出了包含8个数据项进行归并排序的比较次数,对于八个数据项需要七次归并的操作,对于每一次归并最大比较次数是数据项减1,最小比较次数是数据项的一半,加在一起可算出归并排序需要的比较次数在12到17之间。
至此关于递归的思想,和关于递归的归并排序就结束了。
JAVA算法之递归的更多相关文章
- Java算法之递归打破及在真实项目中的使用实例
开心一笑 刚才领导问开发:"你觉得这个项目的最大风险是什么",开发说:"加班猝死" , 气氛尴尬了一分钟!!! 提出问题 1.递归算法简单复习 2.如何实现递归 ...
- Java实现 蓝桥杯VIP 算法提高 递归倒置字符数组
算法提高 递归倒置字符数组 时间限制:1.0s 内存限制:512.0MB 问题描述 完成一个递归程序,倒置字符数组.并打印实现过程 递归逻辑为: 当字符长度等于1时,直接返回 否则,调换首尾两个字符, ...
- JAVA算法系列 快速排序
java算法系列之排序 手写快排 首先说一下什么是快排,比冒泡效率要高,快排的基本思路是首先找到一个基准元素,比如数组中最左边的那个位置,作为基准元素key,之后在最左边和最右边设立两个哨兵,i 和 ...
- JavaScript算法 ,Python算法,Go算法,java算法,系列之【归并排序】篇
常见的内部排序算法有:插入排序.希尔排序.选择排序.冒泡排序.归并排序.快速排序.堆排序.基数排序等.用一张图概括: 归并排序(英语:Merge sort,或mergesort),是创建在归并操作上的 ...
- java算法-数学之美一
巧用数学的思想来解决程序算法问题,这样的代码如诗般优美.通过数学思想来看问题,也能将程序简单化.“斐波那契数列”对于java程序员来说一定不陌生.当然这个问题的解决方案也有很多.用一个例子说明数学思想 ...
- Java中的递归运算
Java中的递归运算是一种在自己的方法内部调用自己的方法 递归的设计思想是:把一个复杂的问题,分解为若干个等同的子问题,重复执行,直到之问题能够简单到直接求解,这样复杂的问题就得以解决. 递归运算有两 ...
- Atitit 表达式原理 语法分析 原理与实践 解析java的dsl 递归下降是现阶段主流的语法分析方法
Atitit 表达式原理 语法分析 原理与实践 解析java的dsl 递归下降是现阶段主流的语法分析方法 于是我们可以把上面的语法改写成如下形式:1 合并前缀1 语法分析有自上而下和自下而上两种分析 ...
- JAVA算法系列 冒泡排序
java算法系列之排序 手写冒泡 冒泡算是最基础的一个排序算法,简单的可以理解为,每一趟都拿i与i+1进行比较,两个for循环,时间复杂度为 O(n^2),同时本例与选择排序进行了比较,选择排序又叫直 ...
- 汉诺塔算法的递归与非递归的C以及C++源代码
汉诺塔(又称河内塔)问题其实是印度的一个古老的传说. 开天辟地的神勃拉玛(和中国的盘古差不多的神吧)在一个庙里留下了三根金刚石的棒,第一根上面套着64个圆的金片,最大的一个在底下,其余一个比一 个小, ...
随机推荐
- angularJS 上传multipart/form-data
var fd = new FormData();fd.append('file', vm.file);CommodityViewImport.post(fd, onSaveSuccess, onSav ...
- python实现马赛克拼图!
python实现马赛克拼图 直接上代码! 代码如下: #!/usr/local/bin/python3# --*-- coding:utf8 --*-- import getoptimport sy ...
- 35-Ubuntu-组管理-01-添加组/删除组/确认组信息
组管理 提示: 创建组/删除组的终端命令都需要sudo执行,标准用户没有权限! 序号 命令 作用 01 sudo groupadd 组名 添加组 02 sudo groupdel 组名 删除组 03 ...
- 十分钟学习 react配套的类型检测库——prop-types的运用
js 有时在定义变量的类型为number 或string 时并不会报错,所以prop-types 是专门用来检测react ,以前的版本是把它放到react架构里面 ,现在作为一个独立的库搬出来了,跟 ...
- 使用cpanel后台的“时钟守护作业”功能完成空间的定时全备份
现在不少虚拟主机都是使用的cpanel控制面板,由于空间商选用的cpanel版本不同,有的带有定时备份功能,而有的就没有这项功能,需要手动备份.不过,还在绝大部分的cpanel后台都有“时钟守护作业” ...
- Linux 进程间通信 信号灯集
1.特点: 信号灯集,是控制访问临界资源 信号灯(semaphore),也叫信号量.它是不同进程间或一个给定进程内部不同线程间同步的机制System V的信号灯是一个或者多个信号灯的一个集合(允许对 ...
- RHEL5/6/7中常用命令及命令之间的差异
System basics Task RHEL5 RHEL6 RHEL7 View subscription information /etc/sysconfig/rhn/systemid /etc/ ...
- tf.matmul() 和tf.multiply() 的区别
1.tf.multiply()两个矩阵中对应元素各自相乘 格式: tf.multiply(x, y, name=None) 参数: x: 一个类型为:half, float32, float64, u ...
- redis主从和集群搭建
主从搭建 redis的主从搭建非常简单,打开配置文件6379.conf,只需要将主节点的protected-mode设置为no,然后在从节点配置中加入:slaveof <masterip> ...
- 利用html2canvas截图,得到base64上传ajax
<script type="text/javascript" src="js/html2canvas.js"></script> //布 ...