本文深入探讨了JavaScript的基础知识,包括数据类型、运算符、流程控制语句和面向对象编程等内容。文章还提供了常见算法与数据结构的示例,并详细解析了大厂面试中的JavaScript真题,帮助读者更好地准备面试中的各种问题和挑战。
JavaScript支持多种内置数据类型,主要包括原始类型和引用类型。
原始类型:
number
:用于表示数字,可以是整数或小数。string
:用于表示文本,由一系列字符组成。boolean
:表示逻辑值,只能是 true
或 false
。null
:表示空值,通常用于表示不存在的对象。undefined
:表示未赋值的变量或函数参数。symbol
(ES6新增):表示一个唯一的标识符。bigint
(ES10新增):用于表示任意精度的整数。引用类型:
object
:可以包含多个键值对,用于表示复杂的数据结构。function
:表示一段可执行的代码块。示例代码:
// 定义不同类型的变量 let numberVar = 42; let stringVar = "Hello, world!"; let booleanVar = true; let nullVar = null; let undefinedVar; let symbolVar = Symbol("unique"); let bigIntVar = 1234567890123456789012345678901234567890n; let objectVar = {}; let functionVar = function() {}; console.log(numberVar, stringVar, booleanVar, nullVar, undefinedVar, symbolVar, bigIntVar, objectVar, functionVar);
JavaScript中的运算符可以分为算术运算符、比较运算符、逻辑运算符等。
算术运算符:
+
:加法-
:减法*
:乘法/
:除法%
:取模(求余数)比较运算符:
==
:等于===
:严格等于(比较值和类型)!=
:不等于!==
:严格不等于(比较值和类型)>
:大于<
:小于>=
:大于等于<=
:小于等于逻辑运算符:
&&
:逻辑与||
:逻辑或!
:逻辑非示例代码:
let a = 10; let b = 5; console.log(a + b); // 15 console.log(a - b); // 5 console.log(a * b); // 50 console.log(a / b); // 2 console.log(a % b); // 0 console.log(a == b); // false console.log(a === b); // false console.log(a != b); // true console.log(a !== b); // true console.log(a > b); // true console.log(a < b); // false console.log(a >= b); // true console.log(a <= b); // false let c = true; let d = false; console.log(c && d); // false console.log(c || d); // true console.log(!c); // false
JavaScript中的流程控制语句包括条件语句和循环语句。
条件语句:
if
语句:根据指定条件执行代码块。if...else
语句:如果条件为真执行一个代码块,否则执行另一个代码块。if...else if...else
语句:多个条件取其一。switch
语句:根据不同的条件选择执行不同的代码块。示例代码:
let age = 25; if (age < 18) { console.log("未成年"); } else if (age >= 18 && age < 60) { console.log("成年"); } else { console.log("老年"); } let day = "Monday"; switch (day) { case "Monday": console.log("今天是星期一"); break; case "Tuesday": console.log("今天是星期二"); break; default: console.log("其他日子"); }
循环语句:
for
循环:适用于已知循环次数的情况。while
循环:适用于不知道循环次数但知道终止条件的情况。do...while
循环:先执行一次,再判断循环条件。示例代码:
// for 循环 for (let i = 0; i < 5; i++) { console.log(i); } // while 循环 let j = 0; while (j < 5) { console.log(j); j++; } // do...while 循环 let k = 0; do { console.log(k); k++; } while (k < 5);
函数是JavaScript中可重用的代码块。JavaScript支持函数声明和函数表达式两种方式。
函数声明:
function greet(name) { console.log(`Hello, ${name}!`); } greet("Alice"); // 输出 "Hello, Alice!"
函数表达式:
const add = function(x, y) { return x + y; }; console.log(add(2, 3)); // 输出 5
作用域分为函数作用域和块作用域(ES6新特性)。
函数作用域:
function outer() { let outerVar = "outer"; function inner() { console.log(outerVar); // 输出 "outer" } inner(); console.log(innerVar); // 输出 undefined } outer(); console.log(outerVar); // 输出 undefined
块作用域:
if (true) { let blockVar = "block"; console.log(blockVar); // 输出 "block" } console.log(blockVar); // 输出 undefined
在JavaScript中,面向对象编程(OOP)主要通过构造函数和原型来实现。下面将详细介绍构造函数、原型、继承与封装以及多态性与抽象类。
构造函数:
构造函数是一种特殊的函数,用于创建和初始化对象。
示例代码:
function Person(name, age) { this.name = name; this.age = age; this.describe = function() { console.log(`Name: ${this.name}, Age: ${this.age}`); }; } let person1 = new Person("Alice", 25); person1.describe(); // 输出 "Name: Alice, Age: 25"
原型(Prototype):
原型用于定义所有实例共享的方法。
示例代码:
function Animal(name) { this.name = name; } Animal.prototype.describe = function() { console.log(`Name: ${this.name}`); }; let animal1 = new Animal("Dog"); animal1.describe(); // 输出 "Name: Dog"
继承与封装:
继承可以让一个对象继承另一个对象的属性和方法。封装则是将数据和操作数据的方法封装在一起。
示例代码:
function Vehicle(model) { this.model = model; } Vehicle.prototype.drive = function() { console.log(`${this.model} is driving`); }; function Car(model, seats) { Vehicle.call(this, model); this.seats = seats; } Car.prototype = Object.create(Vehicle.prototype); Car.prototype.constructor = Car; Car.prototype.drive = function() { console.log(`${this.model} is driving with ${this.seats} seats`); }; let car = new Car("Toyota", 4); car.drive(); // 输出 "Toyota is driving with 4 seats"
多态性与抽象类:
多态意味着同一个方法可以有多个实现,抽象类是一种无法直接实例化的类,通常用于定义其他类的基类。
示例代码:
function Shape() { throw new Error("Cannot call abstract method"); } Shape.prototype.area = function() { throw new Error("Cannot call abstract method"); }; function Rectangle(width, height) { Shape.call(this); this.width = width; this.height = height; } Rectangle.prototype = Object.create(Shape.prototype); Rectangle.prototype.constructor = Rectangle; Rectangle.prototype.area = function() { return this.width * this.height; }; function Circle(radius) { Shape.call(this); this.radius = radius; } Circle.prototype = Object.create(Shape.prototype); Circle.prototype.constructor = Circle; Circle.prototype.area = function() { return Math.PI * this.radius * this.radius; }; let rectangle = new Rectangle(4, 5); console.log(rectangle.area()); // 输出 20 let circle = new Circle(3); console.log(circle.area()); // 输出 28.27...
数组是最常用的数据结构之一,JavaScript提供了多种操作数组的方法。
示例代码:
let arr = [1, 2, 3, 4, 5]; // 遍历数组 for (let i = 0; i < arr.length; i++) { console.log(arr[i]); } // 使用forEach方法 arr.forEach(item => { console.log(item); }); // 使用map方法 let newArr = arr.map(item => item * 2); console.log(newArr); // 输出 [2, 4, 6, 8, 10]
常见的排序算法包括冒泡排序、选择排序、快速排序等。查找算法包括线性查找和二分查找。
示例代码 - 冒泡排序:
function bubbleSort(arr) { let len = arr.length; for (let i = 0; i < len - 1; i++) { for (let j = 0; j < len - i - 1; j++) { if (arr[j] > arr[j + 1]) { let temp = arr[j]; arr[j] = arr[j + 1]; arr[j + 1] = temp; } } } return arr; } console.log(bubbleSort([64, 34, 25, 12, 22, 11, 90])); // 输出 [11, 12, 22, 25, 34, 64, 90]
示例代码 - 二分查找:
function binarySearch(arr, target) { let left = 0; let right = arr.length - 1; while (left <= right) { let mid = Math.floor((left + right) / 2); if (arr[mid] === target) { return mid; } else if (arr[mid] < target) { left = mid + 1; } else { right = mid - 1; } } return -1; } let arr = [2, 3, 4, 10, 40]; console.log(binarySearch(arr, 10)); // 输出 3
栈:后进先出(LIFO)的数据结构。
队列:先进先出(FIFO)的数据结构。
链表:由一系列节点组成,每个节点包含数据和指向下一个节点的引用。
示例代码 - 栈:
class Stack { constructor() { this.items = []; } push(item) { this.items.push(item); } pop() { return this.items.pop(); } peek() { return this.items[this.items.length - 1]; } isEmpty() { return this.items.length === 0; } size() { return this.items.length; } } let stack = new Stack(); stack.push(1); stack.push(2); console.log(stack.peek()); // 输出 2 console.log(stack.pop()); // 输出 2 console.log(stack.isEmpty()); // 输出 false
示例代码 - 队列:
class Queue { constructor() { this.items = []; } enqueue(item) { this.items.push(item); } dequeue() { return this.items.shift(); } peek() { return this.items[0]; } isEmpty() { return this.items.length === 0; } size() { return this.items.length; } } let queue = new Queue(); queue.enqueue(1); queue.enqueue(2); console.log(queue.peek()); // 输出 1 console.log(queue.dequeue()); // 输出 1 console.log(queue.isEmpty()); // 输出 false
示例代码 - 链表:
class Node { constructor(data) { this.data = data; this.next = null; } } class LinkedList { constructor() { this.head = null; } append(data) { if (!this.head) { this.head = new Node(data); } else { let current = this.head; while (current.next !== null) { current = current.next; } current.next = new Node(data); } } display() { let current = this.head; let result = ''; while (current !== null) { result += current.data + ' '; current = current.next; } console.log(result); } } let list = new LinkedList(); list.append(1); list.append(2); list.append(3); list.display(); // 输出 1 2 3
1. JavaScript中的数据类型有哪些?
number
、string
、boolean
、null
、undefined
、symbol
、bigint
。console.log(typeof 42); // 输出 "number" console.log(typeof "Hello"); // 输出 "string" console.log(typeof true); // 输出 "boolean" console.log(typeof null); // 输出 "object" console.log(typeof undefined);// 输出 "undefined" console.log(typeof Symbol()); // 输出 "symbol" console.log(typeof 1234567890123456789012345678901234567890n); // 输出 "bigint"
2. 什么是闭包?
function createCounter() { let count = 0; return function() { count++; return count; }; }
let counter = createCounter();
console.log(counter()); // 输出 1
console.log(counter()); // 输出 2
**3. 什么是原型链?** - 答:原型链是指通过对象的原型链接起来的一系列对象,用于实现继承和属性查找。 - 示例代码: ```javascript function Person(name) { this.name = name; } Person.prototype.describe = function() { console.log(`Name: ${this.name}`); }; let person = new Person("Alice"); person.describe(); // 输出 "Name: Alice"
1. 熟悉基础知识:
2. 刷题练习:
3. 阅读和理解代码:
4. 准备面试常见问题:
1. 旋转数组
给定一个数组nums
,将其旋转k
次。
示例:
function rotateArray(nums, k) { let n = nums.length; k = k % n; let temp = nums.slice(-k); nums.splice(0, n - k); nums.unshift(...temp); return nums; } console.log(rotateArray([1, 2, 3, 4, 5, 6, 7], 3)); // 输出 [5, 6, 7, 1, 2, 3, 4]
2. 两数之和
给定一个整数数组nums
和一个目标值target
,找到和为target
的两个数的索引。
示例:
function twoSum(nums, target) { let map = new Map(); for (let i = 0; i < nums.length; i++) { let complement = target - nums[i]; if (map.has(complement)) { return [map.get(complement), i]; } map.set(nums[i], i); } return []; } console.log(twoSum([2, 7, 11, 15], 9)); // 输出 [0, 1]
大厂面试通常包括以下几个部分:
1. 实现深拷贝
深拷贝需要考虑对象和数组的嵌套情况。
示例代码:
function deepClone(obj) { if (typeof obj !== "object" || obj === null) { return obj; } let cloneObj = Array.isArray(obj) ? [] : {}; for (let key in obj) { if (obj.hasOwnProperty(key)) { cloneObj[key] = deepClone(obj[key]); } } return cloneObj; } console.log(deepClone({ a: 1, b: { c: 2 } })); // 输出 { a: 1, b: { c: 2 } }
2. 实现一个简单的Promise
Promise是一种异步处理的方式,可以替代回调函数。
示例代码:
class MyPromise { constructor(executor) { let self = this; self.status = "pending"; self.reason = null; self.value = null; self.onFulfilledCallbacks = []; self.onRejectedCallbacks = []; function resolve(value) { if (self.status === "pending") { self.status = "fulfilled"; self.value = value; setTimeout(() => { self.onFulfilledCallbacks.forEach(fn => fn(self.value)); }); } } function reject(reason) { if (self.status === "pending") { self.status = "rejected"; self.reason = reason; setTimeout(() => { self.onRejectedCallbacks.forEach(fn => fn(self.reason)); }); } } try { executor(resolve, reject); } catch (e) { reject(e); } } then(onFulfilled, onRejected) { onFulfilled = typeof onFulfilled === "function" ? onFulfilled : (value) => value; onRejected = typeof onRejected === "function" ? onRejected : (reason) => { throw reason; }; if (this.status === "fulfilled") { setTimeout(() => { onFulfilled(this.value); }); } else if (this.status === "rejected") { setTimeout(() => { onRejected(this.reason); }); } else { this.onFulfilledCallbacks.push(onFulfilled); this.onRejectedCallbacks.push(onRejected); } return this; } } let promise = new MyPromise((resolve, reject) => { setTimeout(() => { resolve("Success"); }, 1000); }); promise.then((value) => { console.log(value); // 输出 "Success" });
1. 处理异步代码
使用Promise或async/await可以更好地组织和控制异步代码。
示例代码:
async function fetchData() { try { let response = await fetch('https://api.example.com/data'); let data = await response.json(); console.log(data); } catch (error) { console.error(error); } } fetchData();
2. 性能优化
性能优化包括代码优化、数据库优化、缓存机制等。
示例代码:
function optimizeLoop(arr) { let sum = 0; for (let i = 0; i < arr.length; i++) { let item = arr[i]; if (item > 10) { sum += item; } } return sum; } console.log(optimizeLoop([1, 2, 11, 20, 5])); // 输出 31
1. 写面试总结:
2. 反馈给面试官:
1. 复盘面试经历:
2. 提升自身能力:
1. 了解公司文化:
2. 调整心态:
3. 做好职业规划:
通过以上内容的学习和练习,你将能够更好地准备和应对JavaScript相关的面试挑战。祝你面试成功!