2020 面试要记

1.防抖与节流
防抖是什么? 触发高频事件后n秒内函数只会执行一次,如果n秒内高频事件再次被触发,则重新计算时间
实现方式:每次触发事件时设置一个延迟调用方法,并且取消之前的延时调用方法
缺点:如果事件在规定的时间间隔内被不断的触发,则调用方法会被不断的延迟方法

//防抖debounce代码:
function debounce(fn) {
    let timeout = null; // 创建一个标记用来存放定时器的返回值
    return function () {
        // 每当用户输入的时候把前一个 setTimeout clear 掉
        clearTimeout(timeout); 
        // 然后又创建一个新的 setTimeout, 这样就能保证interval 间隔内如果时间持续触发,就不会执行 fn 函数
        timeout = setTimeout(() => {
            fn.apply(this, arguments);
        }, 500);
    };
}
// 处理函数
function handle() {
    console.log(Math.random());
}
// 滚动事件
window.addEventListener('scroll', debounce(handle));

节流是什么 ?高频事件触发,但在n秒内只会执行一次,所以节流会稀释函数的执行频率
实现方式:每次触发事件时,如果当前有等待执行的延时函数,则直接return

//节流throttle代码:
function throttle(fn) {
    let canRun = true; // 通过闭包保存一个标记
    return function () {
         // 在函数开头判断标记是否为true,不为true则return
        if (!canRun) return;
         // 立即设置为false
        canRun = false;
        // 将外部传入的函数的执行放在setTimeout中
        setTimeout(() => { 
        // 最后在setTimeout执行完毕后再把标记设置为true(关键)表示可以执行下一次循环了。
        // 当定时器没有执行的时候标记永远是false,在开头被return掉
            fn.apply(this, arguments);
            canRun = true;
        }, 500);
    };
}

function sayHi(e) {
    console.log(e.target.innerWidth, e.target.innerHeight);
}
window.addEventListener('resize', throttle(sayHi));

总结:
函数防抖:将多次操作合并为一次操作进行。原理是维护一个计时器,规定在delay时间后触发函数,但是在delay时间内再次触发的话,就会取消之前的计时器而重新设置。这样一来,只有最后一次操作能被触发。

函数节流:使得一定时间内只触发一次函数。原理是通过判断是否有延迟调用函数未执行。

区别: 函数节流不管事件触发有多频繁,都会保证在规定时间内一定会执行一次真正的事件处理函数,而函数防抖只是在最后一次事件后才触发一次函数。 比如在页面的无限加载场景下,我们需要用户在滚动页面时,每隔一段时间发一次 Ajax 请求,而不是在用户停下滚动页面操作时才去请求数据。这样的场景,就适合用节流技术来实现。

2.闭包
闭包:能够读取其他函数内部变量的函数。

 

JS数组去重 你知道几种?

法一 For嵌套for 使用splice去重更改原数组 正向遍历循环

遇到删掉 原数组递减1

{
    let arr = [1,1,'true','true',true,true,15,15,false,false, undefined,undefined, null,null, NaN, NaN,'NaN','NaN', 0, 0, 'a', 'a',{},{}];
    function distinct (arr) {
        for(let i = 0; i < arr.length; i++) {
            for(let j = i + 1; j < arr.length ; j++) {
                if(arr[i] === arr[j]) {
                    arr.splice(j, 1)
                    j--;
                }
            }
        }
    } 
    
    distinct(arr)
    console.log(arr) // [1, "true", true, 15, false, undefined, null, NaN, NaN, "NaN", 0, "a", {…}, {…}]
}
优点:该方法可以顾虑到重复的 String、Boolean、 Number、undefined、null,返回的是去重后的原数组。
缺点:不能过滤掉 NaN、Object

法二 For嵌套for 使用splice去重更改原数组 逆向遍历循环

逆向循环遍历 遇到重复的直接删掉

{
    let arr = [1,1,'true','true',true,true,15,15,false,false, undefined,undefined, null,null, NaN, NaN,'NaN','NaN', 0, 0, 'a', 'a',{},{}];
    function distinct (arr) {
        for(let i = arr.length; i > 0; i--) {
            for(let j = i - 1; j > -1 ; j--) {
                if(arr[i] === arr[j]) {
                    console.log(arr[j])
                    arr.splice(j, 1)
                }
            }
        }
    } 
    
    distinct(arr)
    console.log(arr) // [1, "true", true, 15, false, null, NaN, NaN, "NaN", 0, "a", {…}, {…}]
}
优点:该方法可以顾虑到重复的 String、Boolean、 Number、undefined、`null``,返回的是去重后的原数组。
缺点:不能过滤掉 NaN、Object

法三:includes去重 返回新数组

{
    let arr = [1,1,'true','true',true,true,15,15,false,false, undefined,undefined, null,null, NaN, NaN,'NaN','NaN', 0, 0, 'a', 'a',{},{}];
    function distinct (arr) {
        let newArr = []
        for(let i = 0; i < arr.length; i++) {
            if(!newArr.includes(arr[i])) {
                newArr.push(arr[i])
            } 
        }
        return newArr
    }
    
    console.log(distinct(arr)) // [1, "true", true, 15, false, undefined, null, NaN, "NaN", 0, "a", {…}, {…}]
}
优点:该方法可以顾虑到重复的 String、Boolean、 Number、undefined、null、NaN,返回的是去重后的新数组。
缺点:不能过滤掉 Object

法四 indexOf去重 返回新数组

{
    let arr = [1,1,'true','true',true,true,15,15,false,false, undefined,undefined, null,null, NaN, NaN,'NaN','NaN', 0, 0, 'a', 'a',{},{}];
    function distinct (arr) {
        let newArr = []
        for(let i = 0; i < arr.length; i++) {
            if(newArr.indexOf(arr[i]) < 0) {
                newArr.push(arr[i])
            } 
        }
        return newArr
    }
    
    console.log(distinct(arr)) //  [1, "true", true, 15, false, undefined, null, NaN, NaN, "NaN", 0, "a", {…}, {…}]
}
优点:该方法可以顾虑到重复的 String、Boolean、 Number、undefined、null,返回的是去重后的新数组。
缺点:不能过滤掉 NaN、Object

法五 利用对象的属性key唯一的特性去重

{
    let arr = [1,1,'true','true',true,true,15,15,false,false, undefined,undefined, null,null, NaN, NaN,'NaN','NaN', 0, 0, 'a', 'a',{},{}];
    function distinct(arr) {
        let obj = {}
        let newArr = []
        for(let i = 0; i < arr.length; i++) {
            if(!obj[arr[i]]){
                obj[arr[i]] = 1
                newArr.push(arr[i])
            }
        }
        return newArr
    }

    console.log(distinct(arr)) // [1, "true", 15, false, undefined, null, NaN, 0, "a", {…}]
}
优点:该方法可以顾虑到重复的 String、Boolean、 Number、undefined、null、NaN、Object,返回的是去重后的原数组。
缺点:针对 NaN和'NaN', 对象的key会视为一个key,区分不了NaN和'NaN'

法六 利用ES6的Set数据结构的特性

Set集合里的所有的元素都是唯一的

{
    let arr = [1,1,'true','true',true,true,15,15,false,false, undefined,undefined, null,null, NaN, NaN,'NaN','NaN', 0, 0, 'a', 'a',{},{}];
    function distinct(arr) {
        return Array.from(new Set(arr))
    }

    console.log(distinct(arr)) // [1, "true", true, 15, false, undefined, null, NaN, "NaN", 0, "a", {…}, {…}]
}
优点:该方法可以顾虑到重复的 String、Boolean、 Number、undefined、null、NaN,返回的是去重后的新数组。
缺点:不能过滤重复的Object。

法七 利用ES6的Map数据结构的特性去重

{
    let arr = [1,1,'true','true',true,true,15,15,false,false, undefined,undefined, null,null, NaN, NaN,'NaN','NaN', 0, 0, 'a', 'a',{},{}];
    function distinct(arr) {
        let map = new Map()
        let newArr = []
        for(let i = 0; i < arr.length; i++) {
            if(!map.has(arr[i])) {
                map.set(arr[i])
                newArr.push(arr[i])
            }
        }
        return newArr
    }

    console.log(distinct(arr)) // [1, "true", true, 15, false, undefined, null, NaN, "NaN", 0, "a", {…}, {…}]
}
优点:该方法可以顾虑到重复的 String、Boolean、 Number、undefined、null、NaN,返回的是去重后的新数组。
缺点:不能过滤重复的Object。

法八 利用sort()去重

{
    let arr = [1,1,'true','true',true,true,15,15,false,false, undefined,undefined, null,null, NaN, NaN,'NaN','NaN', 0, 0, 'a', 'a',{},{}];
    function distinct(arr) {
        let sortArr = arr.sort()
        let newArr = []
        for(let i = 1; i < sortArr.length; i++) {
            if(sortArr[i] !== sortArr[i-1]) {
                newArr.push(arr[i])
            }
        }
        return newArr
    }

    console.log(distinct(arr)) // [1, 15, NaN, NaN, "NaN", {…}, {…}, "a", false, null, "true", true, undefined]
}
该方法的缺陷很明显,针对'true','true',true,true,undefined,undefined, null,null,NaN, NaN,0, 0,{},{} 都不能很好的过滤去重,不建议使用该方法去重。

法九 reduce数组去重

{
    let arr = [1,1,'true','true',true,true,15,15,false,false, undefined,undefined, null,null, NaN, NaN,'NaN','NaN', 0, 0, 'a', 'a',{},{}];
    function distinct(arr) {
        return arr.sort().reduce((init, current) => {
            if(init.length === 0 || init[init.length-1] !== current) {
                init.push(current);
            }
            return init;
        }, []);
    }

    console.log(distinct(arr)) // [0, 1, 15, NaN, NaN, "NaN", {…}, {…}, "a", false, null, "true", true, undefined]
}
该方法先对数组进行排序,在进行去重过滤,针对不能过滤重复的 NaN和Object。

从输入URL到页面加载发生了什么

总体来说分为以下几个过程:

  1. DNS解析

  2. TCP连接

  3. 发送HTTP请求

  4. 服务器处理请求并返回HTTP报文

  5. 浏览器解析渲染页面

  6. 连接结束

    JS的5中基本数据类型(简单数据类型):undefined,null,boolean,number,string; 1种复杂数据类型:object.

    typeof操作符--检测给定变量的数据类型
    "undefined" -- 如果这个值未定义;
    "boolean"    --  如果这个值是布尔值;
    "string"        --  如果这个值是字符串;
    "number"     --   如果这个值是数值;
    "object"        --   如果这个值是对象或者null
    "function"     --    如果这个值是函数

    Object类型
    object的每个实例都具有下列属性和方法.
    constructor:保存着用于创建当前对象的函数. 构造函数(constructor)就是Object().

    hasOwnProperty(propertyName): 用于检查给定的属性在当前对象实例中(而不是在实例的原型中)是否存在. 其中,作为参数的属性名(propertyName)必须以字符串形式指定(例如: o.hasOwnProperty("name")).

    isPrototypeOf(object): 用于检查传入的对象是否是当前对象的原型

    propertyIsEnumerable(propertyName): 用于检查给定的属性是否能够使用for-in语法来枚举. 与hasOwnProperty()方法一样,作为参数的属性必须以字符串形式指定.

    toLocaleString(): 返回对象的字符串表示, 该字符传与执行环境的地区对应.

    toString(): 返回对象的字符串表示.

    valueOf(): 返回对象的字符串,数值,或者布尔值表示.通常与toString()方法的返回值相同.

    布尔操作符:
    逻辑非(!)  逻辑与( && )   逻辑或( || )

    Array操作方法:
    concat() 基于当前数组创建一个新的数组
    var colors = ["red","green","blue"]
    var colors2 = colors.concat("yellow",["block","good"] );
    alert(colors2);

    slice() 能够基于当前数组中的一或者多个创建一个新数组, slice可以接受一个或者两个参数,即要返回项的起始和结束位置. 在只有一个参数时, 返回从该参数指定位置到当前数组末尾的所有项. 在有两个参数的情况下, 返回的是起始和结束位置之间的项但不包含结束位置项. 
    var colors = ["red","green","blue"]
    var colors2 = colors.slice(0);
    var colors3 = colors.slice(1,4);
    alert(colors2);
    alert(colors3);
    如果slice()方法的参数中有一个负数,则用数组长度加上该数来确定相应的位置.

    splice()的主要用途是向数组的中部插入项
    splice(0,2)删除数组中的前两项, 只需指定2个参数
    splice(2,0,"red","blue") 会从当前数组的位置2开始插入字符串,需要指定3个参数
    splice(2,2,"热的","冷的") 可以向指定位置插入任意数量的项,且同时删除任意数量的项,只需指定3个参数 (替换)
     

 

js数组乱序输出 数组乱序排列
var arr = [1,2,3,4,5,6,7,8,9,10];
        var newArr = [];
        var len = arr.length;
            for(var i=0; i<len; i++){
                 var index = Math.floor(Math.random()*arr.length);//随机下标
                     newArr.push(arr[index]);//将随机出的元素,存放新数组newArr中去
                     arr.splice(index,1);//    将随机出的元素在arr中删除            
            }
            //arr中删除随机出的元素,arr.length-1,同时i++,导致循环不会10次,会是5次.最后得到newArr中只有一半的随机数字,arr中剩下另一半. 将其合并到一起,得到res
            var res =[...newArr,...arr];
            console.log(res)
js数组扁平化操作
var arr = [1,[2],[3,4],5]
function flatArray(arr){
 return arr.reduce((result,item) =>{
    return result.concat(Array.isArray(item) ? flatArray(item) : item);
    }, [])
}
js字符串反转
1.var name = "My city is WH"; var resultStr = name.split('').reverse().join(''); console.log(resultStr); // HW si ytic yM
2.var strText = "akklafs555"
console.log([...strText].reverse().join('')); //555sfalkka

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值