本文详细介绍了JS基础知识和高级特性,并提供了一系列的代码示例和面试题解析,帮助读者巩固JS技能。此外,文章还分享了大厂面试中的常见题型和实战演练,特别是涵盖了js大厂面试真题的相关内容。
JS基础知识回顾 变量与数据类型在JavaScript中,变量是用来存储数据的容器。JavaScript是一种动态类型语言,这意味着你不需要在声明变量时指定其类型,但是JavaScript的变量主要包含以下几种类型:
- 基本数据类型:包括
Number
、String
、Boolean
、null
、undefined
和Symbol
(ES6新增的类型)。 - 引用数据类型:主要是
Object
,包括数组、函数等。
代码示例
// 基本数据类型
let numberValue = 42; // Number
let stringValue = "Hello, World!"; // String
let booleanValue = true; // Boolean
let nullValue = null; // null
let undefinedValue = undefined; // undefined
let symbolValue = Symbol("symbol"); // Symbol
// 引用数据类型
let arrayValue = [1, 2, 3]; // Array
let objectValue = { key: "value" }; // Object
let functionValue = function() {}; // Function
函数与作用域
在JavaScript中,函数是第一类对象。这意味着函数可以赋值给变量,作为参数传递,以及作为返回值。此外,函数内部可以声明其他函数。JavaScript的作用域分为全局作用域和局部作用域。
代码示例
// 全局作用域
let globalVar = "I'm global!";
function globalFunction() {
console.log(globalVar); // 输出 "I'm global!"
}
globalFunction();
// 局部作用域
function localScope() {
let localVar = "I'm local!";
console.log(localVar); // 输出 "I'm local!"
}
localScope();
// 函数内部声明的函数
function outerFunction() {
let outerVar = "I'm outer!";
function innerFunction() {
console.log(outerVar); // 输出 "I'm outer!"
}
innerFunction();
}
outerFunction();
对象与数组
对象是JavaScript中最灵活的数据结构,可以存储各种类型的值。数组是对象的一个特殊类型,用于存储一组有序的值。
代码示例
// 对象
let person = {
name: "Alice",
age: 30,
sayHello: function() {
console.log(`Hello, I'm ${this.name} and I'm ${this.age} years old.`);
}
};
person.sayHello(); // 输出 "Hello, I'm Alice and I'm 30 years old."
// 数组
let fruits = ["apple", "banana", "cherry"];
console.log(fruits[1]); // 输出 "banana"
fruits.push("orange");
console.log(fruits.length); // 输出 4
常见JS面试题解析
面试题型概述
常见的JS面试题可以分为以下几类:
- 代码实现题:考察基本语法、算法和数据结构。
- 思维逻辑题:考察逻辑推理、问题解决能力和代码设计能力。
示例1:实现一个数组去重函数
function uniqueArray(arr) {
let result = [];
let seen = new Set();
for (let i = 0; i < arr.length; i++) {
if (!seen.has(arr[i])) {
seen.add(arr[i]);
result.push(arr[i]);
}
}
return result;
}
console.log(uniqueArray([1, 2, 2, 3, 4, 4, 5])); // 输出 [1, 2, 3, 4, 5]
示例2:实现一个冒泡排序算法
function bubbleSort(arr) {
let len = arr.length;
for (let i = 0; i < len - 1; i++) {
for (let j = 0; j < len - 1 - i; j++) {
if (arr[j] > arr[j + 1]) {
[arr[j], arr[j + 1]] = [arr[j + 1], arr[j]]; // 交换元素
}
}
}
return arr;
}
console.log(bubbleSort([5, 3, 8, 2, 9, 1])); // 输出 [1, 2, 3, 5, 8, 9]
思维逻辑题解析
示例1:找出一个数组中的最大值和最小值
function findMinMax(arr) {
if (arr.length === 0) {
return null;
}
let min = arr[0];
let max = arr[0];
for (let i = 1; i < arr.length; i++) {
if (arr[i] < min) {
min = arr[i];
}
if (arr[i] > max) {
max = arr[i];
}
}
return { min, max };
}
console.log(findMinMax([5, 3, 8, 2, 9, 1])); // 输出 { min: 1, max: 9 }
示例2:判断一个字符串是否为回文
function isPalindrome(str) {
let cleanedStr = str.replace(/[^a-zA-Z0-9]/g, '').toLowerCase();
let reversedStr = cleanedStr.split('').reverse().join('');
return cleanedStr === reversedStr;
}
console.log(isPalindrome("A man, a plan, a canal: Panama")); // 输出 true
JS高级特性详解
事件循环与异步编程
JavaScript的事件循环机制支持异步编程。异步编程可以提升程序的响应性和效率,常见的异步机制包括回调、Promise、Generator 和 Async/Await。
事件循环
事件循环是一个不断执行回调函数的循环,主要用于处理异步任务,如I/O操作、定时器等。
代码示例
console.log("Start");
setTimeout(() => {
console.log("setTimeout");
}, 0);
process.nextTick(() => {
console.log("nextTick");
});
console.log("End");
输出:
Start
End
nextTick
setTimeout
Promise
Promise是解决异步编程的一种方式,它提供了一种更清晰的回调链表示方法。
代码示例
function delay(ms) {
return new Promise((resolve) => {
setTimeout(() => {
resolve(`Resolved after ${ms} ms`);
}, ms);
});
}
delay(1000).then((value) => {
console.log(value); // 输出 "Resolved after 1000 ms"
});
ES6新特性介绍
ES6引入了许多新特性,如模块化、类、解构赋值等。
模块化
ES6引入了 import
和 export
关键字来实现模块化。
// 导出模块
export const PI = 3.14159;
export function square(x) {
return x * x;
}
// 导入模块
import { PI, square } from './math.js';
console.log(PI); // 输出 3.14159
console.log(square(5)); // 输出 25
类
ES6引入了类的语法,使得面向对象编程更加清晰。
class Rectangle {
constructor(width, height) {
this.width = width;
this.height = height;
}
area() {
return this.width * this.height;
}
}
let rect = new Rectangle(4, 5);
console.log(rect.area()); // 输出 20
高阶函数与闭包
高阶函数是一种可以接受函数作为参数或返回函数的函数。闭包是一种可以访问自由变量的函数。
代码示例
// 高阶函数
function add(n) {
return function(m) {
return n + m;
};
}
let addThree = add(3);
console.log(addThree(4)); // 输出 7
// 闭包
function createCounter() {
let count = 0;
return function() {
return ++count;
};
}
let counter = createCounter();
console.log(counter()); // 输出 1
console.log(counter()); // 输出 2
常见JS错误与调试方法
错误类型汇总
JavaScript中的错误主要分为以下几类:
- SyntaxError:语法错误。
- ReferenceError:引用错误,通常是因为引用了未声明的变量。
- TypeError:类型错误,通常是因为对对象的不正确使用。
- RangeError:范围错误,通常出现在数值超出有效范围时。
- EvalError:EvalError,与eval函数相关。
- URIError:URIError,与URI编码相关。
示例1:未声明的变量
let a = 10;
console.log(b); // 报错 ReferenceError: b is not defined
示例2:类型错误
let a = 10;
a += "20"; // 报错 TypeError: unsupported operand type(s) for +=: 'number' and 'str'
调试工具使用
Chrome DevTools 是一个非常强大的调试工具,可以用来查看和修改HTML、CSS、JavaScript代码,以及进行网络请求分析等。
使用方法
- Console:查看控制台输出,可以输入JavaScript代码执行。
- Sources:查看和修改JavaScript代码,设置断点。
- Elements:查看和修改HTML元素。
- Network:查看和分析网络请求。
示例代码调试
function add(a, b) {
return a + b;
}
let result = add(2, '3');
console.log(result); // 控制台输出 "23"
在Sources面板中,可以设置断点来逐步调试代码。
大厂面试经验分享 面试流程解析面试流程通常包括简历筛选、在线笔试、技术面试、HR面试和最终Offer阶段。技术面试是最重要的环节之一,通常分为电话面试、远程面试和现场面试。
技术面试
- 项目介绍:介绍自己的项目经验,强调解决的问题和实现的功能。
- 基础题目:考察JavaScript基础知识,如变量、函数、DOM操作等。
- 算法题目:考察算法和数据结构,如排序、查找等。
- 设计题目:考察设计模式和系统架构,如单例模式、MVC模式等。
- 系统设计:考察系统设计能力,如设计一个简单的Web应用或API接口。
面试前要保持积极的心态,相信自己的实力,同时也要准备充分。面试过程中要保持冷静,不要紧张,尽量发挥自己的最佳状态。
面试心态调整技巧
- 准备充分:熟悉面试流程,多做练习题。
- 模拟面试:找朋友或同学模拟面试,提高自信心。
- 心理暗示:告诉自己“我可以做到”,增强自信心。
- 保持冷静:面试过程中保持冷静,不要紧张,尽量发挥自己的最佳状态。
面试时要注意以下几点:
- 清晰表达:用清晰、简洁的语言回答问题。
- 逻辑清晰:回答问题时要有逻辑,分步骤描述。
- 现场代码:现场写代码时要注意代码规范,尽量不要有语法错误。
- 提问环节:在面试结束后,可以向面试官提问,展现自己的积极性。
- 反思总结:面试结束后要总结经验,反思不足,为下一次面试做准备。
实际的面试过程中,建议多做模拟面试和实战练习。可以找朋友或同学一起模拟面试场景,互相提问和回答问题。此外,可以参考一些在线编程平台,如慕课网,多做一些练习题,提高自己的编程能力和面试技巧。
示例代码练习
// 示例:实现一个链表
class ListNode {
constructor(val, next = null) {
this.val = val;
this.next = next;
}
}
function addTwoNumbers(l1, l2) {
let dummyHead = new ListNode(0);
let p = l1, q = l2;
let carry = 0;
let curr = dummyHead;
while (p !== null || q !== null) {
let x = (p !== null) ? p.val : 0;
let y = (q !== null) ? q.val : 0;
let sum = carry + x + y;
carry = Math.floor(sum / 10);
curr.next = new ListNode(sum % 10);
curr = curr.next;
if (p !== null) p = p.next;
if (q !== null) q = q.next;
}
if (carry > 0) {
curr.next = new ListNode(carry);
}
return dummyHead.next;
}
通过不断练习和反馈,逐步提高自己的编程水平和面试技巧。
JS面试真题实战演练 真题解析与答案示例1:实现一个深拷贝函数
function deepCopy(obj) {
if (typeof obj !== 'object') {
return obj;
}
let copy = Array.isArray(obj) ? [] : {};
for (let key in obj) {
if (obj.hasOwnProperty(key)) {
copy[key] = deepCopy(obj[key]);
}
}
return copy;
}
let original = { a: 1, b: { c: 3, d: 4 }, e: [5, 6, 7] };
let copy = deepCopy(original);
console.log(original === copy); // 输出 false
console.log(original.b === copy.b); // 输出 false
console.log(original.e === copy.e); // 输出 false
示例2:实现一个扁平化数组函数
function flattenArray(arr) {
return arr.reduce((acc, val) => {
return acc.concat(Array.isArray(val) ? flattenArray(val) : val);
}, []);
}
let nestedArray = [1, [2, [3, 4], 5], 6];
console.log(flattenArray(nestedArray)); // 输出 [1, 2, 3, 4, 5, 6]
实战练习与反馈
实际的面试过程中,建议多做模拟面试和实战练习。可以找朋友或同学一起模拟面试场景,互相提问和回答问题。此外,可以参考一些在线编程平台,如慕课网,多做一些练习题,提高自己的编程能力和面试技巧。
示例代码练习
// 示例:实现一个链表
class ListNode {
constructor(val, next = null) {
this.val = val;
this.next = next;
}
}
function addTwoNumbers(l1, l2) {
let dummyHead = new ListNode(0);
let p = l1, q = l2;
let carry = 0;
let curr = dummyHead;
while (p !== null || q !== null) {
let x = (p !== null) ? p.val : 0;
let y = (q !== null) ? q.val : 0;
let sum = carry + x + y;
carry = Math.floor(sum / 10);
curr.next = new ListNode(sum % 10);
curr = curr.next;
if (p !== null) p = p.next;
if (q !== null) q = q.next;
}
if (carry > 0) {
curr.next = new ListNode(carry);
}
return dummyHead.next;
}
通过不断练习和反馈,逐步提高自己的编程水平和面试技巧。