超级实用手写 JavaScript 方法

1.数组去重方法
ES6 Set关键字
const _uniq = array => {
    return [...new Set(array)]
}
数组方法indexOf
const uniq = array => {
    let result = [];
    for(let i in array){
        if(result.indexOf(array[i]) === -1) result.push(array[i])
    }
    return result;
}
数组方法sort

纯数字,先排序再用相邻的对象进行比较

const uniq = array => {
    let result = [];
    array.sort()
    for(let i = 0;i<array.length;i++){
        if(array[i]!== array[i-1]) result.push(array[i])
    }
    return result;
}
双指针 方法

指针所指的值不相等的时候才加一,并改变原数组,最后用slice方法和慢指针取值

[-1,1,2,2]
const uniq = array => {
    let slow = 0,fast = 0;
    while(fast<array.length){
        if(array[slow] !== array[fast]){
            slow++;
            array[slow] = array[fast]
        }
        fast++
    }
    return array.slice(0,slow+1)
}
数组方法reduce
const uniq = array => {
    return array.reduce((a,b)=>a.includes(b)?a:[...a,b],[])
}
2.reduce方法
for循环 方法

上一次输出的值作为下一次输入的值

Array.prototype._reduce = function(fn){
    let num = 0;
    for(let i = 0;i<this[i];i++){
        num = fn(num,this[i],i,this)
    }
    return num;
}
3.map方法
for循环 方法
Array.prototype._map = function(fn){
    let result = [];
    for(let i in this){
        result[i] = fn(this[i])
    }
    return result;
}
数组方法reduce

上一次输出的值作为下一次输入的值,a初始值为[]

Array.prototype._map = function(fn){
    return this.reduce((a,b)=> [...a, fn(b)], [])
}
4.filter方法
for循环
Array.prototype._filter = function(fn){
    let result = [];
    for(let i in this){
        if(fn(this[i],i,this)) result.push(this[i])
    }
    return result;
}
数组方法reduce

上一次输出的值作为下一次输入的值,a初始值为[]

Array.prototype._filter = function(fn){
    return this.reduce((a, b) => fn(b) ? [...a, b] : a, []);
}
5.some方法
for循环 方法

有满足条件的就返回true

Array.prototype._some = function(fn){
    for(let i in this){
        if(fn(this[i])) return true;
    }
    return false;
}
数组方法reduce

上一次输出的值作为下一次输入的值,a初始值为[]

Array.prototype._filter = function(fn){
    return this.reduce((a, b) => fn(b) ? [...a, b] : a, []);
}
6.flat方法
while 循环+ some方法

判断数组元素中是否还有数组,使用concat拼接,使数组扁平化

function flat(arr) {
    while (arr.some(item=> Array.isArray(item))) {
        arr = [].concat(...arr)
    }
    return arr
}
reduce 方法

上一次输出的值作为下一次输入的值,a初始值为[]

function flat(arr) {
  return arr.reduce((a, b) => a.concat(Array.isArray(b) ? flat(b) : b), [])
};
使用堆栈避免递归 方法

使用数据结构栈的特性取代递归操作,减少时间复杂度

function flat(arr) {
    const result = []; 
    const stack = [].concat(arr);
    while (stack.length !== 0) {
        const val = stack.pop(); 
        if (Array.isArray(val)) stack.push(...val) 
        else result.unshift(val)
    }
    return result;
}

7.字符串全排列

递归 方法
const _permute = string => {
    if(string.length<2) return [string];
    let result = [];
    for(let i of string){
        let arr = string.split('').filter(e=>e!==i);
        _permute(arr.join('')).forEach(item=>{
            result.push(item+i)
        })
    }
    return result;
}
8.instanceof
const _instanceof = (target, Fn) => {
    while(true){
    if(target === null) return false;
    if(target.__proto__ === Fn.prototype) return true;
    target = target.__proto__
}
9.实现函数 call 方法
Function.prototype._call = function(context,...arg){
    if(!context)  context !== window;
    let fn = Symbol();
    context[fn] = this;
    return context[fn](...arg)
}
10.实现函数 bind 方法
Function.prototype._bind = function(context,...args){
    let _this = this;
    return function(...innerArg){
        return _this.call(context,args.concat(innerArg))
    }
}
11.防抖节流柯里化
防抖

search搜索联想,用户在不断输入值时,用防抖来节约请求资源(在输入完毕后执行)

const debounce = (fn,delay = 1000)=>{
    let timer;
    return function(){
        let args = argumens;
        clearTimeout(timer)
        timer = setTimeout(()=>{
            fn.apply(this,args)
        },delay)
    }
}
节流

——鼠标不断点击触发,单位时间内只触发一次

const throttle = (fn,delay = 1000)=>{
    let flag = true;
    return function(){
        if(!flag) return;
        flag = false;
        setTimeout(()=>{
            fn();
            flag = ture;
        },delay)
    }
}
柯里化
const curryIt = (fn)=>{
    let len = fn.length,args = [];
    let result = function(arg){
        args.push(arg);
        len--;
        if(len<=0) return fn.apply(this,args);
        else return result;
    }
    return result;
}
let fn = (a, b, c)=>  a + b + c; 
curryIt(fn)(1)(2)(3); // 6
12.new 关键字
const _new = function(fn,...args) {
    let obj = {}
    obj.__proto__ = fn.prototype
    fn.call(obj,...args)
    return obj
}
13.深浅拷贝
浅拷贝

只复制指向某个对象的指针,而不复制对象本身,新旧对象共享一块内存

利用 = 赋值操作符实现浅拷贝
深拷贝

复制并创建一个一摸一样的对象,不共享内存,修改新对象,旧对象保持不变

const _cloneDeep = (target, map = new Map()) => {
    if(target !== null && typeof target === 'object'){
        let result = Array.isArray(target)?[]:{};
        if(map.has(target)) return map.get(target);
        map.set(target,result);
        for(let i in target){
            result[i] = _cloneDeep(target[i],map);
        }
        return result;
    }
    return target;
}
14.发布订阅模式
class EventEmitter {
    constructor(){
        this.eventList = {}
    }
    on(event,fn){
        let fns = this.eventList[event] || [];
        fns.push(fn);
        this.eventList[event] = fns;
    }
    emit(event,...message){
        let fns = this.eventList[event] ||[];
        if(!fns) return;
        fns.forEach(item=>{
            item(...message)
        })
    }
}
15.斐波那契数列
function fibonacci(n) {
    if(n<3) return 1;
    return fibonacci(n-1) + fibonacci(n-2)
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容