如图,写一个方法,传入一个给定的数字,如 60,使用1,2,5这三个数组成60,求大佬解惑
以下是己一丝拙见,在此抛砖引玉了
1、传入面额无需全部使用(请注意以下方法对入参未进行校验
)
let cions = [3,77], N = 101;
const fn = (cions, N) => {
if(cions.indexOf(1) >= 0 || cions.filter(item => N % item === 0).length > 0) {
return true;
}
let total = cions.reduce((total, current) => (total+current), 0);
let q = cions.filter(item => {
if((N-total) % item === 0) {
return true;
}
let result = false;
let diff = N - 2*total;
while(diff > total) {
if(diff % item === 0) {
result = true;
break;
}
diff -= total;
}
return result;
});
if(q.length > 0) {
return true;
}
return false;
}
console.log(fn(coins, N); //output: true
2、全用上的情况也和未全用上的情况是一样的,就是判断条件仅有一种便可
let cions = [3,77], N = 101;
const fn = (cions, N) => {
let total = cions.reduce((total, current) => (total+current), 0);
let q = cions.filter(item => {
if((N-total) % item === 0) {
return true;
}
let result = false;
let diff = N - 2*total;
while(diff > total) {
if(diff % item === 0) {
result = true;
break;
}
diff -= total;
}
return result;
});
if(q.length > 0) {
return true;
}
return false;
}
console.log(fn(coins, N); //output: true
没有考虑使用 数组中的每个元素
function fun(num, arr) {
// 先对数组从小到大排序
let list = arr.sort((a, b) => { return a - b; }),
residual = null;
// 对数字 num 与数组从往前一次求余数,余数为 0 则返回 true
for (let index = list.length - 1; index >= 0; index--) {
const item = list[index];
- //residual = num % item;
+ residual = num %= item; // 直接将余数复制给 num
if (residual === 0) {
- // break;
+ return true; // 此处直接返回结果
}
- //else {
- // let d = parseInt(num / item);
- // num -= item * d;
- //}
}
if (residual !== 0) {
return false;
}
- //if (residual === 0) {
- // return true;
- //} else {
- // return false;
- //}
}
console.log(fun(60, [1, 2, 5])); // true
console.log(fun(100, [3, 77])); // false
function getCoin(N, coins) {
if (typeof N == 'string' && /\D(?=>\.)/.test(N)) {
console.warn('请传入整数');
return;
}
N = Math.floor(Number(N));
coins.sort((a, b) => b - a);
let index = 0;
function _temp(money) {
money = money % coins[index]
if (index >= coins.length) return false;
if (money > 0) {
return _temp(money, coins, ++index);
} else {
return true;
}
}
return _temp(N)
}
console.log(getCoin("11.01", [1, 5, 2]))
console.log(getCoin(100, [3, 77]))
/*
判断组合[c1,, c2, ...]能够凑够N
*/
function calcBase(coins, N){
// coins中没有内容了
if(!coins.length) return false;
// 只有一种硬币,判断是否能整除N
if(coins.length == 1){
return N % coins[0] == 0;
}
// 依次按大硬币从少到多递归
for(var i=coins.length-1; i>=0; i--){
// 判断是否能够整除
if(N % coins[i] == 0){
return true;
}
// 不能整除时,如果能够使用最大硬币,则最大硬币从1到n
while(N >= coins[i]){
N -= coins[i];
// 递归查询
if(calcBase(coins.slice(0, i), N)){
return true;
}
}
}
return false;
}
/*
思路:
1、将硬币按照面额大小从小到大排序
2、从最小的硬币开始试探能否凑成N
3、如果硬币c1不能凑成N,则使用组合[c1,c2],直到所有硬币用完
*/
function calc(coins, N){
// 如果硬币中包含1,则返回true
if(coins.includes(1)) return true;
// 硬币排序
coins = coins.sort();
// 从最小硬币开始
for(let i=1, len=coins.length; i<=len; i++){
if( calcBase(coins.slice(0, i), N) ){
return true;
}
}
return false;
}
console.time();
console.log(calc([6, 7, 26, 12, 45], 14));
console.timeEnd()
function fn (cion, N) {
let cionArr = cion.sort((a, b) => {return b - a;});
for(var i = 0; i < cionArr.length; i++) {
if(N % cionArr[i] > 0) {
N = N % cionArr[i];
console.log(N);
}else { return true; }
}
if(i == cionArr.length){return false}
}
fn([3,77], 100) // false
fn([1,2,5], 11) // true
这样有什么漏洞么?
错误的示例,各位绕过
function count (coins = [], sum = 0) {
for (let coin of coins) {
let remainder = sum % coin;
// console.log('----> ', sum, coin, remainder);
if (remainder === 0) {
return true;
} else {
// 取余算,如果有这种组合,可最快算出
let result = count(
coins.filter(_coin => (coin !== _coin)),
remainder
);
// 递减算
let result1 = (sum - coin > 0) && count(
coins.filter(_coin => (coin !== _coin)),
sum - coin
);
if (result || result1) {
return true;
}
}
}
return false;
}
10 回答11.4k 阅读
5 回答4.9k 阅读✓ 已解决
4 回答3.2k 阅读✓ 已解决
2 回答2.9k 阅读✓ 已解决
3 回答2.5k 阅读✓ 已解决
3 回答2.3k 阅读✓ 已解决
2 回答2.7k 阅读✓ 已解决
这个题是一个经典的组合问题
分析
一些短路条件
这个策略是单纯的遍历无需递归
这个算法不需要递归,可以转换成常规遍历问题。比如可以根据B中各个值对应bi情况,把bi转换成n位二进制数表示位cbi(表示bi可以有cbi来表示,cbi最大值>=bi),这样整个搜索空间就映射到一个整数上,整数的位数是所有cbi的和,比如cb0=2,cb1=5,则整数的位数就是2+5=7,其中前2位对应于cb0的表达,后5位对应于cb1的表达。就可以很快构造出遍历循环的条件,结合下面的优化处理,过滤掉不符合的情况,可以很快遍历出来。
一些优化处理
另外一些考虑
因为有些语言不支持无限长度的整数(python是支持的),所以可能直接位表达会出现问题,需要做一些特殊处理。
javascript实现