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
相关文章
|
4月前
|
存储 监控 算法
局域网监控其他电脑的设备信息管理 Node.js 跳表算法
跳表通过分层索引实现O(logn)的高效查询、插入与删除,适配局域网监控中设备动态接入、IP映射及范围筛选等需求,相比传统结构更高效稳定,适用于Node.js环境下的实时设备管理。
171 9
|
5月前
|
机器学习/深度学习 存储 算法
动态规划算法深度解析:0-1背包问题
0-1背包问题是经典的组合优化问题,目标是在给定物品重量和价值及背包容量限制下,选取物品使得总价值最大化且每个物品仅能被选一次。该问题通常采用动态规划方法解决,通过构建二维状态表dp[i][j]记录前i个物品在容量j时的最大价值,利用状态转移方程避免重复计算子问题,从而高效求解最优解。
663 1
|
6月前
|
存储 监控 JavaScript
基于布隆过滤器的 Node.js 算法在局域网电脑桌面监控设备快速校验中的应用研究
本文探讨了布隆过滤器在局域网电脑桌面监控中的应用,分析其高效空间利用率、快速查询性能及动态扩容优势,并设计了基于MAC地址的校验模型,提供Node.js实现代码,适用于设备准入控制与重复数据过滤场景。
251 0
|
4月前
|
存储 监控 JavaScript
企业上网监控系统的恶意 URL 过滤 Node.js 布隆过滤器算法
布隆过滤器以低内存、高效率特性,解决企业上网监控系统对百万级恶意URL实时检测与动态更新的难题,通过概率性判断实现毫秒级过滤,内存占用降低96%,适配大规模场景需求。
302 3
|
4月前
|
存储 监控 算法
电脑管控软件的进程优先级调度:Node.js 红黑树算法
红黑树凭借O(log n)高效插入、删除与查询特性,适配电脑管控软件对进程优先级动态调度的高并发需求。其自平衡机制保障系统稳定,低内存占用满足轻量化部署,显著优于传统数组或链表方案,是实现关键进程资源优先分配的理想选择。
245 1
|
5月前
|
运维 监控 JavaScript
基于 Node.js 图结构的局域网设备拓扑分析算法在局域网内监控软件中的应用研究
本文探讨图结构在局域网监控系统中的应用,通过Node.js实现设备拓扑建模、路径分析与故障定位,提升网络可视化、可追溯性与运维效率,结合模拟实验验证其高效性与准确性。
317 3
|
9月前
|
监控 算法 JavaScript
基于 JavaScript 图算法的局域网网络访问控制模型构建及局域网禁止上网软件的技术实现路径研究
本文探讨局域网网络访问控制软件的技术框架,将其核心功能映射为图论模型,通过节点与边表示终端设备及访问关系。以JavaScript实现DFS算法,模拟访问权限判断,优化动态策略更新与多层级访问控制。结合流量监控数据,提升网络安全响应能力,为企业自主研发提供理论支持,推动智能化演进,助力数字化管理。
222 4
|
9月前
|
监控 算法 JavaScript
公司局域网管理视域下 Node.js 图算法的深度应用研究:拓扑结构建模与流量优化策略探析
本文探讨了图论算法在公司局域网管理中的应用,针对设备互联复杂、流量调度低效及安全监控困难等问题,提出基于图论的解决方案。通过节点与边建模局域网拓扑结构,利用DFS/BFS实现设备快速发现,Dijkstra算法优化流量路径,社区检测算法识别安全风险。结合WorkWin软件实例,展示了算法在设备管理、流量调度与安全监控中的价值,为智能化局域网管理提供了理论与实践指导。
235 3
|
9月前
|
存储 监控 算法
公司内部网络监控中的二叉搜索树算法:基于 Node.js 的实时设备状态管理
在数字化办公生态系统中,公司内部网络监控已成为企业信息安全管理体系的核心构成要素。随着局域网内终端设备数量呈指数级增长,实现设备状态的实时追踪与异常节点的快速定位,已成为亟待解决的关键技术难题。传统线性数据结构在处理动态更新的设备信息时,存在检索效率低下的固有缺陷;而树形数据结构因其天然的分层特性与高效的检索机制,逐渐成为网络监控领域的研究热点。本文以二叉搜索树(Binary Search Tree, BST)作为研究对象,系统探讨其在公司内部网络监控场景中的应用机制,并基于 Node.js 平台构建一套具备实时更新与快速查询功能的设备状态管理算法框架。
310 3
|
11月前
|
存储 算法 JavaScript
基于 Node.js 深度优先搜索算法的上网监管软件研究
在数字化时代,网络环境呈现出高度的复杂性与动态性,上网监管软件在维护网络秩序与安全方面的重要性与日俱增。此类软件依托各类数据结构与算法,实现对网络活动的精准监测与高效管理。本文将深度聚焦于深度优先搜索(DFS)算法,并结合 Node.js 编程语言,深入剖析其在上网监管软件中的应用机制与效能。
158 6

热门文章

最新文章