手写简单的call bind apply

call

js 复制代码
  const Person1 = {
    fullName: function (country, city) {
      return this.firstName + this.lastName + " " + country + " " + city
    }
  }
  const Person2 = {
    firstName: "张",
    lastName: "三",
  }
  Person1.fullName.call(Person2, '中国', '河南') // 张三 中国 河南 

call的作用是改变了那个被执行的方法(也就是调用call的那个方法)的直接调用者!

而这个被执行的方法内部的this也会重新指向那个新的调用者,就是call方法所接收的第一个obj参数。

还有两个特殊情况就是当这个obj参数为null或者undefined的时候,this会指向window。

js 复制代码
Function.prototype.myCall = function (obj) {
    // 保存this 谁调用的myCall,this就指向谁,指向obj1.fun
    obj.fn = this
    // 保存参数
    let args = Array.from(arguments).slice(1)
    // 调用函数
    let result = obj.fn(...args)
    // 删除fn属性
    delete obj.fn
    return result
}
var obj1 = {
    num: 20,
    fun: function (n) {
        return this.num + n
    }
};
var obj2 = {
    num: 15,
};
let res = obj1.fun.myCall(obj2, 10)
console.log(res);

apply

js 复制代码
  const person = {
    fullName: function (country, city) {
      return this.firstName + this.lastName + " " + country + " " + city
    }
  }
  const newPerson = {
    firstName: "壮",
    lastName: "志国",
  }
  person.fullName.apply(newPerson, ['中国', '河南']) // 壮志国 中国 河南
js 复制代码
Function.prototype.myApply = function (obj) {
    let result
    // 保存this 谁调用的myCall,this就指向谁,指向obj1.fun
    obj.fn = this
    // 是否传参
    if (arguments[1]) {
        result = obj.fn(...arguments[1])
    } else {
        result = obj.fn()
    }
    delete obj.fn
    return result
}
var obj1 = {
    num: 20,
    fn: function (n, m) {
        return this.num + n + m
    }
};
var obj2 = {
    num: 15,
};
let res = obj1.fn.myApply(obj2, [10, 20])
console.log(res);

bind

js 复制代码
  const person = {
    fullName: function (country, city) {
      return this.firstName + this.lastName + " " + country + " " + city
    }
  }
  const newPerson = {
    firstName: "壮",
    lastName: "志国",
  }
   // 打印出fullName函数
  person.fullName.bind(newPerson, '中国', '河南')() // 壮志国 中国 河南
js 复制代码
Function.prototype.myBind = function (obj) {
    // 这里的 this/self 指的是需要进行绑定的函数本身,比如用例中的 man
    const self = this;
    // 获取 myBind 函数从第二个参数到最后一个参数(第一个参数是 obj)
    // 这里产生了闭包
    const args = Array.from(arguments).slice(1)
    return function () {
        // 这个时候的 arguments 是指 myBind 返回的函数传入的参数
        const bindArgs = [...arguments]
        // 合并
        return self.apply(obj, [...args, ...bindArgs]);
    };
};

const person = {
    name: 'zyj'
}

function man(age) {
    console.log(this.name);
    console.log(age)
}

const test = man.myBind(person)
test(18)//zyj 18
相关推荐
jump_jump17 小时前
Ripple:一个现代的响应式 UI 框架
前端·javascript·前端框架
用户9047066835717 小时前
Nuxt css 如何写?
前端
夏天想17 小时前
element-plus的输入数字组件el-input-number 显示了 加减按钮(+ -) 和 小三角箭头(上下箭头),怎么去掉+,-或者箭头
前端·javascript·vue.js
0思必得017 小时前
[Web自动化] Selenium基础介绍
前端·python·selenium·自动化·web自动化
Filotimo_17 小时前
前端.d.ts文件作用
前端
进击的野人17 小时前
Vue 3 响应式数据解构:toRef 与 toRefs 的深度解析
前端·vue.js·前端框架
ohyeah17 小时前
CSS 作用域隔离实战:React、Vue 与 Styled Components 的三种范式
前端
二哈喇子!17 小时前
前端HTML、CSS、JS、VUE 汇总
开发语言·前端
小白路过17 小时前
node-sass和sass兼容性使用
前端·rust·sass