抽象深度优先搜索 k个数的和

先来看一个非常简单的问题:

给定n个整数, 要求选出 K 个数, 使得选出来的 K 个数的和为 sum

借助 dfs 来解决问题, 对每一个数, 枚举选或者不选两种情况, 我们可以用 dfs 思想来完成这样的枚举过程。

搜素过程中, 用 s 来记录当前选择的数值总和, k 用来记录选择的数的个数, deep 表示当前正在枚举第几个数是否选择。

在第一层dfs的时候, 我们可以枚举是否选第一个数, 如果选第一个数则让 s 加上第一个数且 k 加一, dfs 进入下一层; 否则 dfs 直接进入到下一层。

在第二层, 对第二个数做同样的处理, dfs 的过程中记录已经选取的数的个数, 如果已经选取了 k 个数, 判断 s 值是否等于sum。对于每一层, 我们都有两个选择-------选和不选。 不同的选择, 都会使得搜素进入完全不同得分支继续搜素。

下图是这个搜素过程对应的搜索树

如果不选取第 i 个数, 那么 cnt s 都不会有变化。

如果选取第 i 个数, cnt 加上 1 , s 加上 a[ i ]

对于不同的选择进行不同的分支搜索。

dfs(i+1, cnt, s);
dfs(i+1, cnt+1, s+a[i]);

现在处理边界条件, 边界条件其实很简单, 当 i == n 的时候, 我们已经对所有的数都做出了选择, 这时候我们来判断选出来的数的个数是否等于 k , 和值是否等于 sum

    if(i==n){
        if(cnt==k && s==sum){
            ans++;
        }
        return;
    }

调用 dfs 函数, 初始化的时候参数 i,cnt, s都应该初始化成为0。

dfs(0,0,0);

以下为完整代码:

#include <iostream>
using namespace std;
int n, k, sum, ans;
int a[40];
void dfs(int i, int cnt, int s)
{
    if(i==n){
        if(cnt==k && s==sum){
            ans++;
        }
        return;
    }
    dfs(i+1, cnt, s);
    dfs(i+1, cnt+1, s+a[i]);

}
int main() {
       
    cin >> n >> k >> sum;
    for (int i = 0; i < n; i++) {
        cin >> a[i];
    }
    ans = 0;
    dfs(0,0,0);
    cout<<ans<<endl;
    
    return 0;
}

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值