Java开发 - 递归

简介: Java开发 - 递归

目录


前言

什么是递归

递归不得不知的经典案例:斐波那契数列

递归注意点

SOF和OOM

结语


前言


在讲到二叉树之前,一定要先把递归给讲了,只有先把基础学好了,才可以让学习的这棵二叉树更完整,更稳固一些。所以接下来,我会用代码和文字把递归的思想传递给大家,希望大家可以学到自己感兴趣的东西。


什么是递归


递归是一种思想,再变成上体现为方法的自调用。


举个例子说明:

实现某个数字的阶乘
1! = 1
2! = 1 * 2
3! = 1 * 2 * 3
4! = 1 * 2 * 3 * 4
5! = 1 * 2 * 3 * 4 * 5
6! = 1 * 2 * 3 * 4 * 5 * 6

我们最终要做的就是写一个方法,返回一个值:

public static int fun(int n) {
    return ......
}

我们需要考虑下方法块中要怎么写,观察阶乘,我们发现:

1! = 1
2! = 1! * 2
3! = 2! * 3
4! = 3! * 4
5! = 4! * 5
6! = 5! * 6

我好像发现了什么,如果我要获得n的阶乘,那是不是可以用n-1的阶乘乘以n?n-1的阶乘呢?是n-2的阶乘乘以n-1,依此类推。


方法块内,我们可以这样写:

public static int fun(int n) {
    return fun(n - 1) * n;
}

fun(n - 1)就是n-1的阶乘,这就形成了递归,然后一直递归下去。但是有个问题,这里没有考虑n=1的情况,最终会出现负数,就没有了出口,这也是递归最重要的一点,一定要有出口,否则就是死循环。所以应该这么写:

public static int fun(int n) {
    if (n == 1)
        return 1;
    return fun(n - 1) * n;
}

这样,一个求阶乘的方法就写好了,你以为很难?其实就这么简单,用事实验证了:越是难的东西越是需要简单来做,简单到你不相信这就是答案。


递归不得不知的经典案例:斐波那契数列


斐波那契数列指的是这样一个数列:1,1,2,3,5,8,13,21,34,55,89......


这个数列从第3项开始,每一项都等于前两项之和。


假如我要求第50位的数字,该怎么求?有上面的案例,不知道大家能不能自己写出来呢?


写不出来没关系,我将带领大家一起来解题,请大家跟上我的思路,我们开始。


我们先写一个方法体出来:

public static int fun(int n) {
    return ......
}

方便起见,我们就用上面的方法体,因为我要得到多少位的数字我只需要输入这个数,方法就会把这一位的数字返回给我。


上面文字已经说明:这个数列从第3项开始,每一项都等于前两项之和


那么规律就已经出现了,我们可以假设:


fun(50) = fun(49) + fun(48)


fun(49) = fun(48) + fun(47)


fun(48) = fun(47) + fun(46)


......


这里递归已经开始了,我们可以尝试着写一下这个方法块中的内容了:

public static int fun(int n) {
    return fun(n - 1) + fun(n - 2);
}

你可能已经发现了问题,和前面一样的问题:没有考虑n=1的情况,所以就没有了出口。没错,但这里还要考虑n=2的情况 ,因为是从第三位开始数列才成立。


所以需要像上面那样,通过判断1,2给方法留出口:

public static int fun(int n) {
    if (n == 1 || n == 2)
        return 1;
    return fun(n - 1) + fun(n - 2);
}


递归注意点


其实我们上面已经说到了一些注意点了,不知道大家能不能说下来?


没记住的也没关系,跟着博主一起看:


递归必须有出口,否则会导致栈内存溢出SOF(StackOverflowError);

递归是有深度的,若是深度太深,也会导致栈内存溢出SOF;

关于栈:当调用一个方法时,会在栈上为这个方法分配属于这个栈帧的内存,哪个方法的栈帧则用来保存属于哪个栈帧的局部变量。


局部变量包括:方法参数,方法内声明的一些基本变量。方法结束时,栈帧上的内存才会清除。但是栈的内存并不是无限分配的,所以当出现递归深度太深的时候会导致栈内存溢出。


SOF和OOM


SOF: statckOverFlowError   栈内存溢出

OOM: OutOfMemoryError  堆内存溢出

内存溢出指的是剩余内存不足以分配给请求的资源,此时就出现了内存溢出。可能的原因是:


创建大对象

内存泄漏的不断累积。内存泄漏累积到一定程度,会出现堆内存溢出

内存泄漏指的是分配出去的内存因为一些原因无法回收,此现象就叫内存泄漏。


内存泄漏和内存溢出的关系:

内存泄漏累积到一定程序才会造成内存溢出,并不是内存泄漏一旦出现,则立即出现内存溢出

出现内存溢出并不一定是由于内存泄漏造成的,还可能是因为创建了大对象。


结语


到这里,递归就讲完了,有不明白的小伙伴可以留言一起讨论,下一篇,我们二叉树见。

目录
相关文章
|
2月前
|
安全 前端开发 Java
《深入理解Spring》:现代Java开发的核心框架
Spring自2003年诞生以来,已成为Java企业级开发的基石,凭借IoC、AOP、声明式编程等核心特性,极大简化了开发复杂度。本系列将深入解析Spring框架核心原理及Spring Boot、Cloud、Security等生态组件,助力开发者构建高效、可扩展的应用体系。(238字)
|
3月前
|
消息中间件 人工智能 Java
抖音微信爆款小游戏大全:免费休闲/竞技/益智/PHP+Java全筏开源开发
本文基于2025年最新行业数据,深入解析抖音/微信爆款小游戏的开发逻辑,重点讲解PHP+Java双引擎架构实战,涵盖技术选型、架构设计、性能优化与开源生态,提供完整开源工具链,助力开发者从理论到落地打造高留存、高并发的小游戏产品。
|
3月前
|
存储 Java 关系型数据库
Java 项目实战基于面向对象思想的汽车租赁系统开发实例 汽车租赁系统 Java 面向对象项目实战
本文介绍基于Java面向对象编程的汽车租赁系统技术方案与应用实例,涵盖系统功能需求分析、类设计、数据库设计及具体代码实现,帮助开发者掌握Java在实际项目中的应用。
147 0
|
4月前
|
JavaScript 安全 前端开发
Java开发:最新技术驱动的病人挂号系统实操指南与全流程操作技巧汇总
本文介绍基于Spring Boot 3.x、Vue 3等最新技术构建现代化病人挂号系统,涵盖技术选型、核心功能实现与部署方案,助力开发者快速搭建高效、安全的医疗挂号平台。
256 3
|
4月前
|
安全 Java 数据库
Java 项目实战病人挂号系统网站设计开发步骤及核心功能实现指南
本文介绍了基于Java的病人挂号系统网站的技术方案与应用实例,涵盖SSM与Spring Boot框架选型、数据库设计、功能模块划分及安全机制实现。系统支持患者在线注册、登录、挂号与预约,管理员可进行医院信息与排班管理。通过实际案例展示系统开发流程与核心代码实现,为Java Web医疗项目开发提供参考。
244 2
|
4月前
|
安全 Oracle Java
JAVA高级开发必备·卓伊凡详细JDK、JRE、JVM与Java生态深度解析-形象比喻系统理解-优雅草卓伊凡
JAVA高级开发必备·卓伊凡详细JDK、JRE、JVM与Java生态深度解析-形象比喻系统理解-优雅草卓伊凡
343 0
JAVA高级开发必备·卓伊凡详细JDK、JRE、JVM与Java生态深度解析-形象比喻系统理解-优雅草卓伊凡
|
5月前
|
并行计算 Java API
Java List 集合结合 Java 17 新特性与现代开发实践的深度解析及实战指南 Java List 集合
本文深入解析Java 17中List集合的现代用法,结合函数式编程、Stream API、密封类、模式匹配等新特性,通过实操案例讲解数据处理、并行计算、响应式编程等场景下的高级应用,帮助开发者提升集合操作效率与代码质量。
251 2
|
5月前
|
安全 Java API
Java 17 及以上版本核心特性在现代开发实践中的深度应用与高效实践方法 Java 开发实践
本项目以“学生成绩管理系统”为例,深入实践Java 17+核心特性与现代开发技术。采用Spring Boot 3.1、WebFlux、R2DBC等构建响应式应用,结合Record类、模式匹配、Stream优化等新特性提升代码质量。涵盖容器化部署(Docker)、自动化测试、性能优化及安全加固,全面展示Java最新技术在实际项目中的应用,助力开发者掌握现代化Java开发方法。
262 1
|
4月前
|
移动开发 Cloud Native 安全
Java:跨平台之魂,企业级开发的磐石
Java:跨平台之魂,企业级开发的磐石