JS手写-this绑定实现

在 JavaScript 中,bindcallapply 方法都可以用来改变函数的 this 指向。下面我们将分别实现这些方法的简单版本。

1. 实现 bind

bind 方法创建一个新的函数,在调用时设置 this 值,并返回这个新的函数。

javascript 复制代码
Function.prototype.myBind = function (context) {
    if (typeof this !== 'function') {
        throw new TypeError('Not a function');
    }

    const fn = this;
    const args = Array.prototype.slice.call(arguments, 1);

    return function bound() {
        const boundArgs = Array.prototype.slice.call(arguments);
        return fn.apply(context, args.concat(boundArgs));
    };
};

2. 实现 call

call 方法立即调用函数,并设置 this 值,同时传递参数列表。

javascript 复制代码
Function.prototype.myCall = function (context) {
    if (typeof this !== 'function') {
        throw new TypeError('Not a function');
    }

    context = context || window;
    const args = Array.prototype.slice.call(arguments, 1);

    context.fn = this;
    const result = context.fn(...args);
    delete context.fn;

    return result;
};

3. 实现 apply

apply 方法与 call 类似,但传递的参数是一个数组。

javascript 复制代码
Function.prototype.myApply = function (context, args) {
    if (typeof this !== 'function') {
        throw new TypeError('Not a function');
    }

    context = context || window;
    args = args || [];

    context.fn = this;
    const result = context.fn(...args);
    delete context.fn;

    return result;
};

示例代码

下面是一个完整的示例,展示了如何使用这些自定义的方法:

html 复制代码
<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>自定义 bind, call, apply</title>
</head>
<body>
    <script type="text/javascript">
        Function.prototype.myBind = function (context) {
            if (typeof this !== 'function') {
                throw new TypeError('Not a function');
            }

            const fn = this;
            const args = Array.prototype.slice.call(arguments, 1);

            return function bound() {
                const boundArgs = Array.prototype.slice.call(arguments);
                return fn.apply(context, args.concat(boundArgs));
            };
        };

        Function.prototype.myCall = function (context) {
            if (typeof this !== 'function') {
                throw new TypeError('Not a function');
            }

            context = context || window;
            const args = Array.prototype.slice.call(arguments, 1);

            context.fn = this;
            const result = context.fn(...args);
            delete context.fn;

            return result;
        };

        Function.prototype.myApply = function (context, args) {
            if (typeof this !== 'function') {
                throw new TypeError('Not a function');
            }

            context = context || window;
            args = args || [];

            context.fn = this;
            const result = context.fn(...args);
            delete context.fn;

            return result;
        };

        // 测试对象
        const obj = {
            name: '牛客网'
        };

        // 测试函数
        function greet(message) {
            console.log(`${message}, ${this.name}`);
        }

        // 使用 myBind
        const greetBound = greet.myBind(obj, '欢迎来到');
        greetBound(); // 输出: 欢迎来到, 牛客网

        // 使用 myCall
        greet.myCall(obj, '欢迎来到'); // 输出: 欢迎来到, 牛客网

        // 使用 myApply
        greet.myApply(obj, ['欢迎来到']); // 输出: 欢迎来到, 牛客网
    </script>
</body>
</html>

详细步骤

  1. 实现 myBind

    • 检查调用者是否为函数。
    • 获取上下文 context 和传入的参数 args
    • 返回一个新的函数 bound,在调用时使用 apply 方法设置 this 值并传递参数。
  2. 实现 myCall

    • 检查调用者是否为函数。
    • 获取上下文 context 和传入的参数 args
    • 将函数赋值给 context 的一个临时属性 fn,调用该属性并传递参数,然后删除该属性。
  3. 实现 myApply

    • 检查调用者是否为函数。
    • 获取上下文 context 和传入的参数数组 args
    • 将函数赋值给 context 的一个临时属性 fn,调用该属性并传递参数,然后删除该属性。

测试

  1. 使用 myBind

    • 创建一个绑定了 obj 上下文的新函数 greetBound,并调用它。
  2. 使用 myCall

    • 直接调用 greet 函数,并设置 objthis 值。
  3. 使用 myApply

    • 直接调用 greet 函数,并设置 objthis 值,参数以数组形式传递。
相关推荐
星轨初途几秒前
C++ 类和对象(下):初始化列表、static 成员与编译器优化深度剖析
android·开发语言·c++·经验分享·笔记
量子炒饭大师1 分钟前
【C++ 入门】Cyber动态义体——【vector容器】vector底层原理是什么?该怎么使用他?一文带你搞定所有问题!!!
开发语言·c++·vector·dubbo
AC赳赳老秦3 分钟前
OpenClaw 系统监控实战指南:构建高效的电脑/服务器状态监控与自动告警系统
服务器·开发语言·人工智能·php·ai-native·deepseek·openclaw
宝耶5 分钟前
Java面试题5:List、Set、Map 的区别?各自有哪些实现类?
java·开发语言·list
殷忆枫5 分钟前
基于STM32的ML307R连接Onenet平台
服务器·前端·javascript
Cosmoshhhyyy6 分钟前
《Effective Java》解读第44条:坚持使用标准的函数接口
java·开发语言
yunyun321236 分钟前
动态库热加载技术
开发语言·c++·算法
88号技师6 分钟前
2026年3月一区SCI-B样条曲线优化算法B-spline curves optimizer-附Matlab免费代码
开发语言·算法·数学建模·matlab·优化算法
Java 码农7 分钟前
vue cli 环境搭建
前端·javascript·vue.js
dapeng28707 分钟前
C++中的享元模式实战
开发语言·c++·算法