java算法 递归运算优化(备忘录模式)

一、递归是很常见的一种运算,有明显特征特征的场景一般使用递归比较合适。其一,相邻的数据存在一定的逻辑关系,其二、必须存在出口的值,如第一个元素或者最后元素的值能轻松计算出来。

二、递归和备忘录示例

1、题目:斐波那契数列: 指的是这样一个数列:1、1、2、3、5、8、13、21、34、......。求该数列中第30个数据的值。

java 复制代码
package com.zw.study.algorithm;

import java.util.HashMap;
import java.util.Map;

public class RecursionTest {
    private static int count = 0;   // 统计运算次数
    private static Map<Integer, Integer> bookMap = new HashMap<>();   // 用于存储递归计算过的结果

    public static void main(String[] args) {
    
        long startTime = System.nanoTime();
        int number = getBasicNumber(30);
        long endTime = System.nanoTime();
        System.out.println("基础递归共耗时:" + (endTime - startTime) + "纳秒");
        System.out.println("基础递归共调用方法:" + count + "次");
        System.out.println("基础递归第30个数据的值是:" + number);
        System.out.println();

        count = 0;
        startTime = System.nanoTime();
        number = getBookNumber(30);
        endTime = System.nanoTime();
        System.out.println("备忘录递归共耗时:" + (endTime - startTime) + "纳秒");
        System.out.println("备忘录递归共调用方法:" + count + "次");
        System.out.println("备忘录递归第30个数据的值是:" + number);
    }

    // 基础递归算法
    private static int getBasicNumber(int num) {
        count++;
        if (num == 1 || num == 2) {
            return 1;
        }
        return getBasicNumber(num - 1) + getBasicNumber(num - 2);
    }

    // 备忘录算法
    private static int getBookNumber(int num) {
        count++;
        if (num == 1 || num == 2) {
            return 1;
        }
        if (bookMap.containsKey(num)) {
            return bookMap.get(num);
        } else {
            int bookNumber = getBookNumber(num - 1) + getBookNumber(num - 2);
            bookMap.put(num, bookNumber);
            return bookNumber;
        }
    }
}

如上的代码,getBasicNumber为基本递归的操作,getBookNumber是使用了备忘录算法的递归操作。

运算结果如下:

从运行结果看,常规递归和备忘录算法递归都可以得到正确的结果,不过相对而言,备忘录算法的耗时和运行次数都要更优于常规递归操作。原因很简单,备忘录模式就是借助临时空间,把已经运算过的结果存储到临时空间中,下一次发现为相同运算时,就不需要再次进行逻辑运算,只需要从临时空间中取出值即可。也是一种典型的以空间换时间的想法。

学海无涯苦作舟!!!

相关推荐
xiaoye37082 小时前
Java 自动装箱 / 拆箱 原理详解
java·开发语言
YDS8294 小时前
黑马点评 —— 分布式锁详解加源码剖析
java·spring boot·redis·分布式
迷藏4944 小时前
**发散创新:基于 Rust的开源权限管理系统设计与实战**在现代软件架构中,**权限控制**早已不
java·开发语言·rust·开源
升鲜宝供应链及收银系统源代码服务4 小时前
《IntelliJ + Claude Code + Gemini + ChatGPT 实战配置手册升鲜宝》
java·前端·数据库·chatgpt·供应链系统·生鲜配送
daidaidaiyu4 小时前
Nacos实例一则及其源码环境搭建
java·spring
lvxiangyu114 小时前
MPPI 算法证明重构:基于无穷维泛函变分与 KL 散度的构造性推导
算法·重构·最优控制·随机最优控制
2301_818419015 小时前
C++中的解释器模式变体
开发语言·c++·算法
小江的记录本5 小时前
【Redis】Redis全方位知识体系(附《Redis常用命令速查表(完整版)》)
java·数据库·redis·后端·python·spring·缓存
ab1515175 小时前
3.25完成*23、*24、*28、*30、*33、*38、*39、*40
算法
摇滚侠5 小时前
Java 项目《谷粒商城-1》架构师级Java 项目实战,对标阿里 P6-P7,全网最强,实操版本
java·开发语言