国产黄色免费网站_久久天天做天天爱综合色_精品国产一区久久久_成人黄色激情网站_久久成人在线视频_美女一区二区在线观看_亚洲精品免费一区亚洲精品免费精品一区 _91精品国产综合久久精品麻豆

死磕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.newchinaweekly.com

存檔

无码国模国产在线观看| 国产亚洲精久久久久久| youjizz国产精品| 亚洲图片欧美激情| 欧美三级蜜桃2在线观看| 亚洲电影免费观看高清完整版在线观看| 亚洲天堂男人天堂女人天堂| 2020欧美日韩在线视频| 国产一区精品在线| 亚洲人精品午夜射精日韩| 小泽玛利亚视频在线观看| 亚洲a v网站| 欧美视频xxxx| 偷偷看偷偷操| 视频在线不卡| 青娱乐自拍偷拍| 涩视频在线观看| 天天干天天色天天干| 91免费在线看片| hs视频在线观看| 国产浮力第一页| 中出嫩模无套| a天堂中文在线| 日韩中文字幕在线一区| 免费国产自线拍一欧美视频| 国产综合色产| 久久久www成人免费无遮挡大片| 欧美最新大片在线看| 欧美老女人性生活| 日韩午夜视频在线观看| 一级做a爰片久久| 三日本三级少妇三级99| 日产精品久久久久| 国产人成高清视频观看| 揉丰满老妇老女人的毛片| 欧美xxx.com| 久久久久久毛片免费看| 日韩精选在线| 欧美一区二区三区免费看| 26uuu久久天堂性欧美| 欧美变态tickle挠乳网站| 国产精品吴梦梦| 久久66热这里只有精品| 亚洲毛片aa| 亚洲av无码一区二区三区观看 | 日韩成年人视频| 国产麻豆一区二区三区精品| 国产在线高清理伦片a| 欧美色女视频| 国产视频在线观看一区二区三区| 亚洲国产欧美一区| 久久国产精品偷| 中国人体摄影一区二区三区| 无码人妻aⅴ一区二区三区| 性生交大片免费看女人按摩| h精品动漫在线观看| 老司机在线精品视频| 91免费观看视频| 亚洲天堂av图片| 亚洲电影网站| 欧美成人精品欧美一| 啦啦啦高清在线观看www| 亚洲www.| 国产成人av电影在线观看| 亚洲第一视频在线观看| 欧洲亚洲一区二区三区四区五区| av免费观看网| 国产精品乱码一区二区视频| 国产天堂av| 奇米狠狠一区二区三区| 亚洲欧洲日产国码二区| 97精品视频在线观看| 中文字幕66页| √天堂资源中文www| 日韩电影免费观看| 麻豆一区二区99久久久久| 欧美电影精品一区二区| 日韩高清三级| 日韩美女视频网站| 中国动漫在线观看完整版免费| 国产一区网站| 欧美性69xxxx肥| 成人av网站观看| 黄色av片三级三级三级免费看| 爽爽影院免费观看视频| 第一区第二区在线| 亚洲激情网站免费观看| 国产日韩av在线| 欧洲美熟女乱又伦| 日本wwwwwwwzzzzz视频| 欧洲杯半决赛直播| 在线观看亚洲成人| 欧美激情专区| 国产三级生活片| 成人久久在线| 不卡的国产精品| 亚洲视频香蕉人妖| 亚洲一区二区中文| 狠狠操狠狠干视频| 四虎成人免费| 波多野结衣欧美| 午夜精品免费在线观看| 99久久无色码| 久久精品一二区| 国产网站在线免费观看| 国产一区二区三区精品视频| 久久福利视频网| jjzz黄色片| 成年女人色毛片免费| 欧美gayvideo| 亚洲另类色综合网站| 91精品视频大全| 国产真人真事毛片| 成人性生交大片免费看午夜| 日本vs亚洲vs韩国一区三区| 久久精品在线播放| 精品国产一区在线| 免费一级网站| 9l视频自拍九色9l视频成人| 又紧又大又爽精品一区二区| 国产综合18久久久久久| 中文字幕 日韩有码| 色偷偷色偷偷色偷偷在线视频| 国产精品色在线| 精品一区2区三区| 国产又黄又粗又长| 精品影院一区| 高清国产午夜精品久久久久久| 久久久久久午夜| 欧美精品久久久久久久久46p| 国产高清免费在线播放| 福利电影一区二区| 国产欧美久久久久久| 日韩经典在线观看| 久草在线中文最新视频| 国产在线日韩| 久久躁日日躁aaaaxxxx| 高潮毛片无遮挡| 高h视频在线| 久久久久久黄色| 欧美二区三区| 强开小嫩苞一区二区三区视频| 欧美日韩一区二区三区四区不卡| 国产日韩欧美电影| 蜜桃在线一区二区三区精品| 国模无码一区二区三区| 国产在线播放精品| 亚洲成**性毛茸茸| 欧美夫妇交换xxx| www 日韩| 亚洲欧洲日韩在线| 超碰人人爱人人| 国产又粗又猛又黄又爽无遮挡| 青青青国产精品| 在线播放日韩导航| 日批视频在线看| 黄色电影免费在线看| 久久免费偷拍视频| 66m—66摸成人免费视频| 极品白嫩少妇无套内谢| 日韩av地址| 欧美韩日一区二区三区四区| 日韩福利视频| 91av.cn| 视频一区二区欧美| 国产视频久久久| 99久久精品免费视频| 免费污视频在线| 久久久久99精品一区| 亚洲一区二区三区色| 黑吊大战白xxxxxx| 国产成人小视频| 先锋影音亚洲资源| 香港日本三级视频| 国内精品在线播放| 精品一区久久| 五月网丁香网| 国产自产v一区二区三区c| 欧美日韩精品免费观看视一区二区| 精品国产一区二区三区久久久狼牙| 老牛国产精品一区的观看方式| 91久久久一线二线三线品牌| 香蕉视频免费在线看| 亚洲欧美日韩国产| 国产精品大全| 青青操夜夜操| 91视频www| 欧美丰满熟妇bbbbbb百度| 天堂av在线资源| 午夜精品影院在线观看| 免费看三级黄色片| 乱馆动漫1~6集在线观看| 欧美成va人片在线观看| 国产免费无码一区二区视频| 国产日韩欧美中文在线| 精品久久国产精品| 国产美女精品视频国产| 久久av一区| 一区二区三区四区在线视频| 丝袜国产免费观看|