手记

JS大厂面试真题解析与实战指南

概述

本文详细介绍了JS基础知识和高级特性,并提供了一系列的代码示例和面试题解析,帮助读者巩固JS技能。此外,文章还分享了大厂面试中的常见题型和实战演练,特别是涵盖了js大厂面试真题的相关内容。

JS基础知识回顾
变量与数据类型

在JavaScript中,变量是用来存储数据的容器。JavaScript是一种动态类型语言,这意味着你不需要在声明变量时指定其类型,但是JavaScript的变量主要包含以下几种类型:

  1. 基本数据类型:包括 NumberStringBooleannullundefinedSymbol(ES6新增的类型)。
  2. 引用数据类型:主要是 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. 代码实现题:考察基本语法、算法和数据结构。
  2. 思维逻辑题:考察逻辑推理、问题解决能力和代码设计能力。
代码实现题解析

示例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引入了 importexport 关键字来实现模块化。

// 导出模块
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中的错误主要分为以下几类:

  1. SyntaxError:语法错误。
  2. ReferenceError:引用错误,通常是因为引用了未声明的变量。
  3. TypeError:类型错误,通常是因为对对象的不正确使用。
  4. RangeError:范围错误,通常出现在数值超出有效范围时。
  5. EvalError:EvalError,与eval函数相关。
  6. 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代码,以及进行网络请求分析等。

使用方法

  1. Console:查看控制台输出,可以输入JavaScript代码执行。
  2. Sources:查看和修改JavaScript代码,设置断点。
  3. Elements:查看和修改HTML元素。
  4. Network:查看和分析网络请求。

示例代码调试

function add(a, b) {
    return a + b;
}

let result = add(2, '3');
console.log(result); // 控制台输出 "23"

在Sources面板中,可以设置断点来逐步调试代码。

大厂面试经验分享
面试流程解析

面试流程通常包括简历筛选、在线笔试、技术面试、HR面试和最终Offer阶段。技术面试是最重要的环节之一,通常分为电话面试、远程面试和现场面试。

技术面试

  1. 项目介绍:介绍自己的项目经验,强调解决的问题和实现的功能。
  2. 基础题目:考察JavaScript基础知识,如变量、函数、DOM操作等。
  3. 算法题目:考察算法和数据结构,如排序、查找等。
  4. 设计题目:考察设计模式和系统架构,如单例模式、MVC模式等。
  5. 系统设计:考察系统设计能力,如设计一个简单的Web应用或API接口。
面试心态调整

面试前要保持积极的心态,相信自己的实力,同时也要准备充分。面试过程中要保持冷静,不要紧张,尽量发挥自己的最佳状态。

面试心态调整技巧

  1. 准备充分:熟悉面试流程,多做练习题。
  2. 模拟面试:找朋友或同学模拟面试,提高自信心。
  3. 心理暗示:告诉自己“我可以做到”,增强自信心。
  4. 保持冷静:面试过程中保持冷静,不要紧张,尽量发挥自己的最佳状态。
面试技巧总结

面试时要注意以下几点:

  1. 清晰表达:用清晰、简洁的语言回答问题。
  2. 逻辑清晰:回答问题时要有逻辑,分步骤描述。
  3. 现场代码:现场写代码时要注意代码规范,尽量不要有语法错误。
  4. 提问环节:在面试结束后,可以向面试官提问,展现自己的积极性。
  5. 反思总结:面试结束后要总结经验,反思不足,为下一次面试做准备。
实战练习与反馈

实际的面试过程中,建议多做模拟面试和实战练习。可以找朋友或同学一起模拟面试场景,互相提问和回答问题。此外,可以参考一些在线编程平台,如慕课网,多做一些练习题,提高自己的编程能力和面试技巧。

示例代码练习

// 示例:实现一个链表
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;
}

通过不断练习和反馈,逐步提高自己的编程水平和面试技巧。

0人推荐
随时随地看视频
慕课网APP