Acwing_蓝桥_递归

Posted Angel_Jayce

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Acwing_蓝桥_递归相关的知识,希望对你有一定的参考价值。

一.关于由数据范围反推算法复杂度及其算法

关于输入输出:问题规模小于105:cin,scanf都差不多,但是要是大于105推荐使用scanf和printf。

二.关于递归

1.定义

自己调用自己

2.注意事项:

  • 判断递归结束的边界
  • 少调用局部变量,会占用很大的内存
  • 要怎么调用自身

3.每个递归都可以转化成递归搜索树

例如计算斐波那契数列可以转化成如下(这里不讨论剪枝,也就是不把重复的剪掉)

三.递归练习

1.递归实现指数型枚举

https://www.acwing.com/problem/content/94/

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
using namespace std;



const int N = 16;

int n;
int st[N];//表示状态:0表示还不考虑,1表示选,2表示不选

void dfs(int u)

    if(u > n) // 终止条件
    
        for(int i = 1; i <= n; i++)
            if(st[i] == 1) printf("%d ", i);
        puts("");
        return;
    

    st[u] = 1; 
    dfs(u + 1);
    st[u] = 0;//回溯,要恢复原来的状态

    st[u] = 2; 
    dfs(u + 1);
    st[u] = 0;


int main()

    scanf("%d", &n);
    dfs(1);
    return 0;

2.递归实现排列型枚举

https://www.acwing.com/problem/content/96/

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;

const int N = 10;
int st[N];
bool used[N];
int n;

void bfs(int u)

    if (u > n)
    
        for (int i = 1; i <= n; i++) printf("%d ", st[i]);
        printf("\\n");
        return;
    
    for (int i = 1; i <= n; i++)
    
        if (!used[i]) //表示i没有被用过
        
            used[i] = true;
            st[u] = i;
            bfs(u+1);
            st[u] = 0;
            used[i] = false;
        
    


int main()

    scanf("%d", &n);
    bfs(1);
    return 0;

关于上面递归算法的时间复杂度分析:

第一层中的基本操作是for循环进行深搜,遍历为O(n),然后递归中有n个这样的函数,也就是n个分支。第二层也是一个for循环,然后循环中有n-1个分支,时间复杂度是O(n(n-1))。第三层就是O(n(n-1)(n-2)),以此类推,最后一层是的时间复杂度是O(nn!)。所以总的时间复杂度是O(n(1+n+n(n-1)+...+n!)),该循环是大于O(n!)的,经过放缩法可以证明是小于O(3n!)。所以最终时间复杂度为O(n*n!)

3.递归实现组合型枚举

https://www.acwing.com/problem/content/95/

#include<iostream>
#include<cstring>
#include<algorithm>
#include<cstdio>
using namespace std;

const int N = 30;
int n,m;
int st[N];
bool path[N];

void dfs(int u,int t)

    if(u == m)
    
        for(int i = 0 ; i < m ; i ++ ) printf("%d ", st[i]);
        printf("\\n");
        return;
    
    for(int i = t; i <= n ; i++)
    
        if(u==0&&i + m - 1 > n ) break;
        if(!path[i])
        
            st[u] = i;
            path[i] = true;
            dfs(u+1,i+1);
            if(u)path[i] = false;
        
    

int main()

    scanf("%d%d", &n, &m);
    dfs(0,1);
    return 0;


以上是关于Acwing_蓝桥_递归的主要内容,如果未能解决你的问题,请参考以下文章

AcWing 蓝桥杯C++ AB组辅导课 第一讲递归与递推

AcWing 蓝桥杯C++ AB组辅导课 第一讲递归与递推

ALGO-139_蓝桥杯_算法训练_s01串(递归)

AcWing 796. 子矩阵的和

蓝桥杯DFS正确入门方式 | DFS + 递归与递推习题课(上) | 一节课教你爆搜!——学习笔记

蓝桥杯算法竞赛系列第二章——深入理解重难点之递归(上)