当前位置: 首页 > news >正文

【前端面试】七、算法-递归

常考算法

  1. 排序算法:快速排序、归并排序、堆排序等。

  2. 查找算法:二分查找、哈希表查找等。

  3. 动态规划:解决最优化问题,如斐波那契数列、最长公共子序列等。

  4. 图论算法:最短路径(Dijkstra、Floyd-Warshall)、拓扑排序等。

  5. 字符串处理:KMP算法、正则表达式匹配等。

遍历方法总结

链式调用

  • 数组的很多操作可以构成链式操作,类似这样的格式:…map().filter(…).sort(…).map(….)
  • 链式操作就是对象方法返回类型是自身的。比如map是属于数组的方法,它返回数组,所以构成了链式操作
  • 优势:语义清晰、思考方便,数据量小的时候很有用(<1W)
  • 问题:性能、空间

递归

递归通常需要初始条件和递归表达式

阶乘:n! = n x (n-1) !   

斐波那契:f(1) = 1, f(2) = 1,f(n) = f(n-1) + f(n-2), n>2 

拷贝

push/pop/shift/unshift/splice:都在原始数据上进行修改

concat/slice/map/reduce:都会对原始数据进行浅拷贝

DOM结点的绝对位置

offsetLeft、offsetRight相对于offsetParent的位置
Element.getBoundingClientRect()相对于视窗的位置,受滚动的影响

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>递归</title>
</head>
<body><script>// 阶乘function factorial(n) {if (n === 1) return 1return n * factorial(n - 1)}// 斐波那契数列 1 1 2 3 5 8 13 21 34 55 89 144function fibonacci(n) {if (n === 1 || n === 2) return 1return fibonacci(n - 1) + fibonacci(n - 2)}// 从底端构造递归function fibonacci1(n) {let [a,b] = [1,1]for(let i = 3; i <= n; i++) {[a,b] = [b, a + b]}return b}// console.log('测试 fibonacci1=============');// console.log(fibonacci1(10));function fibonacci2(n) {return Array(n - 2).fill(0).reduce(([a,b],_) => {return [b, a + b]}, [1,1])[1]}// console.log('测试 fibonacci2=============');// console.log(fibonacci2(10));// 递归实现深拷贝function deepClone(obj) {if (typeof obj !== 'object' || obj === null) return obj// const newObj = Array.isArray(obj) ? [] : {}// const newObj = obj instanceof Array ? [] : {}// const newObj = obj.constructor === Array ? [] : {}// const newObj = Object.prototype.toString.call([]) === '[object Array]' ? [] : {}const newObj = new obj.constructor()for(let key in obj) {if(obj.hasOwnProperty(key)) {newObj[key] = deepClone(obj[key])}}return newObj}// 测试用例  function testDeepClone() {  console.log("测试普通对象:");  const obj1 = { a: 1, b: { c: 2 } };  const clonedObj1 = deepClone(obj1);  console.assert(obj1 !== clonedObj1, "对象应该是不同的引用");  console.assert(obj1.b !== clonedObj1.b, "嵌套对象也应该是不同的引用");  console.assert(obj1.b.c === clonedObj1.b.c, "嵌套对象的属性值应该相等");  console.log("测试数组:");  const arr1 = [1, 2, [3, 4]];  const clonedArr1 = deepClone(arr1);  console.assert(arr1 !== clonedArr1, "数组应该是不同的引用");  console.assert(arr1[2] !== clonedArr1[2], "嵌套数组也应该是不同的引用");  console.assert(arr1[2][0] === clonedArr1[2][0], "嵌套数组的元素值应该相等");  console.log("测试特殊对象(Date):");  const date1 = new Date();  const clonedDate1 = deepClone(date1);  console.assert(date1 !== clonedDate1, "Date 对象应该是不同的引用");  console.assert(date1.getTime() === clonedDate1.getTime(), "Date 的时间戳应该相等");  // console.log("测试特殊对象(RegExp):");   // 失败 // const reg1 = /hello/g;  // const clonedReg1 = deepClone(reg1);  // console.assert(reg1 !== clonedReg1, "RegExp 对象应该是不同的引用");  // console.assert(reg1.source === clonedReg1.source && reg1.global === clonedReg1.global, "RegExp 的属性和标志应该相等");  // console.log("测试循环引用:");   // 失败 // const obj2 = {};  // obj2.self = obj2;  // const clonedObj2 = deepClone(obj2);  // console.assert(obj2 !== clonedObj2, "对象应该是不同的引用");  // console.assert(clonedObj2.self === clonedObj2, "循环引用应该被正确处理");  console.log("所有测试通过!");  }  // testDeepClone();// 深度比较function deepCompare(a,b){if (a === null || typeof a !== 'object' || b === null || typeof b !== 'object') {return a === b}// Object.getOwnPropertyDescriptors 方法会返回对象自身的所有属性描述符,包括不可枚举的属性const propsA = Object.getOwnPropertyDescriptors(a)const propsB = Object.getOwnPropertyDescriptors(b)if(Object.keys(propsA).length !== Object.keys(propsB).length) return falsereturn Object.keys(propsA).every(key => deepCompare(a[key],b[key]))}// 测试用例  function testDeepCompare() {  console.log("测试基本相等性:");  console.assert(deepCompare(1, 1), "1 应该等于 1");  console.assert(!deepCompare(1, 2), "1 不应该等于 2");  console.assert(deepCompare(null, null), "null 应该等于 null");  console.assert(deepCompare(undefined, undefined), "undefined 应该等于 undefined");  console.assert(!deepCompare(null, undefined), "null 不应该等于 undefined");  console.log("测试对象比较:");  const obj1 = { a: 1, b: { c: 2 } };  const obj2 = { a: 1, b: { c: 2 } };  const obj3 = { a: 1, b: { c: 3 } };  console.assert(deepCompare(obj1, obj2), "obj1 应该等于 obj2");  console.assert(!deepCompare(obj1, obj3), "obj1 不应该等于 obj3");  console.log("测试数组比较:");  const arr1 = [1, 2, [3, 4]];  const arr2 = [1, 2, [3, 4]];  const arr3 = [1, 2, [3, 5]];  console.assert(deepCompare(arr1, arr2), "arr1 应该等于 arr2");  console.assert(!deepCompare(arr1, arr3), "arr1 不应该等于 arr3");  // console.log("测试循环引用(此实现可能无法正确处理):");  // const obj4 = {};  // obj4.self = obj4;  // const obj5 = {};  // obj5.self = obj5;  // 注意:此实现可能无法正确处理循环引用,因为它会陷入无限递归  // 这里我们假设它不会处理循环引用,并跳过这个测试  // console.assert(deepCompare(obj4, obj5), "循环引用对象应该相等(但这里不测试)");  console.log("所有测试通过!");  }  // testDeepCompare();// DOM节点的绝对位置function getLayout1(el) {if (!el) return;const layout = {width: el.offsetWidth,height: el.offsetHeight,top: el.offsetTop,left: el.offsetLeft}if(el.offsetParent) {const parentLayout = getLayout1(el.offsetParent)layout.top += parentLayout.toplayout.left += parentLayout.left}return layout}function getLayout2(el) {if (!el) return;let left = el.offsetLeftlet top = el.offsetToplet p = el.offsetParentwhile(p) {left += p.offsetLefttop += p.offsetTopp = p.offsetParent}return {width: el.offsetWidth,height: el.offsetHeight,top,left}}</script></body>
</html>

http://www.lryc.cn/news/413761.html

相关文章:

  • CmsEasy逻辑漏洞--零元购
  • Linux 内核源码分析---I/O 体系结构与访问设备
  • 在cPanelWHM中如何重置 MySQL 用户帐户密码
  • 软件测试基础1--功能测试
  • 《计算机网络》(第8版)第9章 无线网络和移动网络 复习笔记
  • 非负数、0和正整数 限制最大值且保留两位小数在elementpuls表单中正则验证
  • Java多线程-----定时器(Timer)及其实现
  • 【Linux修行路】进度条小程序
  • 网络安全入门教程(非常详细)从零基础入门到精通,看完这一篇就够了。
  • 【探索Linux】P.44(数据链路层 —— 以太网的帧格式 | MAC地址 | MTU | ARP协议)
  • <数据集>航拍行人识别数据集<目标检测>
  • 在 Windows 10 系统上部署 Medusa
  • Linux进程 (冯诺依曼体结构 管理 PCB 进程状态 僵尸进程 孤儿进程 运行阻塞挂起状态 进程优先级)
  • 《LlamaIndex 之美》-01-LLM、Prompt、Embedding基础入门
  • C++ 智能指针简单介绍及用法
  • k8s笔记之创建Istio Gateway规则
  • NAND行业回归盈利:AI与云存储需求驱动
  • 【限免】频控阵雷达:概念、原理与应用【附MATLAB代码】
  • 从0开始搭建vue + flask 旅游景点数据分析系统( 六):搭建后端flask框架
  • 学习硬件测试04:触摸按键+PWM 驱动蜂鸣器+数码管(P62~P67、P71、P72)
  • JS原型链
  • 《Java初阶数据结构》----5.<二叉树的概念及使用>
  • git查看记录详解
  • 检索增强生成RAG系列10--RAG的实际案例
  • 程序员自我提升的全面指南
  • 【golang】Golang手写元组 tuple | golang tuple
  • golang中struct的tag -简记
  • 分布式领域扩展点设计稿
  • 玩转微信公众号变现:从新手到专家的全攻略
  • JVM: 方法调用