数据结构与算法 - 递归

一、递归

1. 概述

定义:在计算机科学中,递归是一种解决计算问题的方法,其中解决方案取决于同一类问题的更小子集。

比如单链表递归遍历的例子:

void f(Node node) {if(node == null) {return;}println("before:" + node.value)f(node.next);println("after:" + node.value)
}

说明:

  • 自己调用自己,如果说每个函数对应着一种解决方案,自己调用自己意味着解决方案是一样的(有规律的)
  • 每次调用,函数处理的数据会较上次缩减(子集),而且最后会缩减至无需继续递归
  • 内层函数调用(子集处理)完成,外层函数才能算调用完成

原理

假设链表中有3个节点,value分别为1,2,3,以上代码的执行流程类似于下面的伪码

// 1 -> 2 -> 3 -> null  f(1)void f(Node node = 1) {println("before:" + node.value) // 1void f(Node node = 2) {println("before:" + node.value) // 2void f(Node node = 3) {println("before:" + node.value) // 3void f(Node node = null) {if(node == null) {return;}}println("after:" + node.value) // 3}println("after:" + node.value) // 2}println("after:" + node.value) // 1
}

思路
1. 确定能否使用递归求解

2. 推导出递推关系,即父问题与子问题的关系,以及递归的结束条件

例如之前遍历链表的递推关系为

  • 深入到最里层叫递
  • 从最里层处理叫归
  • 在递的过程中,外层函数内的局部变量(以及方法参数)并未消失,归的时候还可以用到

 2. 单路递归Single Recursion

2.1 阶乘

用递归方法求阶乘

  • 阶乘的定义:n! = 1 * 2 * 3 ... (n - 2) * (n - 1) * n,其中n为自然数,当然 0! = 1
  • 递推关系

代码

private static int f(int n) {if (n == 1) {return 1;}return n * f(n - 1);
}

拆解伪码如下,假设n初始值为3

f(int n = 3) { // 解决不了,递return 3 * f(int n = 2) { // 解决不了,继续递return 2 * f(int n = 1) {if (n == 1) { // 可以解决, 开始归return 1;}}}
}

 2.1 反向打印字符串

用递归反向打印字符串,n 为字符在整个字符串 str 中的索引位置

  • :n 从 0 开始,每次 n + 1,一直递到 n == str.length() - 1

  • :从 n == str.length() 开始归,从归打印,自然是逆序的

递推关系

代码为

public static void reversePrint(String str, int index) {if (index == str.length()) {return;}reversePrint(str, index + 1);System.out.println(str.charAt(index));
}

2.2 二分查找(单路递归)

public static int binarySearch(int[] a, int target) {return recursion(a, target, 0, a.length - 1);
}public static int recursion(int[] a, int target, int i, int j) {if (i > j) {return -1;}int m = (i + j) >>> 1;if (target < a[m]) {return recursion(a, target, i, m - 1);} else if (a[m] < target) {return recursion(a, target, m + 1, j);} else {return m;}
}

2.3 冒泡排序(单路递归)

  • 将数组划分为两部分[0 ... j] [j + 1 ... a.length - 1]
  • 左边[0 ... j]是未排序部分
  • 右边[j + 1 ... a.length - 1]是已排序部分
  • 未排序区间内,相邻的两个元素比较,如果前一个大于后一个,则交换位置
public static void main(String[] args) {int[] a = {3, 2, 6, 1, 5, 4, 7};bubble(a, 0, a.length - 1);System.out.println(Arrays.toString(a));
}private static void bubble(int[] a, int low, int high) {if(low == high) {return;}int j = low;for (int i = low; i < high; i++) {if (a[i] > a[i + 1]) {swap(a, i, i + 1);j = i;}}bubble(a, low, j);
}private static void swap(int[] a, int i, int j) {int t = a[i];a[i] = a[j];a[j] = t;
}
  • low与high为未排序范围
  • j表示的是未排序的边界,下一次递归时的high
  • 发生交换,意味着有无序情况
  • 最后一次交换(以后没有无序)时,左侧i仍是无序,右侧i+1已然有序

private static void bubble(int[] a, int j) {if (j == 0) {return;}int x = 0; // x右侧的元素都是有序的for(int i = 0; i < j; i++) {{if(a[i] > a[i + 1]) {{int t = a[i];a[i] = a[i + 1];a[i + 1] = t;x = i;}}bubble(a, x);
}

C++版冒泡排序:exchange右侧都是有序的

#include<iostream>
#include<vector>
using namespace std;void BubbleSort(vector<int>& data, int length){int j, exchange, bound, temp;exchange = length - 1;  // 第一趟冒泡排序的区间是[0 ~ length - 1]while(exchange != 0){bound = exchange;exchange = 0;for(j = 0; j < bound; j++)  // 一趟冒泡排序的区间是[0 ~ bound]{if(data[j] > data[j + 1]){temp = data[j];data[j] = data[j + 1];data[j + 1] = temp;exchange = j;  // 记载每一次记录交换的位置 }} } 
}

2.4 插入排序(单路递归)

public static void main(String[] args) {int[] a = {3, 2, 6, 1, 5, 7, 4};insertion(a, 1, a.length - 1);System.out.println(Arrays.toString(a));
}private static void insertion(int[] a, int low, int high) {if (low > high) {return;}// low指针指的是未排序区的边界int i = low - 1;  // 已排序区的指针int t = a[low];while (i >= 0 && a[i] > t) {  // 没有找到插入位置a[i + 1] = a[i];  // 空出插入位置i--;}if(i + 1 != low) {a[i + 1] = t;}    insertion(a, low + 1, high);
}
public void insertionSort(int[] nums, int length) {int i, j, temp;for(i = 1; i < length; i++) {temp = nums[i];for(j = i - 1; j >= 0 && temp < nums[j]; j--) {  // 寻找插入位置nums[j + 1] = nums[j];  // 后移}nums[j + 1] = temp;}
}

2.5 约瑟夫问题(单路递归)

n个人排成圆圈,从头开始报数,每次数到第m个人杀之,继续从下一个人重复以上过程,求最后活下来的人是谁?

解法一:

设josephus(n, m)表示在n个人中,每m个人被杀时最后存活的人。递归关系为:

josephus(n, m) = (josephus(n - 1, m) + m) % n,  n > 1  
josephus(1, m) = 0

这个关系的意思是,若我们知道n-1个人时的存活者,我们可以通过调整索引来得出n个人时的存活者。

class Solution {  public int findTheWinner(int n, int m) {  return josephus(n, m) + 1; // 转换为1-based索引  }  private int josephus(int n, int m) {  if (n == 1) return 0; // 只有一个人时,返回其索引0  return (josephus(n - 1, m) + m) % n; // 递归公式  }  public static void main(String[] args) {  Solution solution = new Solution();  int n = 7; // 总人数  int m = 3; // 每次报数到m的人被杀  int winner = solution.findTheWinner(n, m);  System.out.println("最后存活下来的人的位置是: " + winner); // 输出最后存活者的1-based索引  }  
}

3 多路递归 Multi Recursion

如果每个递归函数例包含多个自身调用,称之为multi recursion

3.1 裴波那契数列

假设你正在爬楼梯。需要 n 阶你才能到达楼顶。

每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢?

示例 1:

输入:n = 2
输出:2
解释:有两种方法可以爬到楼顶。
1. 1 阶 + 1 阶
2. 2 阶

示例 2:

输入:n = 3
输出:3
解释:有三种方法可以爬到楼顶。
1. 1 阶 + 1 阶 + 1 阶
2. 1 阶 + 2 阶
3. 2 阶 + 1 阶

提示:

  • 1 <= n <= 45

Java实现

解法一:递归,会超时

class Solution {public int climbStairs(int n) {if(n == 1) {return 1;}if(n == 2) {return 2;}return climbStairs(n - 1) + climbStairs(n - 2);}
}

递归的次数也符合斐波那契数列规律:2 * f(n + 1) - 1

时间复杂度推导过程:

解法二:

class Solution {public int climbStairs(int n) {if(n == 1) {return 1;}if(n == 2) {return 2;}int prev = 1;int curr = 2;for(int i = 3; i <= n; i++) {int temp = curr;curr = prev + curr;prev = temp;}return curr;}
}

3.2 兔子问题

第一个月,有一对未成熟的兔子。第二个月,它们成熟。第三个月,它们能产下一对新的小兔子。所有兔子遵循相同规律,求第n个月的兔子数量。

分析:设第n个月兔子数为f(n)

  • f(n) = 上个月兔子数 + 新生的小兔子数
  • 新生的小兔子数 = 上个月成熟的兔子数
  • 因为需要一个月兔子就成熟,所以【上个月成熟的兔子数】也就是【上上个月的兔子数
  • 上个月兔子数,即f(n - 1);上上个月的兔子数,即f(n-2)

递推公式:f(n) = f(n -1) + f(n - 2)

class Solution {public int f(int n) {if(n == 1 || n == 2) {return 1;}int a = 1; // f(n-2)int b = 1; // f(n-1)int count = 0;for(int i = 3; i <= n; i++) {count = a + b;a = b;  // 更新f(n-2)b = count;   // 更新f(n-1)}return count;}
}

3.3 汉诺塔问题(多路递归)

Tower of Hanoi,是一个源于印度古老传说:大梵天创建世界时做了三根金刚石柱,在一根柱子从下往上按大小顺序摞着 64 片黄金圆盘,大梵天命令婆罗门把圆盘重新摆放在另一根柱子上,并且规定

  • 一次只能移动一个圆盘

  • 小圆盘上不能放大圆盘

下面的动图演示了4片圆盘的移动方法

解法一:递归。时间复杂度:O(2^n)

递归分解:

  • 若有n个盘子,以A、B和C分别表示起始柱、辅助柱和目标柱
  • 首先,将前n-1个盘子从柱A移动到柱B,使用柱C作为辅助
  • 接下来,将第n个盘子(最大的盘子)从柱A移动到柱C
  • 最后,将n-1个盘子从柱B移动到柱C,使用柱A作为辅助
class Solution {  public void hanota(List<Integer> A, List<Integer> B, List<Integer> C) {  int n = A.size();  moveDisks(n, A, C, B); // A -> C, B为辅助  }  private void moveDisks(int n, List<Integer> from, List<Integer> to, List<Integer> aux) {  // 基本情况:如果没有盘子要移动,直接返回  if (n == 0) {  return;  }  // 1. 将 n-1 个盘子从 "from" 移动到 "aux",使用 "to" 作为辅助  moveDisks(n - 1, from, aux, to);  // 2. 将第 n 个盘子从 "from" 移动到 "to"  to.add(from.remove(from.size() - 1)); // 从 "from" 中移除最后一个盘子并添加到 "to"  // 3. 将 n-1 个盘子从 "aux" 移动到 "to",使用 "from" 作为辅助  moveDisks(n - 1, aux, to, from); }  
}  
package com.itheima.algorithm.recursion_multi;import org.springframework.util.StopWatch;import java.util.LinkedList;/*** 递归汉诺塔*/
public class E02HanoiTower {static LinkedList<Integer> a = new LinkedList<>();static LinkedList<Integer> b = new LinkedList<>();static LinkedList<Integer> c = new LinkedList<>();static void init(int n) {for (int i = n; i >= 1; i--) {a.addLast(i);}}/*** <h3>移动圆盘</h3>** @param n 圆盘个数* @param a 由* @param b 借* @param c 至*/static void move(int n, LinkedList<Integer> a,LinkedList<Integer> b,LinkedList<Integer> c) {if (n == 0) {return;}move(n - 1, a, c, b);   // 把 n-1 个盘子由a,借c,移至bc.addLast(a.removeLast()); // 把最后的盘子由 a 移至 c
//        print();move(n - 1, b, a, c);   // 把 n-1 个盘子由b,借a,移至c}// T(n) = 2T(n-1) + c,  O(2^64)public static void main(String[] args) {StopWatch sw = new StopWatch();int n = 1;init(n);print();sw.start("n=1");move(n, a, b, c);sw.stop();print();System.out.println(sw.prettyPrint());}private static void print() {System.out.println("----------------");System.out.println(a);System.out.println(b);System.out.println(c);}
}

3.3 杨辉三角

给定一个非负整数 numRows生成「杨辉三角」的前 numRows 行。

在「杨辉三角」中,每个数是它左上方和右上方的数的和。

示例 1:

输入: numRows = 5
输出: [[1],[1,1],[1,2,1],[1,3,3,1],[1,4,6,4,1]]

示例 2:

输入: numRows = 1
输出: [[1]]

提示:

  • 1 <= numRows <= 30

递推公式:triangle[i][j] = triangle[i - 1][j - 1] + triangle[i - 1][j]

解法一:迭代

class Solution {  public List<List<Integer>> generate(int numRows) {  List<List<Integer>> triangle = new ArrayList<>();  for (int i = 0; i < numRows; i++) {  List<Integer> row = new ArrayList<>(i + 1);  for (int j = 0; j <= i; j++) {  if (j == 0 || j == i) {  row.add(1); // 第一列和最后一列都为1  } else {  // 当前元素等于上一行的两个元素之和  row.add(triangle.get(i - 1).get(j - 1) + triangle.get(i - 1).get(j));  }  }  triangle.add(row); // 将当前行添加到三角形中  }  return triangle;  }  
}

解法二:递归。这个递归方法对于较大的 numRows(接近 30)可能会非常低效,因为会有大量重复计算。对于较大的输入,推荐使用迭代的方法。

class Solution {  public List<List<Integer>> generate(int numRows) {  List<List<Integer>> triangle = new ArrayList<>();  for(int i = 0; i < numRows; i++) {triangle.add(generateRow(i));}return triangle;  }  private List<Integer> generateRow(int rowIndex) {List<Integer> row = new ArrayList<>();for(int j = 0; j <= rowIndex; j++) {row.add(getValue(rowIndex, j));}return row;}private int getValue(int row, int col) {if(col == 0 || col == row) {return 1;}return getValue(row - 1, col - 1) + getValue(row - 1, col);}
}

解法二:递归优化。备忘录模式

二维数组记忆法

class Solution {  public List<List<Integer>> generate(int numRows) {  List<List<Integer>> triangle = new ArrayList<>();  // 创建一个二维数组用于存储已计算的值  Integer[][] memo = new Integer[numRows][numRows];  for (int i = 0; i < numRows; i++) {  triangle.add(new ArrayList<>());  for (int j = 0; j <= i; j++) {  triangle.get(i).add(getValue(i, j, memo));  }  }  return triangle;  }  private int getValue(int row, int col, Integer[][] memo) {  // 如果位置超出边界,返回 0  if (col < 0 || col > row) {  return 0;  }  // 如果是边界元素(最左或最右),返回 1  if (col == 0 || col == row) {  return 1;  }  // 如果已经计算过,直接返回  if (memo[row][col] != null) {  return memo[row][col];  }  // 递归计算并存储结果到 memo  memo[row][col] = getValue(row - 1, col - 1, memo) + getValue(row - 1, col, memo);  return memo[row][col];  }  
}

解法三:一维数组记忆法。

    private static void createRow(int[] row, int i) {if (i == 0) {row[0] = 1;return;}for (int j = i; j > 0; j--) {// 下一行当前项等于 上一行的当前项 + 前一项row[j] = row[j] + row[j - 1];}}public static void print2(int n) {int[] row = new int[n];for (int i = 0; i < n; i++) { // 行createRow(row, i);
//            printSpace(n, i);for (int j = 0; j <= i; j++) {System.out.printf("%-4d", row[j]);}System.out.println();}}

4. 递归优化 - 记忆法

斐波那契数列计算过程中存在很多重复的计算,例如求f(5)的递归分解过程

可以看到(颜色相同的是重复的)

  • f(3)重复了2次
  • f(2)重复了3次
  • f(1)重复了5次
  • f(0)重复了3次

随着n的增大,重复次数非常可观,如何优化呢?

Menoization记忆法(也称备忘录)是一种优化技术,通过存储函数调用结果(通常比较昂贵),当再次出现相同的输入(子问题)时,就能实现加速效果。

public static void main(String[] args) {int n = 13;int[] cache = new int[n + 1];Arrays.fill(cache, -1);cache[0] = 0;cache[1] = 1;System.out.println(f(cache, n));
}public static int f(int[] cache, int n) {if (cache[n] != -1) {return cache[n];}cache[n] = f(cache, n - 1) + f(cache, n - 2);return cache[n];
}

优化后的图示,只要结果被缓存,就不会执行其子问题

  • 改进前的时间复杂度为Θ(1.618^n);空间复杂度O(n)
  • 改进后的时间复杂度为O(n);空间复杂度为O(n),额外的空间缓存结果

5. 递归优化 - 尾递归

用递归做 n + (n - 1) + (n - 2) + ... + 1

public static long sum(long n) {if(n == 1) {return 1;}return n + sum(n - 1);
}

在我的机器上n = 12000时爆栈了

Exception in thread "main" java.lang.StackOverflowErrorat Test.sum(Test.java:10)at Test.sum(Test.java:10)at Test.sum(Test.java:10)at Test.sum(Test.java:10)at Test.sum(Test.java:10)...

为什么呢?

  • 每次方法调用都是需要消耗一定的栈内存的,这些内存用来存储方法参数、方法内局部变量、返回地址等。
  • 方法调用占用的内存需要等待方法结束时才会释放
  • 而递归调用不到最深不会回头,最内层方法没完成之前,外层方法都结束不了。

尾调用

如果函数的最后一步是调用一个函数,那么称为尾调用,例如

function a() {return b()
}

下面三段代码不能叫作尾调用

function a() {const c = b()return c
}
  • 最后一步并非调用函数
function a() {return b() + 1
}
  • 最后一步执行的是加法
function a(x) {return b() + x
}
  • 最后一步执行的是加法

一些语言的编译器能够对尾调用做优化,例如

function a() {// 做前面的事return b() 
}function b() {// 做前面的事return c()
}function c() {return 1000
}a()

没优化之前的伪码

function a() {return function b() {return function c() {return 1000}}
}

优化后的伪码如下

a()
b()
c()

为何尾递归才能优化?

调用a时

  • a返回时发现,没什么可留给b的,将来返回的结果b提供就可以了,用不着我a了,我的内存就可以释放

调用b时

  • b返回时发现,没什么可以留给c的,将来返回的结果c提供就可以了,用不着我b了,我的内存就可以释放了

如果调用a时

  • 不是尾调用,例如return b() + 1,那么a就不能提前结束,因为它还得利用b的结果做加法

尾递归:是尾调用的一种特例,也就是最后一步执行的是同一个函数

尾递归避免爆栈

安装Scala

Scala入门

object Main {def main(args: Array[String]): Unit = {println("Hello Scala")}
}
  • Scala是Java的近亲,Java中的类都可以拿来重用
  • 类型是放在变量后面的
  • Unit表示无返回值,类似于void
  • 不需要以分号作为结尾,当然加上也对

先写一个会爆栈的函数

def sum(n: Long): Long = {if (n == 1) {return 1}return n + sum(n - 1)
}

Scala最后一行代码若作为返回值,可以省略return

在n = 11000时出了异常

println(sum(11000))Exception in thread "main" java.lang.StackOverflowErrorat Main$.sum(Main.scala:25)at Main$.sum(Main.scala:25)at Main$.sum(Main.scala:25)at Main$.sum(Main.scala:25)...

这时因为以上代码,这不是尾调用,要想成为尾调用,那么:

最后一行代码,必须是一次函数调用

内层函数必须摆脱与外层函数的关系,内层函数执行后不依赖于外层的变量或常量

def sum(n: Long): Long = {if (n == 1) {return 1}return n + sum(n - 1)  // 依赖于外层函数的 n 变量
}

如何让它执行后就拜托对n的依赖呢?

  • 不能等递归回来再做加法,那样就必须保留外层的n
  • 把n当作内层函数的一个参数传进去,这时n就属于内层函数了
  • 传参时就完成累加,不必等回来时累加
sum(n - 1, n + 累加器)

改写后代码如下

@tailrec
def sum(n: Long, accumulator: Long): Long = {if (n == 1) {return 1 + accumulator} return sum(n - 1, n + accumulator)
}

accumulator作为累加器

@tailrec注解是scala提供的,用来检查方法是否符合尾递归

这回sum(10000000, 0)也没有问题,打印50000005000000

执行流程如下,以伪码表示sum(4, 0)

// 首次调用
def sum(n = 4, accumulator = 0): Long = {return sum(4 - 1, 4 + accumulator)
}// 接下来调用内层 sum, 传参时就完成了累加, 不必等回来时累加,当内层 sum 调用后,外层 sum 空间没必要保留
def sum(n = 3, accumulator = 4): Long = {return sum(3 - 1, 3 + accumulator)
}// 继续调用内层 sum
def sum(n = 2, accumulator = 7): Long = {return sum(2 - 1, 2 + accumulator)
}// 继续调用内层 sum, 这是最后的 sum 调用完就返回最后结果 10, 前面所有其它 sum 的空间早已释放
def sum(n = 1, accumulator = 9): Long = {if (1 == 1) {return 1 + accumulator}
}

本质上,尾递归优化就是将函数的递归调用,变成了函数的循环调用

改循环避免爆栈

public static void main(String[] args) {long n = 100000000;long sum = 0;for (long i = n; i >= 1; i--) {sum += i;}System.out.println(sum);
}

6. 递归时间复杂度 - 主定理(Master theorem)

若有递归式

其中

  • T(n)是问题的运行时间,n是数据规模
  • a是子问题的个数
  • T(n/b)是子问题运行时间,每个子问题被拆成原问题数据规模的n/b
  • f(n)是除递归外执行的计算

x = log_b a,即x = log_子问题缩小倍数 子问题个数,那么

例1:T(n) = 2T(n/2) + n ^ 4

  • 此时x = 1 < 4,由后者决定整个时间复杂度Θ(n^4)

例2:T(n) = T(7n / 10) + n

  • 此时x = 0 < 1,由后者决定整个时间复杂度Θ(n)

例3:T(n) = 16T(n/4) + n^2

  • a = 16, b = 4, x = 2, x = 2;此时x = c = 2,时间复杂度为Θ(n^2 * logn)

例4:T(n) = 7T(n/3) + n^2

  • a = 7, b = 3, c = 2, x = 1.?;此时x < c,由后者决定整个时间复杂度Θ(n^2)

例5:T(n) = 7T(n/2) + n^2

  • a = 7, b = 2, c = 2, x = 2.? > c,由前者决定整个时间复杂度Θ(n^log_2 7)

例6:T(n) = 2T(n/4) + sqrt(n)

  • a = 2, b = 4, c = 1/2,x = c,时间复杂度为Θ(sqrt(n) * logn)

例7:二分查找递归

int f(int[] a, int target, int i, int j) {if (i > j) {return -1;}int m = (i + j) >>> 1;if (target < a[m]) {return f(a, target, i, m - 1);} else if (a[m] < target) {return f(a, target, m + 1, j);} else {return m;}
}
  • 子问题个数a = 1
  • 子问题数据规模缩小倍数b = 2
  • 除递归之外执行的计算是常数级c = 0
  • 此时x = 0 = c,时间复杂度为Θ(logn)

例8:归并排序递归

void split(B[], i, j, A[])
{if (j - i <= 1)                    return;                                m = (i + j) / 2;             // 递归split(A, i, m, B);  split(A, m, j, B); // 合并merge(B, i, m, j, A);
}
  • 子问题个数a = 2
  • 子问题数据规模缩小倍数b = 2
  • 除递归外,主要时间花在合并上,它可以用f(n) = n表示
  • T(n) = 2T(n/2) + n,此时x = 1 = c,时间复杂度Θ(nlogn)

例9:快速排序递归

algorithm quicksort(A, lo, hi) is if lo >= hi || lo < 0 then return// 分区p := partition(A, lo, hi) // 递归quicksort(A, lo, p - 1) quicksort(A, p + 1, hi) 
  • 子问题个数 a=2

  • 子问题数据规模缩小倍数

    • 如果分区分的好,b=2

    • 如果分区没分好,例如分区1 的数据是 0,分区 2 的数据是 n-1

  • 除递归外,主要时间花在分区上,它可以用 $f(n) = n 表示

情况1 - 分区分的好

T(n) = 2T(n\2) + n

  • 此时 x=1=c,时间复杂度 Θ(nlogn)

情况2 - 分区没分好

T(n) = T(n-1) + T(1) + n

  • 此时不能用主定理求解

7. 递归时间复杂度 - 展开求解

像下面的递归式,都不能用主定理求解

例1:递归求和

long sum(long n) {if (n == 1) {return 1;}return n + sum(n - 1);
}

T(n) = T(n - 1) + c,T(1) = c

下面为展开过程

T(n) = T(n-2) + c + c

T(n) = T(n-3) + c + c + c

...

T(n) = T(n-(n-1)) + (n-1)c

  • 其中 T(n-(n-1)) 即 T(1)

  • 带入求得 T(n) = c + (n-1)c = nc

时间复杂度为 O(n)

例2:递归冒泡排序

void bubble(int[] a, int high) {if(0 == high) {return;}for (int i = 0; i < high; i++) {if (a[i] > a[i + 1]) {swap(a, i, i + 1);}}bubble(a, high - 1);
}

T(n) =T(n - 1) + n,T(1) = c

下面为展开过程

T(n) = T(n-2) + (n-1) + n

T(n) = T(n-3) + (n-2) + (n-1) + n

...

T(n) = T(1) + 2 + ... + n = T(1) + (n-1)*(2+n)/2 = c + n^2/2 + n/2 -1

时间复杂度 O(n^2)

注:等差数列求和为 个数 * (首项 + 末项)/2

例3:递归快排

快速排序分区没分好的极端情况

T(n) = T(n-1) + T(1) + n,T(1) = c

T(n) = T(n-1) + c + n

下面为展开过程

T(n) = T(n-2) + c + (n-1) + c + n

T(n) = T(n-3) + c + (n-2) + c + (n-1) + c + n

...

T(n) = T(n-(n-1)) + (n-1)c + 2+...+n = n^2 / 2 + (2cn + n) / 2 -1

时间复杂度 O(n^2)

不会推导的同学可以进入 Wolfram|Alpha: Computational Intelligence

  • 例1 输入 f(n) = f(n - 1) + c, f(1) = c

  • 例2 输入 f(n) = f(n - 1) + n, f(1) = c

  • 例3 输入 f(n) = f(n - 1) + n + c, f(1) = c

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

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

相关文章

Animate软件基础:在时间轴中添加或插入帧

FlashASer&#xff1a;AdobeAnimate2021软件零基础入门教程https://zhuanlan.zhihu.com/p/633230084 FlashASer&#xff1a;实用的各种Adobe Animate软件教程https://zhuanlan.zhihu.com/p/675680471 FlashASer&#xff1a;Animate教程及作品源文件https://zhuanlan.zhihu.co…

软件测试--python基础

一、python基础 (1)第一个python (2)python解释器 (3)基础语法 ①字面量 什么是字面量 常用的值类型 字符串 ②注释 ③变量 什么是变量 变量的特征 变量的目的是存储运行过程的数据 存储的目的是为了&#xff1a;重复使用 ④数据类型 type()语句 变量有类型吗&#xff1f;…

在欧拉系统中安装数据库

在欧拉系统中的安装 &#xff08;禁止超级用户root登录&#xff09; yum install mariadb-server -y #下载命令 systemctl enable --now mariadb #设置为开机自启&#xff0c;并立即启动该服务 mysql_secure_installation #安全设置&#xff0c;初始化 修…

C语言数据在内存中的存储超详解

文章目录 1. 整数在内存中的存储2. 大小端字节序和字节序判断2. 1 什么是大小端&#xff1f;2. 2 为什么会有大小端&#xff1f;2. 3 练习 3. 浮点数在内存中的存储3. 1 一个代码3. 2 浮点数的存储3. 2. 1 浮点数存的过程3. 2. 2 浮点数取的过程3. 3 题目解析 1. 整数在内存中的…

ctfshow-web入门-sql注入(web171-web175)

目录 1、web171 2、web172 3、web173 4、web174 5、web175 1、web171 单引号测一下&#xff0c;报错 -- 闭合后回显正常 也可以用 # &#xff0c;不过需要 URL 编码 成功闭合之后&#xff0c;先判断下字段数&#xff1a; 1 order by 3-- 3 的时候正常 4 的时候报错&am…

C++必修:STL之vector的了解与使用

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;C学习 贝蒂的主页&#xff1a;Betty’s blog 1. C/C中的数组 1.1. C语言中的数组 在 C 语言中&#xff0c;数组是一组相同类型…

泰迪智能科技AI大模型某医院合作案例介绍

泰迪智能科技AI大模型支持以ChatGLM2-6B、Baichuan-13B、Qwen14B和文心一言等多种大语言模型为底座&#xff0c;实现基于特定领域数据、面向智能客服、问答系统、自动摘要、智能打标、内容创作、信息抽取等应用场景的模型微调、评估和推理&#xff0c;为业务智能升级和价值挖掘…

【C++从小白到大牛】类和对象

目录 一、面向过程和面向对象初步认识 二、类的引入 三、类的定义 类的成员函数两种定义方式&#xff1a; 1. 声明和定义全部放在类体中 2. 类声明放在.h文件中&#xff0c;成员函数定义放在.cpp文件中 成员变量命名规则的建议&#xff1a; 四、类的访问限定符 【访问限…

本地部署持续集成工具Jenkins并配置公网地址实现远程自动化构建

文章目录 前言1. 安装Jenkins2. 局域网访问Jenkins3. 安装 cpolar内网穿透软件4. 配置Jenkins公网访问地址5. 公网远程访问Jenkins6. 固定公网地址 前言 本文主要介绍如何在Linux CentOS 7中安装Jenkins并结合cpolar内网穿透工具实现远程访问管理本地部署的Jenkins服务. Jenk…

大模型算法面试题(十八)

本系列收纳各种大模型面试题及答案。 1、P-tuning v2 思路、优缺点是什么 P-tuning v2是清华大学自然语言处理实验室&#xff08;THUDM&#xff09;等研究机构提出的一种新的预训练模型优化方法&#xff0c;主要关注如何通过动态构建任务相关的提示序列来引导预训练模型进行更…

【数据结构进阶】手撕红黑树

&#x1f525;个人主页&#xff1a; Forcible Bug Maker &#x1f525;专栏&#xff1a; C || 数据结构 目录 &#x1f308;前言&#x1f525;红黑树的概念&#x1f525;手撕红黑树红黑树结点的定义红黑树主体需要实现的成员函数红黑树的插入findEmpty和Size拷贝构造析构函数和…

Redis和Mysql如何保持数据一致性

一般情况下&#xff0c;Redis是用来实现应用和数据库之间读操作得缓存层&#xff0c;主要目的是减少数据库IO&#xff0c;还可以提升数据的IO性能。 当应用程序需要去读取某个数据时&#xff0c;会首先尝试去Redis里面加载&#xff0c;如果命中就直接返回&#xff0c;如果没有…

C++ 操作Git仓库

代码 #include "common.h" #include "args.c" #include "common.c"enum index_mode {INDEX_NONE,INDEX_ADD };struct index_options {int dry_run;int verbose;git_repository* repo;enum index_mode mode;int add_update; };/* Forward declar…

vue项目Nginx部署启动

1.vue打包 &#xff08;1&#xff09;package.json增加打包命令 "scripts": {"dev": "webpack-dev-server --inline --progress --config build/webpack.dev.conf.js --host 10.16.14.110","start": "npm run dev","un…

Halcon 边缘提取(亚像素)

Halcon提供多种边缘提取算法。像素提取方法有常用的边缘提取算子或深度学习分割模型等。考虑到精度问题可能需要提取亚像素边缘。当然也可以提取轮廓&#xff1a;线、圆、椭圆等。本文只讨论提取轮廓。 1 基本概念 正常情况下&#xff0c;无需特殊操作即可提取边缘轮廓。 1…

Linux-4:Shell编程——基础语法(50%-100%)

目录 前言 一、数组 1.数组定义 2.关联数组 3.数组长度 二、运算符 1.算术运算符 2.关系运算符 3.布尔运算符 4.逻辑运算符 5.字符串运算符 6.文件测试运算符 三、read命令 1.接收用户输入 2.开启转义 3. -p 输入提示 4. -s 静默模式 -t 设置超时时间 5.读取…

Fiddler学习笔记

目录 前言 简介 原理 界面 前言 测试可以使用fiddler工具&#xff0c;通过抓包的方式修改前端参数和模拟后端返回&#xff0c;快速定位缺陷。 简介 Fiddler是HTTP协议调试代理工具&#xff0c;可以记录并检查所有客户端和服务器之间的HTTP和HTTPS请求&#xff0c;允许监视…

算法训练1

01背包问题 背包状态方程----动态规划 二维dp 使用 f[i][j] max(f[i-1][j] ,f[i-1][j - w[i]] v[i]); 伪代码&#xff1a; int dp[100][100]; void test6() {int n; //装备数量int m; //背包容量int v[105], w[105]; //前面空间&#xff0c;后面价值for (int i 1; i &l…

ONLYOFFICE文档:为企业和开发者带来强大的文档编辑功能

本文给大家介绍一个开源项目&#xff1a;ONLYOFFICE文档&#xff0c;它能够为文档编辑、多人协作提供强大支持。无论你是个人使用&#xff0c;还是企业、商业开发&#xff0c;都能找到适合你的版本。 关于 ONLYOFFICE 文档 ONLYOFFICE 文档是一套功能强大的文档编辑器&#x…

微信小程序获取AppSecret的步骤

文章目录 微信小程序获取AppSecret的步骤&#xff1a;注意&#xff1a; 微信公众平台 小程序的密钥&#xff08;或称为AppSecret&#xff09;是用于加密解密、验证服务器身份等安全操作的敏感信息。不同的平台&#xff08;如微信小程序、支付宝小程序、百度智能小程序等&am…