手写简单的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
相关推荐
一叶飘零晋7 小时前
【(一)Electron 使用之如何用vite+vue3搭建初始框架】
前端·javascript·electron
光影少年7 小时前
前端SSR和ssg区别
前端·vue.js·人工智能·学习·react.js
广州华水科技7 小时前
北斗形变监测传感器在水库安全监测中的应用与发展
前端
凯瑟琳.奥古斯特8 小时前
Bootstrap快速上手指南
开发语言·前端·css·bootstrap·html
精益数智工坊8 小时前
拆解制造业仓库物料管理流程:如何通过标准化仓库物料管理流程解决账实不符难题
大数据·前端·数据库·人工智能·精益工程
恶猫8 小时前
网页自动化模拟操作时,模拟真实按键触发事件【终级方案】
前端·javascript·自动化·vue·网页模拟
小羊Yveesss8 小时前
2026年前端开发新趋势:智能协同、工具革新与场景深耕
前端·ai
Dxy12393102169 小时前
HTML中的Canvas可以干哪些事情
前端·html
悟乙己9 小时前
解析 Agent 时代的 HTML PPT SKILLS: html-ppt-skill
前端·html·powerpoint
ZC跨境爬虫9 小时前
跟着 MDN 学 HTML day_2:(表单分组与高级输入控件实战)
前端·javascript·css·ui·html