前端算法面试题1--栈、队列、链表、字典与哈希表

2024-09-03 07:36

本文主要是介绍前端算法面试题1--栈、队列、链表、字典与哈希表,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

栈–后进先出

数据结构:数组

使用方法:pop(出栈) push(入栈)
在这里插入图片描述

算法题:有效的扣号

function isValid(s) {let stack = [];let map = {"(": ")","{": "}","[": "]"};for (let i = 0; i < s.length; i++) {let ch = s[i];// 如果是左括号,则入栈if (map[ch]) {stack.push(ch);} else {// 如果是右括号,检查栈顶的左括号是否与之匹配if (ch !== map[stack.pop()]) {return false;}}}// 最后检查栈是否为空,如果为空说明所有的括号都匹配,否则说明有不匹配的括号return stack.length === 0;
}console.log(isValid("()[]{}")); // true
console.log(isValid("(]")); // false
console.log(isValid("([)]")); // false
console.log(isValid("{[]}")); // true

队列-先进先出

数据结构:数组

使用方法:shift(出栈) push(入栈)

重点:js的任务队列
在这里插入图片描述

算法题:用两个栈实现队列。

队列(Queue)是一种先进先出(FIFO,First In First Out)的数据结构。以下是一个 JavaScript 队列问题的例子:用两个栈实现队列。

问题描述:
使用两个栈实现一个队列。提供两个方法:push,用于添加元素到队列,pop,用于从队列中删除元素。假设所有的操作都是有效的(比如,对一个空的队列不会调用 pop 操作)。

var Queue = function() {this.stack1 = [];this.stack2 = [];
};Queue.prototype.push = function(x) {this.stack1.push(x);
};Queue.prototype.pop = function() {if (this.stack2.length === 0) {while (this.stack1.length > 0) {this.stack2.push(this.stack1.pop());}}return this.stack2.pop();
};var queue = new Queue();
queue.push(1);
queue.push(2);
console.log(queue.pop()); // 1
queue.push(3);
console.log(queue.pop()); // 2
console.log(queue.pop()); // 3

这个解决方案中,我们使用两个栈:stack1 和 stack2。当添加元素到队列时,我们只需要将元素压入 stack1。当从队列中删除元素时,我们将所有 stack1 中的元素依次弹出并压入 stack2,然后弹出 stack2 的栈顶元素。这样,就实现了队列的先进先出特性。如果 stack2 中还有元素,我们就直接弹出,而不需要再次将 stack1 中的元素转移过来。这是因为 stack2 中的元素都是比 stack1 中的元素更早添加到队列中的,所以应该优先出队。

链表

什么是链表

1.多个元素存储的列表
2.链表中的元素在内存中不是顺序存储的,而是通过“next”指针联系在一起的

链表和数组的区别

1、数组是有序存储的,在中间某个位置删除或者添加某个元素,其他元素要跟着动
2、链表中的元素在内存中不是顺序存储的,而是通过“next”
指针联系在一起的
区别一:
数组:下标
链表:next指针联系在一起
区别二:数据插入
数组:中间插入新的元素、其他元素需要重新计算
链表:不会重新计算,赋值/替换的感觉
区别三:查找
数组:通过下标进行查找即可
链表:每次查找都需要从头开始找

单向链表:

在这里插入图片描述

双向链表:

在这里插入图片描述

数据结构

let a={key:'a'}
let b={key:'b'}
let c={key:'c'}
let d={key:'d'}
a.next=b
b.next=c
c.next=d
d.next=null
console.log(a)
{"key": "a","next": {"key": "b","next": {"key": "c","next": {"key": "d","next": null}}}
}
//遍历链表
let obj=a;
while(obj&&obj.key){console.log(obj.key);obj=obj.next
}
//链表中插入元素
let m={key:'m'}
c.next=m;
m.next=d;
console.log(a)
//删除操作
c.next=d
console.log(a)

js中的原型链原理就是链表结构 通过__proto__连接

function fun(){}
console.log(new fun().__proto__)

在这里插入图片描述

instanceof的原理

JavaScript 中的 instanceof 运算符用于检测构造函数的 prototype 属性是否出现在某个实例对象的原型链上。

其原理可以概括为以下步骤:

获取对象的原型(proto 属性)。
对比该原型与构造函数的 prototype 属性。
如果相等,则返回 true。
如果不相等,则继续向上(proto.proto)追溯原型链并进行比较,直到原型链的终点 null。
如果在原型链上都没有找到与构造函数 prototype 属性相等的原型,则返回 false。
以下是一个简单的示例:

function MyConstructor() {}let instance = new MyConstructor();console.log(instance instanceof MyConstructor); // 输出:true

在这个例子中,instance 是 MyConstructor 的一个实例,所以 instance instanceof MyConstructor 返回 true。

基于 instanceof 的原理,我们也可以手动实现一个 instanceof 的功能:

function myInstanceOf(instance, constructor) {let proto = Object.getPrototypeOf(instance);while (proto) {if (proto === constructor.prototype) {return true;}proto = Object.getPrototypeOf(proto);}return false;
}console.log(myInstanceOf([1,2,3],Array)); 
// 输出:true
环形链表在这里插入图片描述

环形链表是一种特殊的链表,其中最后一个元素指向链表的某个位置,形成一个环。

问题:给定一个链表,确定链表中是否有环。

为了解决这个问题,我们可以使用“快慢指针”或者“龟兔赛跑”的算法。这个算法的原理是创建两个指针,一个快指针和一个慢指针。快指针每次移动两个节点,慢指针每次移动一个节点。如果链表中存在环,那么快指针和慢指针最终会在环中的某个位置相遇。

以下是使用JavaScript实现的代码:

function ListNode(val) {this.val = val;this.next = null;
}function hasCycle(head) {if (head === null || head.next === null) {return false;}let slow = head;let fast = head.next;while (slow !== fast) {if (fast === null || fast.next === null) {return false;}slow = slow.next;fast = fast.next.next;}return true;
}// 创建一个有环的链表
let node1 = new ListNode(1);
let node2 = new ListNode(2);
let node3 = new ListNode(3);
let node4 = new ListNode(4);node1.next = node2;
node2.next = node3;
node3.next = node4;
node4.next = node2;  // 创建环console.log(hasCycle(node1));  // 输出:true

字典和哈希表

字典

字典:键值对存储的,类似于js的对象(键[key]都是字符串类型或者会转换成字符串类型)

字典==》map来表示,map的键不会转换类型(键[key]可以伊任何形式存在)

哈希表 又叫==>散列表

在js中没有哈希表,哈希表是字典的一种实现
记住这句就行 下面可以不用看了 避免混乱

哈希表与字典的区别:
区别一:如果找key对应的value需要遍历key,那么想省略这个步骤可以使用哈希表表示
区别二:排列顺序
字典是根据添加的顺序进行排列的,哈希表不是添加顺序进行排列的

哈希表(Hash Table),也被称为散列表或哈希映射,是一种便于查找和存储数据的数据结构。它的特点是通过一种称为哈希函数(Hash Function)的计算方法,将存储的数据与表中的一个位置(或者说索引)关联起来,从而实现高效的数据查找。

下面是一种通俗的解释:

想象你拥有一个巨大的仓库,需要在其中存储各种各样的物品。一个直接的方法就是,把每一件物品都放入一个巨大的箱子中,然后每次需要某个物品时,都从箱子的一头开始找,直到找到需要的物品。但是,这种方法效率低下,尤其当你需要找的物品在箱子的另一头时。

于是,你想出了一个新的方法:为每一个物品都分配一个特定的编号,然后按照编号的顺序将物品放置在仓库的特定位置。这样,下次需要某个物品时,你只需要查看这个物品的编号,就可以直接找到物品所在的位置,大大提高了查找的效率。

哈希表就是这样一个"仓库",它使用哈希函数为每一份数据生成一个"编号"(我们称之为哈希值),然后将数据存储在相应的位置。当你需要查找数据时,只需要输入数据或者关键字,哈希函数会告诉你数据在哪里,从而实现快速查找。

然而,哈希表并非完美无缺。有时,不同的数据经过哈希函数计算可能会得到相同的哈希值,这种情况我们称之为哈希碰撞。为了解决这个问题,哈希表有多种解决碰撞的方法,如开放定址法、链地址法等。
在 JavaScript 中,没有专门的哈希表(Hash Table)这个数据类型。但是,JavaScript 提供了一些可以实现哈希表功能的对象和数据结构,例如对象(Object)、Map 对象以及 Set 对象。

JavaScript 对象:JavaScript 的对象(Object)可以被看作是键值对的集合,或者说字符串到值的映射。因此,它们可以用来实现哈希表的功能。但是,JavaScript 对象的键必须是字符串或者 Symbol,不能是其他类型的值。

<JAVASCRIPT>
let obj = {};
obj["key1"] = "value1";
obj["key2"] = "value2";

console.log(obj[“key1”]); // 输出:“value1”
Map 对象:ES6 引入了 Map 类型,它类似于普通的 Object,但 Map 的键可以是任何类型的值。Map 提供了一些有用的方法,如 set(key, value)、get(key)、has(key)、delete(key) 和 size 等。

<JAVASCRIPT>
let map = new Map();
map.set("key1", "value1");
map.set("key2", "value2");console.log(map.get("key1"));  // 输出:"value1"

Set 对象:Set 是 ES6 引入的新的数据结构,类似于数组,但 Set 的成员是唯一且无序的,没有重复的值。
以上这些数据结构都可以实现哈希表的功能,如存储键值对、检查键是否存在、删除键等。但实际上,这些数据结构的内部实现可能并不完全基于哈希算法,它们可能使用了其他的数据结构和算法优化性能。

这篇关于前端算法面试题1--栈、队列、链表、字典与哈希表的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1132414

相关文章

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

Vue中组件之间传值的六种方式(完整版)

《Vue中组件之间传值的六种方式(完整版)》组件是vue.js最强大的功能之一,而组件实例的作用域是相互独立的,这就意味着不同组件之间的数据无法相互引用,针对不同的使用场景,如何选择行之有效的通信方式... 目录前言方法一、props/$emit1.父组件向子组件传值2.子组件向父组件传值(通过事件形式)方

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

css中的 vertical-align与line-height作用详解

《css中的vertical-align与line-height作用详解》:本文主要介绍了CSS中的`vertical-align`和`line-height`属性,包括它们的作用、适用元素、属性值、常见使用场景、常见问题及解决方案,详细内容请阅读本文,希望能对你有所帮助... 目录vertical-ali

浅析CSS 中z - index属性的作用及在什么情况下会失效

《浅析CSS中z-index属性的作用及在什么情况下会失效》z-index属性用于控制元素的堆叠顺序,值越大,元素越显示在上层,它需要元素具有定位属性(如relative、absolute、fi... 目录1. z-index 属性的作用2. z-index 失效的情况2.1 元素没有定位属性2.2 元素处

Python实现html转png的完美方案介绍

《Python实现html转png的完美方案介绍》这篇文章主要为大家详细介绍了如何使用Python实现html转png功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 1.增强稳定性与错误处理建议使用三层异常捕获结构:try: with sync_playwright(

Vue 调用摄像头扫描条码功能实现代码

《Vue调用摄像头扫描条码功能实现代码》本文介绍了如何使用Vue.js和jsQR库来实现调用摄像头并扫描条码的功能,通过安装依赖、获取摄像头视频流、解析条码等步骤,实现了从开始扫描到停止扫描的完整流... 目录实现步骤:代码实现1. 安装依赖2. vue 页面代码功能说明注意事项以下是一个基于 Vue.js

Python容器类型之列表/字典/元组/集合方式

《Python容器类型之列表/字典/元组/集合方式》:本文主要介绍Python容器类型之列表/字典/元组/集合方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 列表(List) - 有序可变序列1.1 基本特性1.2 核心操作1.3 应用场景2. 字典(D

CSS @media print 使用详解

《CSS@mediaprint使用详解》:本文主要介绍了CSS中的打印媒体查询@mediaprint包括基本语法、常见使用场景和代码示例,如隐藏非必要元素、调整字体和颜色、处理链接的URL显示、分页控制、调整边距和背景等,还提供了测试方法和关键注意事项,并分享了进阶技巧,详细内容请阅读本文,希望能对你有所帮助...

Nginx实现前端灰度发布

《Nginx实现前端灰度发布》灰度发布是一种重要的策略,它允许我们在不影响所有用户的情况下,逐步推出新功能或更新,通过灰度发布,我们可以测试新版本的稳定性和性能,下面就来介绍一下前端灰度发布的使用,感... 目录前言一、基于权重的流量分配二、基于 Cookie 的分流三、基于请求头的分流四、基于请求参数的分