优美的数列(经典递归回溯)

526. 优美的排列(题目地址)

假设有从 1 到 N 的 N 个整数,如果从这 N 个数字中成功构造出一个数组,使得数组的第 i 位 (1 <= i <= N) 满足如下两个条件中的一个,我们就称这个数组为一个优美的排列。条件:

   1.  第 i 位的数字能被 i 整除
   2.   i 能被第 i 位上的数字整除

现在给定一个整数 N,请问可以构造多少个优美的排列?

输入: 2
输出: 2
解释: 

第 1 个优美的排列是 [1, 2]:
  第 1 个位置(i=1)上的数字是1,1能被 i(i=1)整除
  第 2 个位置(i=2)上的数字是2,2能被 i(i=2)整除

第 2 个优美的排列是 [2, 1]:
  第 1 个位置(i=1)上的数字是2,2能被 i(i=1)整除
  第 2 个位置(i=2)上的数字是1,i(i=2)能被 1 整除

数据说明:N是一个正整数,并且不会超过15。

这道题由于它的N比较小,一开始我想着去找规律,一个个将他全部计算出来,而实际上这确实也是个方法,并且也有许多人将它实现了。但我估计他们也是先通过回溯算法或者是状态dp先把答案算出来,然后再直接提交答案。

本题符合经典dfs回溯的特点,我们可以从左到右依次确定每个位置上的数,到最后一个数时在往回重新更改。具体来说就是dfs(index,n)确定index位置上的数,然后继续搜索dfs(index+1,n)。为了提高dfs的效率,我们设置一个vis数组将搜索过的数标记起来,这样可以减少dfs的次数,然后在回溯到这个标记的位置时,将它取消标记。为了得到每一个index上可放置的数有哪些,我们在dfs之前先设置一个math二维数组,将index位置对应的数放于math中的相应行中。

具体表达可看代码:

class Solution {
public:
    //经典dfs回溯算法
    int ans=0;
    vector<vector<int>> mask;
    vector<int> vis;
    void dfs(int index,int n){
        if(index==n+1){
            ans++;
            return;
        }
        for(auto ch:mask[index]){
            if(!vis[ch]){
                vis[ch]=1;
                dfs(index+1,n);
                vis[ch]=0;
            }
        }
    }
    int countArrangement(int n) {
        mask.resize(n+1);
        vis.resize(n+1);
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++){
                if(i%j==0||j%i==0)
                mask[i].emplace_back(j);
            }
        }
        dfs(1,n);
        return ans;
    }
};
递归回溯和动态规划都是常用的问题求解方法,它们在不同的场景下有不同的应用。 递归(Recursion)是一种通过将问题分解为更小的子问题来求解的方法。递归函数会重复调用自身,直到达到某个终止条件。递归通常用于解决问题的定义是自身的情况,例如计算斐波那契数列、求解二叉树的深度等。递归的实现简单直观,但在某些情况下可能会导致效率低下或者堆栈溢出的问题回溯(Backtracking)是一种通过尝试所有可能的情况来求解问题的方法。回溯算法通常用于求解排列组合、搜索问题,如八皇后问题、组合总和等。回溯算法通过遍历所有可能的选择并进行回退,以找到所有满足条件的解。回溯算法可以通过剪枝操作来减少不必要的尝试,提高效率。 动态规划(Dynamic Programming)是一种通过将问题划分为若干个重叠子问题,并保存子问题的解来避免重复计算的方法。动态规划通常用于求解最优化问题,如最长公共子序列、最短路径、背包问题等。动态规划算法通过状态转移方程和边界条件来确定子问题之间的关系,并按照一定的计算顺序求解子问题,最终得到原问题的解。动态规划算法的时间复杂度通常较低,可以大大提高算法的效率。 总结来说,递归是一种问题分解的方法,回溯是一种穷举搜索的方法,动态规划是一种优化求解的方法。它们在不同的问题场景下有着各自的应用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ZW钟文

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值