89 k数和
原题网址:https://www.lintcode.com/problem/k-sum/description
描述
给定n个不同的正整数,整数k(k < = n)以及一个目标数字。
在这n个数里面找出K个数,使得这K个数的和等于目标数字,求问有多少种方案?
样例
给出[1,2,3,4],k=, target=,[1,4] and [2,3]是个符合要求的方案
dp[i][j][t]=dp[i-][j][t];
if (t>=A[i])//注意是大于等于;
{
dp[i][j][t]+=dp[i-][j-][t-A[i]];
}
意思就是,每个元素都有两种状态,取或者不取:
(1)若不取A[i]这个值,当前方案数等于从前i-1个数中取j个使其和为t的方案数,即dp[i - 1][j][t]。
(2)若取当前值A[i],则当前方案数等于从前i-1个数中取j个使其和为t的方案数再加上考虑A[i]的情况,即dp[i - 1][j - 1][t - A[i]](前提是t - A[i]要大于等于0)。
class Solution {
public:
/**
* @param A: An integer array
* @param k: A positive integer (k <= length(A))
* @param target: An integer
* @return: An integer
*/
int kSum(vector<int> &A, int k, int target) {
// write your code here
int size=A.size();
if (target<)
{
return ;
}
vector<vector<vector<int>>> dp(size+,vector<vector<int>>(k+,vector<int>(target+,))); for (int i=;i<=size;i++)
{
for (int j=;j<=k;j++)
{
for (int t=;t<=target;t++)
{
if (j==&&t==)//前i个数中取0个和为0只有一种方案,就是不取任何数;
{
dp[i][j][t]=;
}
else if (!(i==||j==||t==))
{
dp[i][j][t]=dp[i-][j][t];
if (t>=A[i-])//注意是大于等于;
{
dp[i][j][t]+=dp[i-][j-][t-A[i-]];
}
}
}
}
}
return dp[size][k][target]; }
};
class Solution {
public:
/**
* @param A: An integer array
* @param k: A positive integer (k <= length(A))
* @param target: An integer
* @return: An integer
*/
int kSum(vector<int> &A, int k, int target) {
// write your code here
int size=A.size();
if (target<)
{
return ;
}
vector<vector<int>> dp(k+,vector<int>(target+,));
dp[][]=;
for (int i=;i<=size;i++)
{
for (int j=k;j>=;j--)
{
for (int t=target;t>=;t--)
{
if (!(i==||j==||t==))
{
if (t>=A[i-])//注意是大于等于;
{
dp[j][t]+=dp[j-][t-A[i-]];
}
}
}
}
}
return dp[k][target]; }
};
int kSum2(vector<int> &A, int k, int target)
{
int size=A.size();
if (target<)
{
return ;
}
vector<vector<int>> dp(k+,vector<int>(target+,));
dp[][]=; for (int i=;i<=size;i++)//注意此处下标范围与下面A下标对应;
{
for (int j=k;j>;j--)
{
for (int t=target;t>;t--)
{
if (t>=A[i-])//注意是大于等于;
{
dp[j][t]+=dp[j-][t-A[i-]];
}
}
}
}
return dp[k][target];
}
还有更简洁的版本:
class Solution {
public:
/**
* @param A: an integer array.
* @param k: a positive integer (k <= length(A))
* @param target: a integer
* @return an integer
*/
int kSum(vector<int> A, int k, int target) {
// wirte your code here T(n, k, target) = O(n*k*target). area(n, k, target) = O(k*target)
int n = A.size();
int dp[k+][target+];
memset(dp, , sizeof(dp));
dp[][] = ;
for (int x = ; x < n; x++)
for (int y = k; y >= ; y--)
for (int z = target; z >= A[x]; z--)
dp[y][z] += dp[y-][z-A[x]];
return dp[k][target];
}
};
优化成二维dp后,i的遍历可以从0~size-1,也可以从1~size,并不影响结果,就是要注意与A下标对应,前者是A【i】,后者是A【i-1】。因为最初始的那张表是dp【0】【0】=1,其他位置都为0。循环开始后,数组A中每个元素都对应一个二维表,计算过程就是用当前二维表不断代替上一状态的二维表。
class Solution {
public:
/**
* @param A: An integer array
* @param k: A positive integer (k <= length(A))
* @param target: An integer
* @return: An integer
*/
int kSum(vector<int> &A, int k, int target) {
// write your code here
int result=;
if (A.empty())
{
return result;
}
ksum(A,k,target,,,,result);
return result;
} void ksum(vector<int> &A, int k, int target,int sum,int ind,int size,int &result)
{
if (size==k)
{
if (sum==target)
{
result++;
}
return ;
}
if (ind>=A.size())
{
return ;
}
ksum(A,k,target,sum+A[ind],ind+,size+,result);
ksum(A,k,target,sum,ind+,size,result);
} };
89 k数和的更多相关文章
- lintcode 中等题:k Sum ii k数和 II
题目: k数和 II 给定n个不同的正整数,整数k(1<= k <= n)以及一个目标数字. 在这n个数里面找出K个数,使得这K个数的和等于目标数字,你需要找出所有满足要求的方案. 样例 ...
- [LeetCode]Median of Two Sorted Arrays 二分查找两个有序数组的第k数(中位数)
二分.情况讨论 因为数组有序,所以能够考虑用二分.通过二分剔除掉肯定不是第k位数的区间.如果数组A和B当前处理的下标各自是mid1和mid2.则 1.假设A[mid1]<B[mid2], ①.若 ...
- 90 k数和 II
原题网址:https://www.lintcode.com/problem/k-sum-ii/description 描述 Given n unique integers, number k (1&l ...
- K数和问题
问题描述 给定n个不同的正整数(数组num),整数k(k < = n)以及一个目标数字target.在这n个数里面找出k个数,使得这k个数的和等于目标数字,求问有多少种方案? 解决思路 该类问题 ...
- 220. Contains Duplicate III 数组指针差k数值差t
[抄题]: Given an array of integers, find out whether there are two distinct indices i and j in the arr ...
- 陕西师范大学第七届程序设计竞赛网络同步赛D ZQ的睡前故事【约瑟夫环1-N数到第k个出队,输出出队顺序/ STL模拟】
链接:https://www.nowcoder.com/acm/contest/121/D来源:牛客网 题目描述 ZQ是一个拥有n女朋友的万人迷,她的每一个女朋友每天晚上都会挨个给他打电话,要他讲了睡 ...
- lintcode:快乐数
快乐数 写一个算法来判断一个数是不是"快乐数". 一个数是不是快乐是这么定义的:对于一个正整数,每一次将该数替换为他每个位置上的数字的平方和,然后重复这个过程直到这个数变为1,或是 ...
- Project Euler 92:Square digit chains 平方数字链
题目 Square digit chains A number chain is created by continuously adding the square of the digits in ...
- [AGC005D] ~K Perm Counting [dp]
题面 传送门 思路 首先可以明确的一点是,本题中出现不满足条件的所有的数,都是分组的 只有模$K$意义下相同的数之间才会出现不满足条件的情况,而且仅出现在相邻的情况 那么我们考虑把这个性质利用起来 我 ...
随机推荐
- this.$router.go()和this.$router.push()的差别
1.this.$router.go(val) => 在history记录中前进或者后退val步,当val为0时刷新当前页面. 2.this.$router.push(path) => 在h ...
- 8张图带你轻松温习Java知识
年初四好,一图胜千言,下面图解均来自Program Creek 网站,目前它们拥有最多的票选. 如果图解没有阐明问题,那么你可以借助它的标题来一窥究竟. 1 字符串不变性 下面这张图展示了这段代码做了 ...
- Spring Boot集成Shiro实战
Spring Boot集成Shiro权限验证框架,可参考: https://shiro.apache.org/spring-boot.html 引入依赖 <dependency> < ...
- 关于a[::-1]
b = a[i:j] 表示复制a[i]到a[j-1],以生成新的list对象,a[:]就相当于完整复制一份a b = a[i:j:s]表示:i,j与上面的一样,但s表示步进,缺省为1.即从i到j每 ...
- js原型继承四步曲及原型继承图
一:js原型继承四步曲 //js模拟类的创建以及继承 //动物(Animal),有头这个属性,eat方法 //名字这个属性 //猫有名字属性,继承Animal,抓老鼠方法 //第一步:创建父类 fun ...
- VBA文件对话框的应用(VBA打开文件、VBA选择文件、VBA选择文件夹)
在VBA中经常要用到文件对话框来进行打开文件.选择文件或选择文件夹的操作.用Microsoft Office提供的文件对话框比较方便.用法如下Application.FileDialog(fileDi ...
- INSTALL_FAILED_TEST_ONLY oppor11p 安装时出现的问题
刚开始出现这个问题 我很懵逼, 因为我之前一直运行都好好的 !!我在网上查了好多资料 试了好几种办法 我先说下我成功的办法 方法1: Android Studio 3.0会在debu ...
- What is the difference between HTTP_CLIENT_IP and HTTP_X_FORWARDED_FOR
What is the difference between HTTP_CLIENT_IP and HTTP_X_FORWARDED_FOR? it is impossible to say. Dif ...
- https://vjudge.net/contest/321565#problem/C 超时代码
#include <iostream> #include <cstdio> #include <queue> #include <algorithm> ...
- 概率dp——cf518D
通过最后的概率求最终的期望 #include<bits/stdc++.h> using namespace std; ; double p,dp[maxn][maxn]; int n,t; ...