还是书接上篇文章,分享一些手写的题目,对于特别常见的题目最好能“背”下来,在面试的时候不需要再进行推导分析直接一把梭,这里的题目有一些是自己遇到的,也有收集到的,后续会整理分享一些其他的信息,希望对你能有所帮助

前端面试必须掌握的手写题:基础篇

前端面试必须掌握的手写题:场景篇

前端面试必须掌握的手写题:进阶篇

0.1+0.2问题

转成整数处理

 1function accAdd(arg1,arg2){
 2  var r1,r2,m;
 3  try{
 4    r1=arg1.toString().split(".")[1].length
 5  }catch(e){
 6    r1=0
 7  }
 8  try{
 9    r2=arg2.toString().split(".")[1].length
10  }catch(e){
11    r2=0
12  }
13  m=Math.pow(10,Math.max(r1,r2));
14  return (arg1*m+arg2*m)/m;
15}
16var result = accAdd(0.1,0.2)
17console.log(result) 

大数相加解决[415.字符串相加]

传两个字符串进来,返回一个字符串

 1var addStrings = function (num1, num2) {
 2    let result = '';
 3    let i = num1.length - 1, j = num2.length - 1, carry = 0;
 4    while (i >= 0 || j >= 0) {
 5        let n1 = i >= 0 ? +num1[i] : 0;
 6        let n2 = j >= 0 ? +num2[j] : 0;
 7        const temp = n1 + n2 + carry;
 8        carry = temp / 10 | 0;
 9        result = `${temp % 10}${result}`;
10        i--; j--;
11    }
12    if (carry === 1) result = `1${result}`;
13    return result;
14};

大数相乘[43.字符串相乘]

传两个字符串进来,返回一个字符串

  • 转成数字相加的问题
  • 注意处理全零字符串的情况
 1var multiply = function (num1, num2) {
 2    let result = '0';
 3    let i = num1.length - 1;
 4    while (i >= 0) {
 5        let subfixZero = new Array(num1.length - 1 - i).fill('0').join('');
 6        let sumCount = +num1[i];
 7        let tempSum = '0';
 8        while (sumCount > 0) {
 9            tempSum = bigSum(tempSum, num2);
10            sumCount--;
11        }
12        tempSum = `${tempSum}${subfixZero}`;
13        result = bigSum(result, tempSum);
14        i--;
15    }
16    
17    for (let i = 0; i < result.length; i++) {
18        if (result[i] !== '0') {
19            return result.slice(i);
20        }
21    }
22    return '0';
23
24    function bigSum(n1, n2) {
25        let result = '';
26        let i = n1.length - 1, j = n2.length - 1, curry = 0;
27        while (i >= 0 || j >= 0) {
28            let l1 = i >= 0 ? +n1[i] : 0;
29            let l2 = j >= 0 ? + n2[j] : 0;
30            let sum = l1 + l2 + curry;
31            curry = sum / 10 | 0;
32            result = `${sum % 10}${result}`;
33            i--; j--;
34        }
35        if (curry === 1) result = `1${result}`;
36        return result;
37    }
38};

数组乱序输出

Math.random输出的结果是0-1内的小数,可以直接通过length映射

 1const randomIndex = Math.round(Math.random()*(array.length - 1 -i) + 1);

数组去重复(7种方法)

关键点是NaN怎么判断,对NaN进行去重,这个题目的另一个考察点是对API的灵活运用,虽然很多方法不可能用在实际的场景中,但是who care,面试官只会觉得你懂得好多~

  • 1.利用Set()+Array.from()

    • 方式对NaN和undefined类型去重也是有效的,是因为NaN和undefined都可以被存储在Set中, NaN之间被视为相同的值
  • 2.利用两层循环+数组的splice方法

    • 此方法对NaN是无法进行去重的,因为进行比较时NaN !== NaN
  • 3.利用数组的indexOf方法

    • 新建一个空数组,遍历需要去重的数组,将数组元素存入新数组中,存放前判断数组中是否已经含有当前元素,没有则存入。此方法也无法对NaN去重
    • indexOf() 方法:返回调用它的String对象中第一次出现的指定值的索引
  • 4.利用数组的includes方法

    • 此方法逻辑与indexOf方法去重异曲同工,只是用includes方法来判断是否包含重复元素。
  • 5.利用数组的filter()+indexOf()

    • 输出结果中不包含NaN,是因为indexOf()无法对NaN进行判断
  • 6.利用Map()

    • 使用Map()也可对NaN去重,原因是Map进行判断时认为NaN是与NaN相等的
  • 7.利用对象

    • 和Map()是差不多的,主要是利用了对象的属性名不可重复这一特性。

数组扁平化flatten(6种方法)

  • 递归
  • reduce
  • 扩展运算符
  • toString,split
  • es6 flat
  • 正则和json,json.stringify
 1function flatten(arr) {
 2  let result = [];
 3  for (let i = 0; i < arr.length; i++) {
 4    if (Array.isArray(arr[i])) {
 5      result = result.concat(flatten(arr[i]));
 6    } else {
 7      result.push(arr[i]);
 8    }
 9  }
10  return result;
11}
12
13function flatten(arr) {
14  return arr.reduce((p, c) => {
15    return p.concat(Array.isArray(c) ? flatten(c) : c);
16  }, [])
17}

🔥对象扁平化flatObj

多次遇到,建议背诵

 1
 2var entryObj = {
 3    a: {
 4        b: {
 5            c: {
 6                    dd: 'abcdd'
 7            }
 8        },
 9        d: {
10            xx: 'adxx'
11        },
12        e: 'ae'
13    }
14}
15
16
17var outputObj = {
18	'a.b.c.dd': 'abcdd',
19	'a.d.xx': 'adxx',
20	'a.e': 'ae'
21}
22
23function flat(obj, path = '', res = {}, isArray) {
24  for (let [k, v] of Object.entries(obj)) {
25    if (Array.isArray(v)) {
26      let _k = isArray ? `${path}[${k}]` : `${path}${k}`;
27      flat(v, _k, res, true);
28    } else if (typeof v === 'object') {
29      let _k = isArray ? `${path}[${k}].` : `${path}${k}.`;
30      flat(v, _k, res, false);
31    } else {
32      let _k = isArray ? `${path}[${k}]` : `${path}${k}`;
33      res[_k] = v;
34    }
35  }
36  return res;
37}
38
39console.log(flat({ a: { aa: [{ aa1: 1 }] } }))

数字千分位分割

注意可能有小数

 1function format(number) {
 2  number = number.toString();
 3  let decimals = '';
 4  number.includes('.') ? decimals = number.split('.')[1] : decimals;
 5
 6  let len = number.length;
 7  if (len < 3) {
 8    return number;
 9  } else {
10    let temp = '';
11    let remainder = len % 3;
12    decimals ? temp = '.' + decimals : temp;
13    if (remainder > 0) {
14      return number.slice(0, remainder) + ',' + number.slice(remainder, len).match(/\d{3}/g).join(',') + temp;
15    } else {
16      return number.slice(0, len).match(/d{3}/g).join(',') + temp;
17    }
18  }
19}

js下划线转驼峰处理「快手」

正则法

 1function camelCase(str) {
 2  return str.replace(/_([a-z])/g, function(match, group1) {
 3    return group1.toUpperCase();
 4  });
 5}
 6
 7console.log(camelCase("some_string"));  

补充

 1function camelCase(str) {
 2  return str.replace(/([-_])([a-z])/g, function(match, group1, group2) {
 3    return group2.toUpperCase();
 4  });
 5}
 6
 7console.log(camelCase("some-string_with-underscores"));

Hex转RGB的方法

 1function hexToRgb(val) {
 2  
 3  
 4  var reg = /^#([0-9a-fA-f]{3}|[0-9a-fA-f]{6})$/;
 5  
 6  var color = val.toLowerCase();
 7  var result = '';
 8  if (reg.test(color)) {
 9    
10    if (color.length === 4) {
11      var colorNew = '#';
12      for (var i = 1; i < 4; i += 1) {
13        colorNew += color.slice(i, i + 1).concat(color.slice(i, i + 1));
14      }
15      color = colorNew;
16    }
17    
18    var colorChange = [];
19    for (var i = 1; i < 7; i += 2) {
20      colorChange.push(parseInt('0x' + color.slice(i, i + 2)));
21    }
22    result = 'rgb(' + colorChange.join(',') + ')';
23    return { rgb: result, r: colorChange[0], g: colorChange[1], b: colorChange[2] };
24  } else {
25    result = '无效';
26    return { rgb: result };
27  }
28}

实现模版字符串解析

 1var template = `
 2<div>
 3    <% if(name){ %>
 4        <span>%= name =%</span>
 5    <% } %>
 6    %= age =%
 7<div>`
 8let str = rander(template, {name: '小明', age: 18})
 9
 1function parseTemplateString (templateString, data) {
 2  
 3  const regex = /${(.*?)}/g;
 4  
 5  const parsedString = templateString.replace(regex, (match, key) => {
 6    
 7    return eval(`data.${key}`);
 8  });
 9  return parsedString;
10}

🔥数组转树形结构的三种方法

递归解法非常好理解,代码量也很少,题目出现概率很高

 1{
 2  "city": [
 3    { "id": 12, "parent_id": 1, "name": "朝阳区" },
 4    { "id": 241, "parent_id": 24, "name": "田林街道" },
 5    { "id": 31, "parent_id": 3, "name": "广州市" },
 6    { "id": 13, "parent_id": 1, "name": "昌平区" },
 7    { "id": 2421, "parent_id": 242, "name": "上海科技绿洲" },
 8    { "id": 21, "parent_id": 2, "name": "静安区" },
 9    { "id": 242, "parent_id": 24, "name": "漕河泾街道" },
10    { "id": 22, "parent_id": 2, "name": "黄浦区" },
11    { "id": 11, "parent_id": 1, "name": "顺义区" },
12    { "id": 2, "parent_id": 0, "name": "上海市" },
13    { "id": 24, "parent_id": 2, "name": "徐汇区" },
14    { "id": 1, "parent_id": 0, "name": "北京市" },
15    { "id": 2422, "parent_id": 242, "name": "漕河泾开发区" },
16    { "id": 32, "parent_id": 3, "name": "深圳市" },
17    { "id": 33, "parent_id": 3, "name": "东莞市" },
18    { "id": 3, "parent_id": 0, "name": "广东省" }
19  ]
20}
 1function arrayToTreeV3(list, root) {
 2  return list
 3    .filter(item => item.parent_id === root)
 4    .map(item => ({...item, children: arrayToTreeV3(list, item.id)}))
 5}

获取URL中的参数

这里主要还是正则表达式的设计

  • /?&/igm,前面是?或者&,任意字符直到遇到=,使用非贪婪模式,等号后面是非&符号的任意字符,然后去匹配就好了
  • 理论上可以用matchAll,然后用迭代器去处理
 1function name(url) {
 2    const _url = url || window.location.href;
 3    const _urlParams = _url.match(/[?&](.+?=[^&]+)/igm);
 4    return _urlParams ? _urlParams.reduce((a,b) => {
 5        const value = b.slice(1).split('=');
 6        a[value[0]] = value[1];
 7        return a;
 8    }, {}) : {} 
 9    
10}

小结

场景题目其实很多,没办法去枚举,但是这里标记出来的是相对高频的题目,对上面题目有些好的解法的也可以在评论区分享

个人笔记记录 2021 ~ 2025