个人算法题精简导航整理(精炼汇总,含知识点、模板题、题单)

前言

本章节内容主要做一个全局算法题导航指引,含有代码基本模板、相对应习题以及相关知识点,所有题目围绕这个导航索引进行补充扩展,目前博主水平有限也在不断学习更新当前博客内容。

所有博客文件目录索引:博客目录索引(持续更新)

当前博客更新日志

2023.4.5:更新区间DP思路及模板题、树形DP模板题、贪心-区间问题(区间选点、区间分组、区间覆盖)、贪心-均值不等式问题
2023.4.4:更新数学-数学-约数找一个数所有公因子模板
2023.4.3:更新数据结构-树、图的邻接表数组实现(包含dfs遍历)
2023.4.2:更新动态规划-线性DP:数字三角形(最大路径)、最长上升子序列(朴素、贪心优化)、最长公共子序列、最短编辑距离的核心思想及模板、左右区间枚举
2023.3.31:更新动态规划-背包问题:01背包、完全背包、多重背包、分组背包的思路及精简模板
2023.2-2023.4.1:更新大体框架,添加大部分模板

导航

OI Wiki:我愿称之为算法最全知识点合集!

各类算法动画手动可调试网站

旧金山大学的数据结构与算法在线动效:https://www.cs.usfca.edu/~galles/visualization/Algorithms.html

image-20220925213132044

对于Java的一些注意事项以及API可见:算法竞赛Java选手的语言快速熟悉指南

C++一般运行1秒(运行107-108)

时间复杂度

  • n ∈ 1000:O(n2),dp
  • n ∈ 10000:O(nlogn),二分、排序
    • 210=1024,220=1048576(106),230=1073741824(109),240=1e12,250=1e15。【220约等于100万,231就爆一秒了】
    • 10! = 362万,11! = 3900万,12! = 4亿,13! = 60亿

关于数据类型以及对应的内存范围

  • int:最大2147483647,20亿,2x1010,10的10次方
  • long:占8个字节,64位,[-9223372036854775808到9223372036854775807] 百亿亿,9*219,10的19次方
    • 1亿亿等于1兆,一百兆。
  • 64MB:64MB最多可以开16777216个intimage-20221002140104445 ,相当于一千六百七十万个。

调试技巧

计算机程序花费时间:

public static void main(String[] args) {
    
    
    long startTime = System.currentTimeMillis();
	//...
    long endTime = System.currentTimeMillis();
    System.out.println("花费时间" + (en	dTime - startTime) + "ms");
    System.out.println("花费时间" + (endTime - startTime) / 1000 + "s");
}

①acwing官网问题

  • 若是出现Segmentation fault,那么可以使用exit(0)来进行调试确定在哪一行出了错:
void func(){
    exit(0);
}

//调用
func();
  • 若是在func()放在当前行执行没有出现Segmentation fault,说明再此之前没有可能出现Segmentation fault,那么就可以将func放置到后面。

②关于输入输出

//输入、输出函数
static BufferedReader cin = new BufferedReader(new InputStreamReader(System.in));
static PrintWriter out = new PrintWriter(new BufferedOutputStream(System.out));

③自定义类及属性 > 定义一维数组效率。具体可见leetcode1235题


注意事项

1、关于long res = 0; res = int * int出现精度问题

原因:int * int 得到的值会转为int类型,哪怕乘积是一个long类型,所以我们这边应该先将乘数先转为一个long类型才可。

案例1:1237. 螺旋折线,在java中对于这种k参与运算并且相乘结果超int范围的,必须将其本身也设置为long类型。

image-20221027142427447

案例2:1814. 统计一个数组中好对子的数目

image-20230117142107555

2、关于1e2问题

1e1就是10,若是有10个0,那么就是1e9情况。


技巧类

自定义Pair

自定义的键值对集合Pair:在acwing中需要自定义

static class Pair<K, V> {
    
    
    K x;
    V y;
    public Pair(K x, V y) {
    
    
        this.x = x;
        this.y = y;
    }
}

排序

①方式一:自定义类的话需要继承comparable接口,也就是实现compareTo方法

static class Node implements Comparable<Node> {
    
    
    public int ts;
    public int id;
    public Node(Integer ts, Integer id) {
    
    
        this.ts = ts;
        this.id = id;
    }

    public boolean equals(Node node) {
    
    
        return this.ts == node.ts && this.id == node.id;
    }

    @Override
    public int compareTo(Node o) {
    
    
        if (this.ts == o.ts) return this.id - o.id;
        return this.ts - o.ts;
    }
}

②方式二:

//该接口是实现compare接口
Arrays.sort(list, (o1, o2)->{
    
    })

N维数组转一维

二维数组转一维:

int A, B;//A表示行数、B表示列数

public int get(int i, int j) {
    
    
    return (i * B) + j;
}

三维数组转一维:

int A, B, C;

public int get(int i, int j, int k) {
    
    
    return (i * B + j) * C + k;
}

位运算

与运算:

场景一:判断是偶数还是奇数
ch & 1 == 1 => 奇数     ch & 1 == 0  偶数

异或:

场景一:大写、小写字母转换,无需写if判断大小写来进行+32-32,可直接进行ch ^= 1 << 5;  也就是异或32
	示例:https://leetcode.cn/problems/letter-case-permutation/
ch[i] ^= 1 << 5;

//异或情况
a ^ b = c
a ^ c = b

状态压缩

5位状态:state = (1 << 5) - 1,此时即为11111

查看当前状态没有选择的:i = (1 << 5) - 1 - state

  • 例如state=01100,最终i为10011。

当前状态补上新加的:state |= newState

  • 例如state=00010,newState = 01100,最终结果为01110。

消除掉原先的一些状态:state = state & ~pack或者state ^ (state & pack),两个等价

  • 例如state = 11111,消除目标pack = 01100,最终结果为10011。

算法基础

枚举 √

枚举是否选择

指数型枚举

模板题链接:92. 递归实现指数型枚举

题目:从 1∼n这 n 个整数中随机选取任意多个,输出所有可能的选择方案。

复杂度分析:时间复杂度O(2n);空间复杂度O(n)

import java.util.*;

class Main{
    
    
    
    private static int n;
    private static int[] arr;//0表示初始,1表示选,2表示不选
    
    public static void dfs(int u) {
    
    
        if (u == n) {
    
    
            //从选好的一组情况中来找到选的物品
            for (int i = 0; i < n; i++) {
    
    
                if (arr[i] == 1) {
    
    
                    System.out.printf("%d ",i + 1);
                }
            }
            System.out.println();
            return;
        }
        //递归多种状态
        //选
        arr[u] = 1;
        dfs(u + 1);
        arr[u] = 0;
        
        //不选
        arr[u] = 2;
        dfs(u + 1);
        arr[u] = 0;
    }
    
    public static void main(String[] args) {
    
    
        Scanner sc = new Scanner(System.in);
        n = sc.nextInt();
        arr = new int[n];
        dfs(0);
    }
}

排列型枚举

知识点

模板例题:94. 递归实现排列型枚举

示例:把 1∼n这n个整数排成一行后随机打乱顺序,输出所有可能的次序。

复杂度分析:时间复杂度为 O(n*n!);空间复杂度为O(n)

class Main {
    
    
    
    static int N = 9;
    static int n;
    //存储结果集
    private static int[] state = new int[N];
    //存储该路径是否访问过
    private static int[] vis = new int[N];
    
    //递归处理
    //dfs(0)开始
    public static void dfs (int u) {
    
    
        if (u == n) {
    
    
            //输出对应的方案
            for (int i = 0; i < n; i ++) {
    
    
                System.out.printf("%d", state[i]);
            }
            System.out.println();
            return;
        }
        //遍历枚举多种情况
        for (int i = 0; i < n; i ++) {
    
    
            if (!visited[i]) {
    
    
                visited[i] = true;
                state[u] = i + 1;//真实的值
                //递归
                dfs(u + 1);
                
                //恢复
                visited[i] = false;
                state[u] = 0;
            }
        }
    }
    
 
    
}

题单

蓝桥杯4届真题-带分数(全排列枚举、递归)


组合型枚举

模板题目:93. 递归实现组合型枚举

介绍:在排列型中进行升级,原本给定3个数让你找到所有3个排列方案,而在这里有n个数,让你找对应m个( <= n)个数组合的排列情况。

复杂度分析:时间复杂度O(mn!)

class Main {
    
    
    
    static final int N = 15;
    static int n, m;
    //每个结果集都
    static int[] state = new int[N];
    
    public static void dfs(int u, int start) {
    
    
        //优化剪枝,提前结束
        if (u + (n - start) < m) return;
        //若是遍历的到终点个数个
		if (u == m) {
    
    
            //输出对应的方案
            for (int i = 0; i < n; i ++) {
    
    
                System.out.printf("%d ", state[i]);
            }
            System.out.println();
        }
        for (int i = start; i < n; i ++) {
    
    
            statue[u] = i + 1;
            dfs(u + 1, i + 1);
            statue[u] = 0;
        }
    }
}

左右区间枚举

//枚举左端点  时间复杂度O(n^2)
int n = 10;
for (int len = 1; len <= n; len ++) {
    
    
    for (int l = 1; l <= n - len + 1; l ++) {
    
    
        int r = l + len - 1;
        System.out.printf("(%d,%d) ", l, r);
    }
    System.out.println();
}

image-20230402151434461


模拟 √

日期天数问题:平年闰年情况

//平年28天,闰年为29天

//判断闰年:不能被100整除,可以被4整除 或者  整除400
if (year % 100 != 0 && year % 4 == 0 || year % 400 == 0) 

static int[] months = {
    
    0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
//判断8位数是否是合法日期
public static boolean check(int d) {
    
    
    int year = d / 10000;
    int month = d % 10000 / 100;
    int day = d % 100;
    if (month > 12 || month == 0 || day > 31 || day == 0) return false;
    //闰年判断
    if (month != 2 && day > months[month]) return false;
    if (month == 2) {
    
    
        int leap = year % 4 == 0 && year % 100 != 0 || year % 400 == 0 ? 1 : 0;
        if (day > 28 + leap) return false;
    }
    return true;
}

//判断日期的合法功能函数
public static boolean check(int year, int month, int day) {
    
    
    if (month > 12 || month == 0 || day > 31 || day == 0) return false;
    //闰年判断
    if (month != 2 && day > months[month]) return false;
    if (month == 2) {
    
    
        int leap = year % 4 == 0 && year % 100 != 0 || year % 400 == 0 ? 1 : 0;
        if (day > 28 + leap) return false;
    }
    return true;
}

日期时间点问题:

求 HH:mm:ss

//获取到h小时m分钟s秒的总共秒数
public static int get_seconds(int h, int m, int s) {
    
    
    return h * 3600 + m * 60 + s;
} 

//将对应的总秒数去换算得到小时,分钟,秒数
int hour = time / 3600, minute = time % 3600 / 60, second = time % 60;

递归&分治 √

递归包含内容:递归枚举、递推枚举、dfs、bfs


递推

根据之前的状态来进行不断推举,例如:斐波那契数列。


贪心 √

区间问题

区间选点(最少点覆盖最多区间)、最大不相交区间数量(选择一种方案,该方案中不相交区间数量最大):

思路:根据右端点来进行排序,尽可能覆盖的范围越大,从前往后枚举不断地选择ed >= range[i].l中的最大右边端点,最终统计ed < range[i].l的数量。
    
//核心代码
//对所有的区间根据右端点进行排序
Arrays.sort(rans, 0, n);

//枚举所有区间
int ed = Integer.MIN_VALUE;
int res = 0;
for (int i = 0; i < n; i ++) {
    
    
    if (rans[i].l > ed) {
    
    
        res ++;
        ed = rans[i].r;	
    }
}

区间分组:选出的组数要尽可能少,每组中的区间互不相交

思路:使用一个小根堆来进行维护一个分组(一个组中只存储其分区的最晚结束时间)
1、根据左端点来进行排序。
2、从左到右进行枚举,使用一个小根堆来进行维护分组。
	若是当前的左边界<=小根堆堆顶,那么此时就说明需要进行添加一个分组,将当前区间右端点添加到小根堆中。
	若是当前左边界>小根堆堆顶,此时我们移除小根堆堆顶,将当前区间右端点入小根堆,实现一个替换操作。

//核心代码
//根据左端点来进行排序
Arrays.sort(rans, 0, n, (o1, o2) -> o1.l - o2.l);
//枚举所有的区间
for (int i = 0; i < n; i ++) {
    
    
    if (minQueue.isEmpty() || minQueue.peek() >= rans[i].l) {
    
    
        minQueue.offer(rans[i].r);//新增分组
    }else {
    
    
        //更新某个分组中的r
        minQueue.poll();
        minQueue.offer(rans[i].r);
    }
}

区间覆盖:给定一个目标区间和一组区间,让你在一组区间中找到最少得区间数量能够覆盖掉目标区间。

思路:根据左端点来进行排序,从左到右枚举每个区间,在所有能够覆盖掉start区间中选择右端点最大的区间,并将start更新成右端点的最大值(这个选择过程使用的是双指针)。

//核心代码
//根据左端点来进行排序
Arrays.sort(rans, 0, n, (o1, o2)->o1.l - o2.l);
int res = 0;//存储结果值
boolean success = false;
//枚举所有区间
for (int i = 0; i < n; i ++) {
    
    
    int j = i, r = Integer.MIN_VALUE;//定义一个右指针j进行移动,ed表示当前的最大右端点
    while (j < n && rans[j].l <= st) {
    
    
        r = Math.max (rans[j].r, r);
        j ++;
    }
    //若是遍历走完r依旧是<st,那么此时说明没有方案
    if (r < st) break;

    res++;//找到一种方案

    if (r >= ed) {
    
     //根据当前最新的依据右端点来进行判定
        success = true;
        break;
    }
    st = r;//更新最新的一个左端点
    i = j - 1;
}
if (!success) res = -1;
System.out.println(res);

均值不等式

货仓选址:给你多个点,让你去确定在那个地点建仓库可以让来回距离最短。

  • 思路:将所有水平点存储到数组中对其进行排序,接着mid = (n + 1) / 2即可确定中间点,最后就是来进行求取距离。若是有对应的公式推导成最终这个样子:|A1 - B| + |A2 - B| + |A3 - B| … + |An - B|,那么最后就是ans += Math.abs(A[i] - A[mid])
  • 模板题:AcWing 104. 货仓选址
//核心代码
//排序
Arrays.sort(a, 0, n);
int mid = (n - 1) / 2;//找到中值点,即为最优建立点
long res = 0;//枚举所有的点
for (int i = 0; i < n; i ++) {
    
    
    res += Math.abs(a[mid] - a[i]);
}

排序 √

归并排序

前缀和&差分 √

前缀和

一维前缀和:

s[i] = s[i - 1] + nums[i]
# 推导,其中nums的值需要从坐标1开始,若是默认给的从0开始则需要为s[i] = s[i - 1] + nums[i - 1]
s[1] = s[0] + nums[1]
s[2] = s[1] + nums[2] = nums[1] + nums[2]
s[3] = s[2] + nums[3] = nums[1] + nums[2] + nums[3]
# 计算范围:[1,3]   s[3] - s[1 - 1]   =>     [i, j]   s[i] - s[j - 1]

二维前缀和:

s[i][j] = s[i - 1][j] + s[i][j - 1] - s[i - 1][j - 1] + a[i][j]

# 计算范围 x1,y1   x2y2
sum = s[x2][y2] - s[x2][y1 - 1] - s[x1 - 1][y2] + s[x1 - 1][y1 - 1]

差分(一维、二维、三维)

蓝桥杯三维差分题:AcWing 1232. 三体攻击

一维:

步骤一(反推b):b[i] = a[i] - a[i - 1]

中间步骤(范围操作):
b[l] += c;
b[r + 1] -= c;

步骤三:a[i] = a[i - 1] + b[i]

二维:

步骤一(反推b):b[i][j] = a[i][j] + a[i - 1][j - 1] - a[i - 1][j] - a[i][j - 1]

中间步骤(范围操作):
b[x1][y1] += c
b[x2 + 1][y1] -= c
b[x1][y2 + 1] -= c
b[x2 + 1][y2 + 1] += c

步骤三(反推a):a[i][j] = a[i - 1][j] + a[i][j - 1] - a[i - 1][j - 1] + b[i][j]

三维:

//步骤一:首先确定三维前缀和公式
b(x, y, z) = a(x, y, z) - a(x - 1, y, z) - a(x, y - 1, z) + a(x - 1, y - 1, z)
           - a(x, y, z - 1) + a(x - 1, y, z - 1) + a(x, y - 1, z - 1) - a(x - 1, y - 1, z - 1)

中间步骤(范围操作):
二维正面(以z1为)
b[x1    ][y1    ][z1] += val
b[x1    ][y2 + 1][z1] -= val
b[x2 + 1][y1    ][z1] -= val
b[x2 + 1][y2 + 1][z1] += val
转为z2+1,且符号改变
b[x1    ][y1    ][z2 + 1] -= val
b[x1    ][y2 + 1][z2 + 1] += val
b[x2 + 1][y1    ][z2 + 1] += val
b[x2 + 1][y2 + 1][z2 + 1] -= val

步骤三:最后反推求出a数组推导来进行计算
a(x, y, z) = b(x, y, z) + a(x - 1, y, z) + a(x, y - 1, z) - a(x - 1, y - 1, z)
           + a(x, y, z - 1) - a(x - 1, y, z - 1) - a(x, y - 1, z - 1) + a(x - 1, y - 1, z - 1) 

扩展:

1、前缀异或

相关题目:

二分 √

规律:一组单调递增或者递减情况(不限制于数字)情况时可以采用二分来进行优化。【O(n) -> O(logn)】

模板:

//第一类二分写法:check
int l = 0, r = n;
while (l < r) {
    
    
    int mid = l + r >> 1;
    if (mid >= target) l = mid + 1;
    else r = mid;
}

int l = 0, r = n;
while (l < r) {
    
    
    int mid = (l + r + 1) >> 1;
    if (nums2[i] >= nums1[mid]) l = mid - 1;
    else r = mid;
}

//第二种二分写法:
while (l != r) {
    
    
    int mid = l + ((r - l) >> 1);
    if (nums1[mid] < nums2[i]) l = mid + 1;
    else r = mid;
}

题单

蓝桥杯13届真题-求阶乘(算数基本定理、二分)

搜索

DFS √

题型:最大长度

模板:

void dfs(int step)    //步长
{
    
    
    if(/*跳出循环的条件*/){
    
    
        return;    //return十分关键,否则循环将会无法跳出
    }
    /*函数主体
    对功能进行实现*/
    for(/*对现有条件进行罗列*/){
    
    
        if(/*判断是否合理*/){
    
    
            //将条件修改
            dfs(/*新的step*/)
            /*!重中之重,当跳出那层循环后将数据全部归位*/
        }    
    }
}

矩阵模板:

int f[4][2]={
    
    {
    
    0,1},{
    
    0,-1},{
    
    1,0},{
    
    -1,0}};    //用于判断下一步怎么走向几个方向走就是几个数据
void dfs(int x,int y){
    
            //进入点的坐标
    if(/*跳出循环的条件*/){
    
    
        /*作出相应操作*/
        return;        //不要忘了return
    }
    for(int i=0;i</*f的长度*/;i++){
    
    
        int x0=x+f[i][0];    
        /*此处是更新点的坐标,注意是直接让原来的点加上这个数据,不是直接等于*/
        int y0=y+f[i][1];
        if(/*用新坐标x0,y0判断是否符合条件*/){
    
    
            dfs(x0,y0);    //用新的坐标进行递归
        }
    }
}

BFS √

题单

3、蓝桥杯13届真题-回忆迷宫(模拟、BFS)

模板

题型:最短路径。

模板:

1、二叉树:

class Solution {
    
    
    public static void main (String[] args) {
    
    
        Queue<TreeNode> queue = new LinkedList<>();
        queue.offer(xx);//添加root节点
        while (!queue.isEmpty()) {
    
    
            int size = queue.size();
            for (int i = 0; i < size; i++) {
    
    
                //取出node结点进行操作
                TreeNode node = queue.poll();
                //放置左右子节点
                if (node.left != null) queue.offer(node.left);
                if (node.right != null) queue.offer(node.right);
            }
            res.add(1.0 * sum / size);
        }
    }
}

2、二维矩阵:

class Point {
    
    
	private int x;
	private int y;
	public Point(int x, int y) {
    
    
		this.x = x;
		this.y = y;
	}
}

class Solution {
    
    
    public static void main (String[] args) {
    
    
        Queue<Point> queue = new LinkedList<>();
        queue.offer(new Point(1,2));//出发点
        while (!queue.isEmpty()) {
    
    
            int size = queue.size();
            for (int i = 0; i < size; i++) {
    
    
                //取出Point结点进行操作
                Point point = queue.poll();
                //进行操作
                //放置四个方向的节点
				for (int d = 0; d < dicts.length; d++) {
    
    
					int x = point.x + dicts[d][0];
					int y = point.y + dicts[d][1];
					queue.offer(new Point(x, y))
				}
            }
        }
    }
}

3、坐标点写法:

class Solution {
    
    
    
    static final int N = 110;
    static int[] q = new int[N * N];
    static int hh, tt;//hh表示头指针(用于出队),tt表示入队指针
    //四个方向  (0, 1) (0, -1) (1, 0) (-1, 0)
	static int[] dx = {
    
    0, 0, 1, -1};
    static int[] dy = {
    
    1, -1, 0, 0};
    //矩阵
    static int[][] g = new int[N][N];
    static int H, W;
    
    //将二维坐标转为一个数字
    public static int get(int x, int y) {
    
    
        return x * (W + 1) + y;
    }
    
    public static void main (String[] args) {
    
    
        //bfs过程
        while (hh < tt) {
    
    
            int top = q[hh ++];
            int x = top / (W + 1);
            int y = top % (W + 1);
            //四个方向
            for (int k = 0; k < 4; k ++) {
    
    
                int xx = x + dx[k];
                int yy = y + dy[k];
                //搜索校验(边界情况 && 其他情况xxx)
                if (xx >= 1 && yy >= 1 && xx <= H && yy <= W) {
    
    
                    //入队
                    q[tt ++] = get(x, y);
                    //相关动作xxx
                }
            }
        }
    }
}

IDA*

考虑事项:

  1. 迭代加深:进行逐层判断,是否能够完全覆盖。
  2. 选择最少的列:尽可能选择情况少的来进行搜索。
  3. 可行性剪枝:通过使用一个估价函数h(state)表示对于状态state至少需要多少行。若是符合当前的搜索的行数则继续向下,若是不符合提前剪枝结束。

题单

蓝桥杯真题 糖果(状压+IDA*、dp状态压缩)

回溯

字符串

KMP算法

模板:

package com.changlu.string;

public class KMP {
    
    

    public static void main(String[] args) {
    
    
        //API
        System.out.println("ababcabcaabbcdeabcdef".indexOf("abcaabb"));
        //手写
        System.out.println(kmp("ababcabcaabbcdeabcdef", "abcaabb"));
    }
	
    //kmp匹配
    public static int kmp (String str, String sub) {
    
    
        int[] next = getNext(sub);
        for (int i = 0, j = 0; i < str.length(); i++) {
    
    
            while (j > 0 && str.charAt(i) != sub.charAt(j)) {
    
    
                j = next[j - 1];
            }
            if (str.charAt(i) == sub.charAt(j)) j++;
            //若是匹配到最后
            if (j == sub.length()) {
    
    
                return i - j + 1;
            }
        }
        return -1;
    }

    //构建next数组
    public static int[] getNext(String str) {
    
    
        int[] next = new int[str.length()];//根据字符串长度来创建数组
        next[0] = 0;//初始化第一个位置为0
        for (int i = 1, j = 0; i < str.length(); i ++) {
    
    
            //j下标>0,j指针与i不相同
            while (j > 0 && str.charAt(i) != str.charAt(j)) {
    
    
                j = next[j - 1];
            }
            //若是当前字符相等
            if (str.charAt(i) == str.charAt(j)) {
    
    
                j++;
            }
            next[i] = j;
        }
        return next;
    }

}

动态规划 √

线性DP

来源:AcWing 902. 最短编辑距离【闫式DP大法好:)】

image-20230401201125500

数字三角形问题:找最大路径问题

//时间复杂度O(n^2) 空间O(n)
//上至下 边界填充负无穷值,滚动数组从后往前,最后一层需要进行计算最小值
f(i, j) += max(f(i - 1, j - 1), f(i - 1, j))
    
//从下之上 
f(i, j) += max(f(i + 1, j), f(i + 1, j + 1)) + a[i][j]

子序列问题:这个子序列并非是连续的。

最长上升子序列:找严格递增(减)子序列长度

//暴力法 时间O(n^2) 空间O(n)
for (int i = 1; i <= n; i ++) {
    
    
    fn[i] = 1;
    for (int j = 1; j < i; j ++) {
    
    
        //递增情况
        if (a[i] > a[j]) fn[i] = Math.max(fn[i], fn[j] + 1);
    }
}

//贪心优化 时间O(n.logn)  空间O(n)
//核心思想:若是添加数>末尾数,直接添加到末尾;若是<=,则找到队列中>=该添加数的第一个数(从前往后)
//注意点:这种做法最终在q队列数组中的剩余元素并非是最长递增的正确元素,在这个过程中我们对于替换队列中的元素目的为(每次替换元素都是,增加了序列长度增长的潜力!!!)
q = new int[N] //优先队列
q[++ cnt] = a[1]; //第一个数先入队
for (int i = 2; i <= n; i ++) {
    
    
    if (a[i] > q[cnt]) {
    
     //若是递增直接添加到后面
        q[++ cnt] = a[i];
    }else {
    
    
        //替换从前往后第一个>=a[i]的位置  find()是二分操作
        q[find(a[i])] = a[i];
    }
}

public static int find (int num) {
    
    
    int l = 1, r = cnt;
    while (l < r) {
    
    
        int mid = l + r >> 1;
        //若是目标元素值>=num,此时范围为[l, mid]
        if (q[mid] >= num) r = mid;
        else l = mid + 1;//若是<num,则需要直接到右半部分去寻找,范围为[mid + 1, r]
    }
    return r;
}

最长公共子序列:给定A、B两个字符串,求两者之间的最长公共子序列

//时间复杂度O(n*m)  空间O(n*m)
//不重不漏针对于第i个第j个分为如下四种情况
f(i - 1, j - 1):两个都不选,00
f(i - 1, j):第i个不选,第j个选,01
f(i, j - 1):第i个选,第j个不选,10
f(i - 1,j - 1) + 1:两个都选,11
    
//转移方程
a[i] != a[j]:max (a(i - 1, j)a(i, j - 1)) //【实际上由于a(i - 1, j)和a(i, j - 1)是包含a(i - 1, j - 1)所以我们可以省略掉a(i - 1, j - 1);原本是max (a(i - 1, j),a(i, j - 1), a(i - 1, j - 1))】
a[i] == a[j]:`f(i - 1, j - 1) + 1`

最短编辑距离问题:一个字符串替换成另一个字符串的最短替换操作次数,给定插入、删除、编辑三种情况

//时间复杂度O(n*m)  空间O(n*m)
//初始化 0->1,2,3,4,5  (i->j) 插入操作  |    1,2,3,4,5 -> 0 (i->j) 删除操作      
fn[0][j] = j;                              fn[i][0] = i;

//删除与插入情况:
min(f(i - 1, j) + 1, f(i, j - 1) + 1)
//a[i] == b[j]
min(f(i, j), f(i - 1, j - 1))
//a[i] != b[j](需要进行替换操作一次)
min(f(i, j), f(i - 1, j - 1) + 1)

背包DP

01背包问题:每件物品选一次或者不选,每件物品最多选1次

  • fn(i, j)表示的选择i个,重量为j的最大价值
//二维数组
//不选:
fn[i][j] = fn[i - 1][j];
//选:	
fn[i][j] = max (fn[i][j], fn[i - 1][j - v[i]] + w[i])
    
//一维滚动数组(需要从后往前进行遍历)
fn[j] = max (f[j], f[j - v[i] + w[i])

完全背包问题:每件物品选一次或者不选,每件物品有无限个

//二维数组,优化之后(三重循环转二重循环)
//不选:
fn[i][j] = fn[i - 1][j];
//选:注意此时后面的fn[i]不再是i - 1
fn[i][j] = max (fn[i][j], fn[i][j - v[i]] + w[i])
    
//一维滚动数组(可直接从前往后遍历)
fn[j] = max(f[j], f[j - v[i]] + w[i])

多重背包(含优化):每件物品最多有有限个,题目给出限制

//三重暴力
 for (int i = 1; i <= n; i ++)   //前i个物品
     for (int j = 0; j <= m; j ++)   //体积为k
         //指定物品的数量(注意:k * v[i] <= j范围下进行选择)
         for (int k = 0; k <= s[i] && k * v[i] <= j; k ++) 
             fn[i][j] = Math.max (fn[i][j], fn[i - 1][j - k * v[i]] + k * w[i]);

//时间复杂度O(n*logs*m)
//二进制优化:利用二进制去将1个物品s个数量优化为logs个组合箱子数,接着就是一个01背包思路解决方案
//读取每一个物品体积、价值以及数量,根据数量(二进制优化,拆分指定组箱子,每组有2进制阶乘个数)
int cnt = 0;
for (int i = 1; i <= n; i ++) {
    
    
    //得到vw、ww、s(体积、价值以及数量)
	int k = 1;
    //根据s来分配每组物品的数量
    while (k <= s) {
    
    
        cnt ++;
        v[cnt] = k * vv;
        w[cnt] = k * ww;
        s -= k;//减去总数
        k *= 2;//2的倍乘
    }
    //处理多余的s
    if (s > 0) {
    
    
        cnt ++;
        v[cnt] = s * vv;
        w[cnt] = s * ww;
    }
}
//最后就是一个01背包模板解决(一维滚动数组)
fn[j] = max (f[j], f[j - v[i] + w[i])

分组背包:有多组,每一组里面可以选一个

//三重暴力,以i组为单位对每组中的物品进行选择  s[i]、v[i][k]、w[i][k] 分别记录每组数量、指定i组第k个体积、指定i组第k个价值
for (int i = 1; i <= n; i ++) {
    
      //遍历前i组
    for (int j = m; j >= 0; j --) {
    
      //体积重量 滚动数组,从后往前(01背包)
        for (int k = 0; k < s[i]; k ++) {
    
     //组内第k个
            //必须放在for循环内部,否则会直接提前结束
            if (j >= v[i][k]) {
    
    
                fn[j] = Math.max(fn[j], fn[j - v[i][k]] + w[i][k]);
            }

区间DP

石子合并:相邻的左边和右边进行合并,求最终合并成一堆的最小代价

//思路:枚举所有的区间,区间大小依次从[1, n]不断扩大,在这个区间枚举过程中我们去搜索相对的所有最优解
dp(i, j) = Math.min(dp(i, j), dp(i, k) + dp(k + 1, j) + s[j] - s[i - 1]),k ∈ [i, j]

//时间复杂度O(n^3);空间复杂度O(n^2)
//枚举左右区间
for (int len = 2; len <= n; len ++) {
    
    
    for (int l = 1; l <= n - len + 1; l ++) {
    
      //左端点
        int r = l + len - 1; //右端点
        fn[l][r] = Integer.MAX_VALUE;
        for (int k = l; k <= r; k ++) {
    
      //取[1, k]+[k+1,r]的最优解
            fn[l][r] = Math.min(fn[l][r], fn[l][k] + fn[k + 1][r] + s[r] - s[l - 1]);
        }
    }
}

树形DP

物体与物体之间有上下关联能够组成一棵树,简而言之就是基于一颗树的动态规划。

没有上司的舞会:人与人有上下级关系,没有职员愿意与上司一起,每个人有最大快乐值,求最大的选择方案快乐值

状态定义:f(u, s),u结点选择s状态的最大快乐指数。(u表示第u个节点,s包含两个状态,1为选,0为不选)
状态转移:
  f(u, 0)f(u, 0) += max(f(child, 0), f(child, 1)),child表示u的子节点
  f(u,1)f(u, 1) += f(child, 0)
    
//实现思路:使用邻接表数组代码实现一棵树,然后进行遍历树,在遍历过程中对于父子节点来进行动态规划
public static void dfs (int u) {
    
    
    //设置当前u状态选择的值
    fn[u][1] = happy[u];
    //遍历树
    for (int i = h[u]; i != -1; i = ne[i]) {
    
    
        int j = e[i];//获取到节点值
        dfs (j);
        //j为u的子节点,来去枚举当前u的状态
        //若是当前节点不选,子节点可以选或者不选
        fn[u][0] += Math.max (fn[j][0], fn[j][1]);
        //若是当前节点选,那么子节点绝对不选
        fn[u][1] += fn[j][0];
    }
}

状态压缩DP

计数DP

数学 √

数论

算法基本定理

知识点

算数基本定理

题单

蓝桥杯13届真题-求阶乘(算数基本定理、二分)

约数

约数个数及约数之和知识点(含公式)

找到一个数所有的因子:

import java.util.ArrayList;
import java.util.List;

public class Main {
    
    

	public static void main(String[] args) {
    
    
		List<Integer> list = new ArrayList<>();
		int n = 8;
		// 枚举sqrt(n)个,找到所有因子
		for (int i = 1; i <= n / i; i++) {
    
    
			if (n % i == 0) {
    
    
				list.add(i);
				// 由于枚举的i范围是sqrt(n),所以这里需要提前找到相对应的公因子(>sqrt(n)部分)
				if (i != n / i) {
    
    
					list.add(n / i);
				}
			}
		}
		System.out.println(list);
	}

}

image-20230404162740064

最大公约数与公倍数

//最大公约数 greatest common divisor
int gcd(int a, int b) {
    
    
    return b == 0 ? a : gcd(b, a % b);
}
//最小公倍数  Lowest Common Multiple
int lcm(int a, int b) {
    
    
    return a * b / gcd(a, b);
}

欧拉筛法(含朴素筛法、埃式筛法)

判断一组数是否为质数?暴力法O(n2)、埃式法O(n*log(logn))、欧拉筛O(n)

数论之欧拉筛法(含朴素筛选、埃式筛选详细代码)

质数-欧拉筛

模板:

//欧拉筛所需要数组
//flag表示合数数组,true为合数
static boolean[] flag = new boolean[N];
//存储质数
static int[] primes = new int[N];
static int cnt = 0;

//欧拉筛
public static void getPrimes(int n) {
    
    
    //遍历所有情况
    for (int i = 2; i <= n; i++) {
    
    
        if (!flag[i]) primes[cnt++] = i;
        //枚举所有primes数组中的情况来提前构造合数
        for (int j = 0; j < cnt && primes[j] * i <= n; j ++) {
    
    
            int pre = primes[j] * i;
            flag[pre] = true;
            if (i % primes[j] == 0) break;
        }
    }
}


//判断是否是质数(由于之前primes数组仅仅开了sqrt(20亿)也就只有50万,所以这里需要进行遍历一遍质数数组来进行判断校验)
public static boolean isPrime(int x) {
    
    
    //若是x在50万范围,直接从flag数组中判断返回即可
    if (x < N) return !flag[x];
    //若是>=50万,那么就进行遍历质数数组看是否有能够整除的,如果有那么直接返回
    for (int i = 0; primes[i] <= x / primes[i]; i++) {
    
    
        if (x % primes[i] == 0) return false;
    }
    return true;
}

欧几里得与扩展欧几里得

欧几里得与扩展欧几里得算法(含推导过程及代码)

辗转相除(含辗转相减法)

辗转相除以及辗转相减法

组合数学

容斥定理

容斥定理:能被 ab 整除的数的个数 = 能够被 a 整除的数的个数 + 能够被 b 整除的数的个数 - 既能被 a 又能被 b 整除的数的个数

题单

leetcode、878. 第 N 个神奇数字(困难)

快速幂

快速幂及矩阵快速幂分析及代码实现

模板:

private static final long MOD = 1000000007;

/**
     * 递归快速幂
     * @param a 实数a
     * @param n 阶数n,三种情况,n=0,n=奇数,n=偶数
     * @return
     */
public static long qpow(long a, long n){
    
    
    if (n == 0){
    
    
        return 1;
    }else if ((n & 1) == 1) {
    
      //奇数
        return qpow(a, n -1) * a % MOD;
    }else {
    
    
        long temp = qpow(a, n / 2) % MOD;
        return temp * temp % MOD;
    }
}

/**
     * 非递归方式
     */
public static long qpow2(long a, long n) {
    
    
    long ans = 1;
    while ( n != 0) {
    
    
        if ((n & 1) == 1) {
    
     //若是n为奇数
            ans *= a % MOD;
            ans %= MOD;//求模处理
        }
        a *= a % MOD; //这个就表示偶数情况
        a = a % MOD;//求模处理
        n = n >> 1;
    }
    return ans;
}

矩阵

数据结构

树、图

下面是树、图邻接表实现:使用dfs深搜,其中针对于图的话是需要st表(add需要将一对节点两次),若是树的话无需st表

import java.util.Arrays;

public class Main {
    
    

    static int N = 8;
    //e与ne分别表示的是链表节点的值与next指针,h表示的是多个单链表的表头
    static int[] e = new int[N], ne = new int[N], h = new int[N];
    //判断是否访问过对应的单链表
    static boolean[] st = new boolean[N];
    static int idx = 0;
	
    //将b添加到a
    public static void add(int a, int b) {
    
    
        e[idx] = b;
        ne[idx] = h[a];
        h[a] = idx++;
    }

    public static void dfs(int u) {
    
    
        st[u] = true; // st[u] 表示点u已经被遍历过
        System.out.println(u);
        for (int i = h[u]; i != -1; i = ne[i]) {
    
    
            int j = e[i];
            if (!st[j]) dfs(j);
        }
    }

    public static void main(String[] args) {
    
    
        Arrays.fill(h, -1);//初始化头结点都指向空
        add(1, 2);
        add(2, 1);
        add(1, 3);
        add(3, 1);
        dfs(1);
    }
}

哈希表 √

树状数组 √

并查集

线段树 √

#图论 √

树上问题

树的直径

计算机几何

杂项

双指针 √

猜你喜欢

转载自blog.csdn.net/cl939974883/article/details/129502039
今日推荐