整数数组中找到相加等于某个数的组合

给定一个正整数的数组和一个数sum,在数组抽取n个数相加等于sum,找出所有的可能,也可能无解。

每个数有两种状态取和不取,用0,1表示,数组中数字的组合可能有就 2**(length-1)种,如果是暴利破解就是2**length -1 次计算。

想想可以优化一下,求和的话可以先把数组排序,找到数组中比sum小的那一项a,然后用sum减去a得到b,那么可以看成是在a后面那一截数组中找到相加等于b的组合,依次这样递归下去。具体实现如下:

 

const array = [1,2,3,4,5,6,10,9,7,11,23];
const sum = 30;
array.sort(function(a,b){
return a-b;
})

console.log(array)
const resultArray = [];

function count(array,sum)
{
var length = array.length;
if(array[0]>sum)
{
return false; 
}
else if(array.indexOf(sum)!=-1)
{
resultArray.push(sum);
console.log(resultArray);
resultArray.pop();

}
else{

for(var i=length-1;i>=0;i--)
{ 
if(array[i]<sum)
{
resultArray.push(array[i]);
count(array.slice(0,i),sum-array[i])
resultArray.pop();	
}
}

} 
}

count(array,sum);

  

结果是

[ 23, 7 ]
[ 11, 10, 9 ]
[ 11, 9, 7, 3 ]
[ 11, 9, 6, 4 ]
[ 11, 9, 5, 4, 1 ]
[ 11, 9, 5, 3, 2 ]
[ 11, 9, 4, 3, 2, 1 ]
[ 11, 7, 6, 5, 1 ]
[ 11, 7, 6, 4, 2 ]
[ 11, 7, 6, 3, 2, 1 ]
[ 11, 7, 5, 4, 3 ]
[ 11, 6, 5, 4, 3, 1 ]
[ 10, 9, 7, 4 ]
[ 10, 9, 6, 5 ]
[ 10, 9, 5, 4, 2 ]
[ 10, 9, 5, 3, 2, 1 ]
[ 10, 7, 6, 5, 2 ]
[ 10, 7, 6, 4, 3 ]
[ 10, 7, 5, 4, 3, 1 ]
[ 10, 6, 5, 4, 3, 2 ]
[ 9, 7, 6, 5, 3 ]
[ 9, 7, 6, 4, 3, 1 ]
[ 9, 7, 5, 4, 3, 2 ]
[ 9, 6, 5, 4, 3, 2, 1 ]

转载于:https://www.cnblogs.com/chillaxyw/p/10633007.html

好的,我会为您解答这个问题。这是一个经典的“背包问题”,可以使用回溯算法来解决。在回溯算法中,我们需要维护一个“路径”变量,记录当前已选的的集合,以及一个“剩余值”变量,记录还需要达到的目标和值。每次选择一个,加入路径中,同时更新剩余值,如果剩余值为0,则找到一个解;如果剩余值小于0,则说明当前路径不可行,需要回溯到上一个状态,撤销选择。 下面是程序的实现,其中backtrack函是回溯函,path和res分别是路径和结果集,start是控制遍历的起始位置,注意到同一位置的可以重复使用,因此需要从当前位置开始遍历。 ```python def backtrack(path, res, start, target, nums): if target == 0: res.append(path[:]) return for i in range(start, len(nums)): if target - nums[i] < 0: break path.append(nums[i]) backtrack(path, res, i, target - nums[i], nums) path.pop() nums = [4, 3, 2, 2, 1, 1] target = 4 path, res = [], [] backtrack(path, res, 0, target, nums) print(res) ``` 下面是程序调试过程中搜索空间树上每个结点的值: ``` # 空路径 [] # 选择4,剩余0,找到一个解 [4] # 选择3,剩余1 [4, 3] # 选择2,剩余-1,回溯 [4] # 选择2,剩余2 [4, 2] # 选择2,剩余0,找到一个解 [4, 2, 2] # 选择1,剩余-1,回溯 [4, 2] # 选择1,剩余1 [4, 2, 1] # 选择1,剩余0,找到一个解 [4, 2, 1, 1] # 回溯到[4] # 选择1,剩余3 [4, 1] # 选择2,剩余1 [4, 1, 2] # 选择2,剩余-1,回溯 [4, 1] # 选择1,剩余0,找到一个解 [4, 1, 1] # 回溯到[] # 选择1,剩余3 [1] # 选择2,剩余1 [1, 2] # 选择2,剩余-1,回溯 [1] # 选择1,剩余0,找到一个解 [1, 1] ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值