js 实现 贪心算法和动态规划 贪心找零问题, 动态规划 青蛙跳台阶问题

简介: js 实现 贪心算法和动态规划 贪心找零问题, 动态规划 青蛙跳台阶问题
// 贪心算法和动态规划
// 当遇到一个求解全局最优解问题时,如果可以将全局问题切分为小的局部问题,
// 并寻求局部最优解,同时可以证明局部最优解累计的结果就是全局最优解,则可以使用贪心算法
// 找零问题
// 示例:假设你有一间小店,需要找给客户46分钱的硬币,
// 你的货柜里只有面额为25分、10分、5分、1分的硬币,如何找零才能保证数额正确并且硬币数最小
/**
 * 找零问题 使用贪心算法
 * @param total {Number} 总金额
 * @param coinArr {Array} 面值的结合
 * @returns {*[]|[...[number], ...[]|[number]]}
 */
function getCoinCount(total = 0, coinArr = []) {
    if (total <= 0 || coinArr.length === 0) return [];
    // 取出最大的面额
    let max = 0;
    for (let i = 0, l = coinArr.length; i < l; i++) {
        if (coinArr[i] <= total && coinArr[i] >= max) {
            max = coinArr[i];
        }
    }
    // 通过上面就可以找出来一个最大的面额
    let res = [max];
    // 获取下一个局部最大的解
    let nextRes = getCoinCount(total - max, coinArr);
    // 合并结果
    res = [...res, ...nextRes];
    return res;
}
// console.log(getCoinCount(155, [50, 25, 10, 5, 1])); [ 50, 50, 50, 5 ]
// console.log(getCoinCount(51, [30, 25, 10, 5, 1])); [ 30, 10, 10, 1 ]
// 但是找零问题,使用贪心算法是有缺陷的,如 getCoinCount(51, [30, 25, 10, 5, 1])
// 得出的结果是 [ 30, 10, 10, 1 ]  但是实际上还有更优解 [25,25,1]
// 动态规划
// 分治法有一个问题,就是容易重复计算已经计算过的值,使用动态规划,
// 可以讲每一次分治时算出的值记录下来,防止重复计算,从而提高效率。
// 青蛙跳台阶 有N级台阶,一只青蛙每次可以跳1级或两级,一共有多少种跳法可以跳完台阶?
// 分析:
// 台阶 1 级  跳法  1 种
// 台阶 2 级  跳法  2 种
// 台阶 3 级  跳法  3 种
// 台阶 4 级  跳法  5 种
// ....  发现有点像斐波那契数列
// 使用分治法
let num = 0;
/**
 * 分治法 求青蛙跳台阶
 * @param n {Number} 台阶数量
 * @returns {number|*}
 */
function frogJumpSteps(n = 0) {
    if (n <= 2) return n;
    num += 1;
    return frogJumpSteps(n - 1) + frogJumpSteps(n - 2);
}
// console.log(frogJumpSteps(10), num); // 结果 89 54
let num2 = 0;
/**
 * 动态规划来计算 通过牺牲空间来换取时间
 * @param num {Number} 跳的台阶数量
 * @returns { Number} 返回跳的方式数量
 */
function frogJumpStepsDynamic(num = 0) {
    let cache = {};
    function _frogJumpStepsDynamic(num) {
        if (num <= 2) return num;
        else if (cache[num]) return cache[num];
        else {
            num2++;
            let res = _frogJumpStepsDynamic(num - 2) + _frogJumpStepsDynamic(num - 1);
            cache[num] = res;
            return res;
        }
    }
    return _frogJumpStepsDynamic(num)
}
// console.log(frogJumpStepsDynamic(10), num2);  // 结果 89 8
// 最长公共子序列问题(LCS)
// 有的时候,我们需要比较两个字符串的相似程度,通常就是比较两个字符串有多少相同的公共子序列
// 例如有两个字符串
//
// abc1223de
// bc987de,但是他的女朋友们不喜欢
// 以上两个字符串的最长公共子序列为:bcde
// 分析:
// 情况1: 第一位的字符串相同  那继续比较后面的字符串
// 情况2: 第一位字符串不一样,会产生两种情况
//        1. 第一个字符串去除掉首位
//        2. 第二个字符串去掉首位
//
let num3 = 0;
/**
 * 使用分治法 求最长子序列
 * @param str1 {String}
 * @param str2 {String}
 * @returns {string|*}
 */
function getLCS(str1, str2) {
    if (str1.length === 0 || str2.length === 0) return '';
    num3++;
    // 情况1
    if (str1[0] === str2[0]) return str1[0] + getLCS(str1.substr(1), str2.substr(1));
    else {
        let str1Res = getLCS(str1.substr(1), str2);
        let str2Res = getLCS(str1, str2.substr(1));
        return str1Res.length > str2Res.length ? str1Res : str2Res;
    }
}
console.log(getLCS('测试分治法特有的', '分治法,测试他的'),num3); // 分治法的 6863
let num4 = 0;
/**
 * 使用动态规划做缓存,实现求最长子序列
 * @param str1 {String}
 * @param str2 {String}
 * @returns {string}
 */
function getLCSDynamic(str1, str2) {
    let cache = [];
    let _getLCSDynamic = (str1, str2) => {
        if (str1.length === 0 || str2.length === 0) return '';
        num4 ++;
        for (let i = 0, l = cache.length; i < l; i++) {
            if (cache[i].str1 === str1 && cache[i].str2 === str2) return cache[i].res;
        }
        if (str1[0] === str2[0]) {
            let res = str1[0] + _getLCSDynamic(str1.substr(1), str2.substr(1));
            cache.push({
                str1: str1.substr(1),
                str2: str2.substr(1),
                res: res
            })
            return res;
        } else {
            let str1Res = _getLCSDynamic(str1.substr(1), str2);
            cache.push({
                str1: str1.substr(1),
                str2: str2,
                res: str1Res
            });
            let str2Res = _getLCSDynamic(str1, str2.substr(1));
            cache.push({
                str1: str1,
                str2: str2.substr(1),
                res: str2Res
            });
            return str1Res.length > str2Res.length ? str1Res : str2Res;
        }
    }
    return _getLCSDynamic(str1, str2);
}
console.log(getLCSDynamic('测试分治法特有的', '分治法,测试他的'), num4); //分治法的 79
相关文章
|
9天前
|
运维 监控 JavaScript
内网网管软件中基于 Node.js 的深度优先搜索算法剖析
内网网管软件在企业网络中不可或缺,涵盖设备管理、流量监控和安全防护。本文基于Node.js实现深度优先搜索(DFS)算法,解析其在网络拓扑遍历中的应用。通过DFS,可高效获取内网设备连接关系,助力故障排查与网络规划。代码示例展示了图结构的构建及DFS的具体实现,为内网管理提供技术支持。
32 11
|
2天前
|
存储 算法 Java
算法系列之动态规划
动态规划(Dynamic Programming,简称DP)是一种用于解决复杂问题的算法设计技术。它通过将问题分解为更小的子问题,并存储这些子问题的解来避免重复计算,从而提高算法的效率。
17 4
算法系列之动态规划
|
6天前
|
监控 网络协议 算法
基于问题“如何监控局域网内的电脑”——Node.js 的 ARP 扫描算法实现局域网内计算机监控的技术探究
在网络管理与安全领域,监控局域网内计算机至关重要。本文探讨基于Node.js的ARP扫描算法,通过获取IP和MAC地址实现有效监控。使用`arp`库安装(`npm install arp`)并编写代码,可定期扫描并对比设备列表,判断设备上线和下线状态。此技术适用于企业网络管理和家庭网络安全防护,未来有望进一步提升效率与准确性。
23 8
|
10天前
|
算法 安全 调度
【动态规划篇】穿越算法迷雾:约瑟夫环问题的奇幻密码
【动态规划篇】穿越算法迷雾:约瑟夫环问题的奇幻密码
|
10天前
|
机器学习/深度学习 算法 测试技术
【动态规划篇】01 背包的逆袭:如何用算法装满你的 “财富背包”
【动态规划篇】01 背包的逆袭:如何用算法装满你的 “财富背包”
|
2月前
|
JavaScript 算法 安全
深度剖析:共享文件怎么设置密码和权限的 Node.js 进阶算法
在数字化时代,共享文件的安全性至关重要。本文聚焦Node.js环境,介绍如何通过JavaScript对象字面量构建数据结构管理文件安全信息,包括使用`bcryptjs`库加密密码和权限校验算法,确保高效且安全的文件共享。通过实例代码展示加密与权限验证过程,帮助各行业实现严格的信息资产管理与协作。
|
2月前
|
算法 Java C++
【潜意识Java】蓝桥杯算法有关的动态规划求解背包问题
本文介绍了经典的0/1背包问题及其动态规划解法。
56 5
|
2月前
|
存储 监控 算法
局域网网络管控里 Node.js 红黑树算法的绝妙运用
在数字化办公中,局域网网络管控至关重要。红黑树作为一种自平衡二叉搜索树,凭借其高效的数据管理和平衡机制,在局域网设备状态管理中大放异彩。通过Node.js实现红黑树算法,可快速插入、查找和更新设备信息(如IP地址、带宽等),确保网络管理员实时监控和优化网络资源,提升局域网的稳定性和安全性。未来,随着技术融合,红黑树将在网络管控中持续进化,助力构建高效、安全的局域网络生态。
56 9
|
3月前
|
监控 算法 JavaScript
基于 Node.js Socket 算法搭建局域网屏幕监控系统
在数字化办公环境中,局域网屏幕监控系统至关重要。基于Node.js的Socket算法实现高效、稳定的实时屏幕数据传输,助力企业保障信息安全、监督工作状态和远程技术支持。通过Socket建立监控端与被监控端的数据桥梁,确保实时画面呈现。实际部署需合理分配带宽并加密传输,确保信息安全。企业在使用时应权衡利弊,遵循法规,保障员工权益。
62 7
|
2月前
|
存储 监控 JavaScript
深度探秘:运用 Node.js 哈希表算法剖析员工工作时间玩游戏现象
在现代企业运营中,确保员工工作时间高效专注至关重要。为应对员工工作时间玩游戏的问题,本文聚焦Node.js环境下的哈希表算法,展示其如何通过快速查找和高效记录员工游戏行为,帮助企业精准监测与分析,遏制此类现象。哈希表以IP地址等为键,存储游戏网址、时长等信息,结合冲突处理与动态更新机制,确保数据完整性和时效性,助力企业管理层优化工作效率。
35 3