92国产精品视频_亚洲a级在线观看_国产精品电影观看_国产精品免费观看在线_精品伊人久久97_亚洲人成在线观_尤物九九久久国产精品的特点_成人激情在线播放_成人黄色大片在线免费观看_亚洲成人精品久久久_久久免费视频在线观看_久久精品国产一区_国产一区二区三区18_亚洲欧美中文字幕在线一区_日韩美女中文字幕_日韩视频免费在线

死磕javascript的手寫面試題

2021-4-15    前端達人

1.實現lodash的_.get方法

function _getValue(target, valuePath, defalutVal) {
  let valueType = Object.prototype.toString.call(target)
  console.log(valueType)
  // if (valueType == "[object Array]") {
    let paths = valuePath.replace(/\[(\d+)\]/, `.$1`).split('.')
    let result = target
    for(const path of paths){
      result = Object(result)[path]
      if(result == undefined){
        return defalutVal
      }
    }
    return result
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
測試:
let obj = {
  a:{
    b:[
      {
        c:2
      }
    ]
  }
}

console.log(_getValue(obj, 'a.b[0].c')) //2 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

2.寫一個函數判斷兩個變量是否相等

function isEqual(res1, res2) {
  let a = getTypeOf(res1)
  let b = getTypeOf(res2)
  if(a !== b){
    return false
  }else if(a === 'base'){
    console.log('base',res1,res2)
    return res1 === res2
  } else if(a === 'array'){
    if(res1.length !== res2.length){
      console.log('array',res1,res2)
      return false
    }else{
      //遍歷數組的值比較
      for(let i =0;i<res1.length;i++){
        if(!isEqual(res1[i],res2[i])){
          console.log('array',res1[i],res2[i])
          return false
        }
      }
      return true
    }
    return true
  }else if(a === 'object'){
    let ak = Object.keys(a)
    let bk = Object.keys(b)
    if(ak.length !== bk.length){
      return false
    }else{
      for(let o in res1){
        console.log(res1[o])
        if(!isEqual(res1[o],res2[o])){
          console.log('object',res1[o],res2[o])
          return false
        }
      }
      return true
    } 
  }else if(a === 'null' || a === 'undefined'){
    console.log('null')
    return true
  }else if(a === 'function'){
    console.log('function')
    return a === b
  }
}

function getTypeOf(res) {
  let type = Object.prototype.toString.call(res)
  switch (type) {
    case "[object Array]":
      return 'array'
    case "[object Object]":
      return 'object'
    case "[object Null]":
      return 'null'
    case "[object Undefined]":
      return 'undefined'
    case "[object Number]"||"[object String]"||"[object Boolean]":
      return 'base'
    case "[object Function]":
      return 'function'
    default:
      return 'typeError'
  }
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
測試:
let a = {
  a:20,
  b:{
    c:30,
    d:[1,2,3]
  }
}
let b = {
  a:20,
  b:{
    c:30,
    d:[1,2,3]
  }
}
console.log(isEqual(a,b)) //true 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

3.數組扁平化的方法

function _flat(arr){
  let result = []
  for(let i = 0;i<arr.length;i++){
    if(Array.isArray(arr[i])){
      result = result.concat(_flat(arr[i]))
    }else{
      result.push(arr[i])
    }
  }
  return result;
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
let arr = [1,2,[3,4,[5,6]]]
_flat(arr) //[1,2,3,4,5,6] 
  • 1
  • 2
//es6
function _flat2(arr){
  while(arr.some(item=>Array.isArray(item))){
    arr = [].concat(...arr)
  }
  return arr
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
let arr = [1,2,[3,4,[5,6]]]
_flat2(arr) //[1,2,3,4,5,6] 
  • 1
  • 2

4.深克隆

簡單深克隆,不考慮內置對象和函數

function deepClone(obj){
  if(typeof obj !== 'object') return
  let newObj = obj instanceof Array?[]:{}
  for(let key in obj){
      if(obj.hasOwnProperty(key)){
          newObj[key] = typeof obj[key] === 'object'?deepClone(obj[key]):obj[key]
      }
  }
  return newObj
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

復雜版深度克隆 考慮內置對象 比如date regexp 函數 以及對象的循環引用的問題

const isObject = (target) => typeof target === "object"&& target !== null;

function deepClone2(target, map = new WeakMap()) {
  console.log(target)
    if (map.get(target)) {
        return target;
    }
    // 獲取當前值的構造函數:獲取它的類型
    let constructor = target.constructor;
    // 檢測當前對象target是否與正則、日期格式對象匹配
    if (/^(RegExp|Date)$/i.test(constructor.name)) {
        // 創建一個新的特殊對象(正則類/日期類)的實例
        return new constructor(target);  
    }
    if (isObject(target)) {
        map.set(target, true);  // 為循環引用的對象做標記
        const cloneTarget = Array.isArray(target) ? [] : {};
        for (let prop in target) {
            if (target.hasOwnProperty(prop)) {
                cloneTarget[prop] = deepClone(target[prop], map);
            }
        }
        return cloneTarget;
    } else {
        return target;
    }
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27

5.數組去重

filter去重

function _unique(arr){
  return arr.filter((item,index,array)=>{
    return array.indexOf(item) === index
  })
} 
  • 1
  • 2
  • 3
  • 4
  • 5

es6 Set

function _unique2(arr){
  return [...new Set(arr)]
} 
  • 1
  • 2
  • 3

includes

function _unique3(arr){
  let newArr = []
  arr.forEach(item => {
      if(!newArr.includes(item)){
        newArr.push(item)
      }
  });
  return newArr
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

雙層for循環

function _unique4(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--
      }
    }
  }
  return arr
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

indexof

function _unique5(arr){
  let newArr = []
  for(let i = 0;i<arr.length;i++){
    if(newArr.indexOf(arr[i] === -1){
      newArr.push(arr[i])
    })
  }
  return newArr
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

6.判斷數據的類型

function _typeOf(obj){
  let res = Object.prototype.toString.call(obj).split(' ')[1]
  let mold = res.substring(0,res.length-1).toLowerCase()
  return mold
} 
  • 1
  • 2
  • 3
  • 4
  • 5
_typeOf(5) //number
_typeOf('5') //string 
  • 1
  • 2

7.解析url參數為對象

function getParamsObj(params){
  let paramsStr = params.replace(/^.+\?(.+)/,"$1")
  let paramsArr = paramsStr.split('&')
  let paramsObj = {}

  for(let [key,value] of paramsArr.entries()){
      if(/=/.test(value)){
          let valArr = value.split('=')
          val = decodeURIComponent(valArr[1]) //解碼
          val = /^\d+$/.test(val)?parseFloat(val):val //判斷是不是數字
          if(paramsObj.hasOwnProperty(valArr[0])){
              paramsObj[valArr[0]] = [].concat(paramsObj[valArr[0]],val)
          }else{
              paramsObj[valArr[0]] = val
          }
      }  

  }
  return paramsObj
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20

8.函數柯里化

//從一次傳入多個參數  編程多次調用每次傳入一個參數
function add(a, b, c, d, e) {
  return a + b + c + d + e
}

function curry(fn) {
   let dFn = (...args)=>{
     if(args.length == fn.length) return fn(...args)
     return (...arg)=>{
       return dFn(...args,...arg)
     }
   }
   return dFn
}
let addCurry = curry(add)
addCurry(1,2,3)(2)(3) 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

9.圖片懶加載

//添加了兩個功能
// 圖片加載完成后 移除事件監聽
// 加載完的圖片從imgList中移除
let imgList = [...document.querySelectorAll('img')]
let length = imgList.length

const imgLazyLoad = function () {
  let count = 0
  let deleteIndexList = []
  imgList.forEach((img, index) => {
    let rect = img.getBoundingClientRect() 
    //獲取元素到視圖的距離 top元素上邊到視圖上邊的距離 left元素左邊到視圖左邊的距離  right... bottom...
    if (rect.top < window.innerHeight) {
      // img.src = img.dataset.src
      img.src = img.getAttribute('data-src')
      deleteIndexList.push(index)
      count++
      if (count === length) {
        document.removeEventListener('scroll', imgLazyLoad)
      }
    }
  })
  imgList = imgList.filter((img, index) => !deleteIndexList.includes(index))
}
imgLazyLoad()

document.addEventListener('scroll', imgLazyLoad) 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27

圖片懶加載:https://juejin.cn/post/6844903856489365518#heading-19

10節流防抖

函數防抖 觸發高頻事件 事件在n后執行,如果n秒鐘重復執行了 則時間重置

//簡易版
function debounce(func,wait){
  let timer; 
  return function(){
    let context = this;
    let args = arguments;
    console.log(timer)
    clearTimeout(timer)
    timer = setTimeout(function(){
      func.apply(context,args)
    },wait)
  }

}
let btn = document.querySelector('button');
function aa(){
  console.log(111)
}
btn.onclick = debounce(aa,2000) 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
// 復雜版
// 1.取消防抖  
// 2.立即執行功能(點擊之后立即執行函數  但是 wait時間之后在點擊才能在立即執行)  
// 3.函數可能有返回值
function debounce(func,wait,immediate){
  let timer,result;

  const debounce = function () {
    const context = this
    const args = arguments

    if(timer) clearTimeout(timer)
    if(immediate){
      console.log(timer)
      var callNow = !timer
      timer = setTimeout(function () {
          timer =null
      },wait)
      if(callNow) result = func.apply(context,args)
    }else{
      timer = setTimeout(function (params) {
        result = func.apply(context,args)
      },wait)
    }
    return result
  }

  debounce.cance = function () {
    clearTimeout(timer)
    timer=null
  }

  return debounce

}

let btn = document.querySelector('button');
function aa(){
  console.log(111)
}
btn.onclick = debounce(aa,2000,true)``` 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41

函數節流 觸發高頻事件 且n秒只執行一次

//使用時間戳
function  throttle(func,wait) {
  var context,args;
  var previous = 0

  return function () {
    context = this;
    args = arguments;
    let nowDate = +new Date()
    if(nowDate-previous>wait){
      func.apply(context,arguments)
      previous = nowDate
    }
  }
} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
//定時器
function throttle(func,wait) {
  var context,args;
  var timer;
  return function(){
    context = this;
    args = arguments;
    if(!timer){
      timer = setTimeout(function () {
        timer = null;
        func.apply(context,args)
      },wait)
    }
  }

} 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
//組合版 options.leading 為true 立即執行一次 options.trailing為true  結束之后執行一次 默認為true
function throttle(func, wait ,options = {}) {
  var context, args, timer,result;
  var previous = 0;

  var later = function () {
    previous = options.leading === false ? 0 : new Date().getTime();
    timer = null;
    func.apply(context, args)
    if (!timer) context = args = null;
  }

  var throttle = function () {
    var now = new Date().getTime()
    if (!previous && options.leading === false) previous = now;
    context = this;
    args = arguments;

    //下次觸發 func 剩余的時間
    var remaining = wait - (now - previous);
    if (remaining <= 0 || remaining > wait) {
      // if (timer) {
      //   clearTimeout(timer);
      //   timer = null;
      // }
      previous = now;
      func.apply(context, args);
      if (!timer) context = args = null;
    } else if (!timer&& options.trailing !== false) {
      timer = setTimeout(later, remaining);
    }
  }

  throttled.cancel = function() {
    clearTimeout(timer);
    previous = 0;
    timer = null;
  }

  return throttle
}

function aa(e) {
  console.log(111)
  console.log(e)
}

let btn = document.querySelector('button');
btn.onclick = throttle(aa, 2000,{
  leading:false,
  trailing:true 

})

轉自:csdn論壇 作者:Selfimpr歐

日歷

鏈接

個人資料

藍藍設計的小編 http://www.skdbbs.com

存檔

92国产精品视频_亚洲a级在线观看_国产精品电影观看_国产精品免费观看在线_精品伊人久久97_亚洲人成在线观_尤物九九久久国产精品的特点_成人激情在线播放_成人黄色大片在线免费观看_亚洲成人精品久久久_久久免费视频在线观看_久久精品国产一区_国产一区二区三区18_亚洲欧美中文字幕在线一区_日韩美女中文字幕_日韩视频免费在线
97精品一区二区三区| 亚洲在线观看视频网站| 日韩电影免费观看在线| 亚洲欧洲午夜一线一品| 久久国产精品免费视频| 亚洲国产欧美在线成人app| 久久99精品久久久久久琪琪| 欧美午夜精品在线| 欧美极品美女电影一区| 日韩免费av片在线观看| 久久在线免费观看视频| 69视频在线免费观看| 日韩在线免费视频观看| 日韩av理论片| 91在线国产电影| 精品国产91久久久| 福利视频第一区| 97热精品视频官网| 日本伊人精品一区二区三区介绍| 日本午夜精品理论片a级appf发布| 久热在线中文字幕色999舞| 2019中文字幕免费视频| 久久高清视频免费| 国产成人啪精品视频免费网| 国产97色在线| 国产精品爱啪在线线免费观看| 国产精品精品久久久久久| 欧美性做爰毛片| 日韩乱码在线视频| 欧美一级电影免费在线观看| 日本精品免费一区二区三区| 国产精品久久久久久av福利| 97碰碰碰免费色视频| 欧美性猛交xxxx黑人| 亚洲成人av中文字幕| 亚洲精品91美女久久久久久久| xvideos成人免费中文版| 亚洲国产精品yw在线观看| 久久精品国产91精品亚洲| 欧美在线观看日本一区| 久久久久久18| 91精品视频专区| 国产亚洲欧美日韩一区二区| 精品香蕉在线观看视频一| 日韩欧美综合在线视频| 亚洲一区二区精品| 国产一区二区日韩| 精品国产欧美一区二区五十路| 91免费观看网站| 欧美孕妇孕交黑巨大网站| 亚洲精品99久久久久中文字幕| 青青久久aⅴ北条麻妃| 欧美激情区在线播放| 欧美日韩国产一区中文午夜| 欧美精品18videos性欧| 国产在线精品成人一区二区三区| 97精品视频在线播放| 亚洲色图在线观看| 免费91麻豆精品国产自产在线观看| 欧美性猛交99久久久久99按摩| 亚洲国产小视频在线观看| 久久久久999| 91在线观看免费高清| 在线观看欧美日韩| 国产精品入口免费视频一| 成人免费视频网| 国产精品一区二区av影院萌芽| 欧美伊久线香蕉线新在线| 在线日韩欧美视频| 国产一区二区视频在线观看| 自拍偷拍亚洲在线| 欧美激情一级精品国产| 欧美xxxx综合视频| 日韩av电影免费观看高清| 久色乳综合思思在线视频| 亚洲成人av片在线观看| 亚洲精品在线91| 亚洲精品日韩av| 亚洲的天堂在线中文字幕| 亚洲成人在线网| 精品欧美国产一区二区三区| 91精品久久久久久久久中文字幕| 欧美老少做受xxxx高潮| 最好看的2019年中文视频| 国产精品久久久久免费a∨| 亚洲第一页在线| 国产一区二区香蕉| 国产精品福利久久久| 国产精品福利小视频| 51久久精品夜色国产麻豆| 欧美日韩免费观看中文| 亚洲视频在线观看网站| 日韩欧美视频一区二区三区| 91天堂在线观看| 一区二区三区无码高清视频| 国产日韩精品综合网站| 亚州av一区二区| 日韩中文字幕视频| 亚洲视频欧洲视频| 日韩高清电影免费观看完整版| 亚洲视频第一页| 久久欧美在线电影| 原创国产精品91| 亚洲老头同性xxxxx| 最近2019中文字幕mv免费看| 日本伊人精品一区二区三区介绍| 欧美成人合集magnet| 欧美日韩亚洲视频一区| 亚洲人成电影网| 欧美成人自拍视频| 国产又爽又黄的激情精品视频| 国产日韩精品一区二区| 日韩欧美精品免费在线| 国产精品一区二区久久久| 国产精品户外野外| 亚洲综合在线做性| 国产一区红桃视频| 日韩成人av一区| 亚洲国产女人aaa毛片在线| 久久精品国产亚洲一区二区| 欧美激情第6页| 在线视频免费一区二区| 在线播放国产一区中文字幕剧情欧美| 激情懂色av一区av二区av| 国产女人18毛片水18精品| 国产97在线|亚洲| 欧美日韩在线一区| 亚洲va久久久噜噜噜| 91美女片黄在线观| 欧美成aaa人片在线观看蜜臀| 91色视频在线导航| 97视频在线观看免费高清完整版在线观看| 日韩美女免费线视频| 欧美诱惑福利视频| 日本精品性网站在线观看| 久久躁狠狠躁夜夜爽| 亚洲三级 欧美三级| 最近2019中文免费高清视频观看www99| 久久人人97超碰精品888| 久久久久中文字幕2018| 日韩网站在线观看| 亚洲一区二区三区sesese| 中文字幕亚洲专区| 国产极品精品在线观看| 中文字幕日韩欧美在线视频| 欧美激情网友自拍| 日韩av在线网页| 国产91精品久久久久| 成人在线观看视频网站| 欧美日韩在线视频首页| 久久精品视频导航| 精品亚洲一区二区三区| 国产香蕉97碰碰久久人人| 性欧美视频videos6一9| 亚洲精品一区二区三区婷婷月| 欧美午夜精品在线| 精品性高朝久久久久久久| 九九热精品视频国产| 国产精品海角社区在线观看| 奇米四色中文综合久久| 久久精品国产96久久久香蕉| 国产成人精品a视频一区www| 亚洲国产欧美久久| 欧美日在线观看|