在js中,我们实现重载常用的方式有:

1、根据传入参数的类型执行不同的操作。

2、利用参数中特殊的参数值进行不同的操作。

3、根据参数的个数进行重载。

这里对第三种重载方式的实现进行说明。

实现第三种方法,最简单的就是使用switch case进行参数个数的判断,然后执行相应的操作,但这样的判断方法导致代码不是很整洁,逼格也不算是太高。

如:

function fn() {
switch (arguments.length) {
case 0:
//执行语句块
break;
case 1:
//执行语句块
break;
case 2:
//执行语句块
break;
default:
break;
}
}

下面介绍一种方法,先不考虑是否适用,但可以通过了解这种重载方式,让我们学到一些JS中的技巧和对闭包的一些理解。毕竟多了解些不算件坏事。

先上代码:

var arr = {
value: ["a", "b", "c"]
};
bindMethod(arr, "find", function () {
console.log(0)
});
bindMethod(arr, "find", function (a) {
console.log(1)
});
bindMethod(arr, "find", function (a, b) {
console.log(2)
});
function bindMethod(obj, name, fn) {
var old = obj[name]; //每个方法中都会存在一个各自的old对象,产生了闭包
obj[name] = function () {
if (fn.length == arguments.length) { //这里的fn.length是方法定义参数的个数,和arguments.length传入的个数是两回事
return fn.apply(this, arguments); //返回传入在匿名函数,执行
}
else {
return old.apply(this, arguments); //返回数组的方法,再次向上进行查找
}
}
}
arr.find(); //
arr.find(1); //
arr.find(1, 2); //
arr.find(1, 2, 3); //Uncaught TypeError: Cannot read property 'apply' of undefined /**
第一次进行绑定时,先赋值old,因为attr.find并没有被创建,所以old=undefined。然后为arr对象find属性创建一个匿名方法,这个匿名方法fn.length=0。
第二次进行绑定时,将old对象指向第一次创建的匿名方法,然后会将arr对象find属性指向到当前创建的匿名方法,因为old指向上个方法的引用,这就会造成上个方法不会被释放。
第三次也是一样的步骤,保证第二次绑定的方法不会被释放。
重载方法的执行,像一个链条一样,先从最后绑定的方法内部开始寻找 方法定义的参数个数和传入的个数是否匹配,如果匹配则返回当前的传入的匿名函数,否则继续向上查找,直到找到为止。
如果最后没有找到结果,当前程序会报错,因为第一次绑定方法时,old对象是一个undefined
**/

绑定阶段

先将arr绑定三个方法,每个方法都有一个old的对象,存放之前保存的方法,以此类推,(当前的方法中,old对象都会指向上一次绑定方法,但是注意,在第一次绑定的方法中,old对象是undefind),这样就可以形成一个链条。也就是说内存中分别有三个attr.find的内存空间,都指向各自创建bindMethod内的匿名方法。

调用阶段
当传入0个参数,进行调用时,会先调用最后一个绑定的方法。然后进行判断,如果传入参数为2个,则返回匿名方法,否则
调用old指向的上个方法。再次进入,再次进行判断,直到找到符合参数个数的方法,返回它的匿名函数。

在这里对于attr对象的绑定,其实是进行了三次绑定覆盖的操作,依次进行对attr.find进行赋值,最后只保留最后一次的赋值对象。

这也是为什么在进行调用时,是从最后一次绑定的方法中开始执行的原因。对于attr.find方法被覆盖,但却没有被自动释放的原因是因为新方法的old

始终指向被覆盖的方法,所以这里产生了闭包。

JS实现重载的更多相关文章

  1. JS函数重载解决方案

    JS的函数定义可以指定形式参数名称,多多少少我们会以为js至少可以支持参数个数不同的方法重载,然而遗憾的是这仅仅是一个假象,js所有的参数都是以arguments传递过去的,这个参数类似于数组,在函数 ...

  2. js没有重载

    javascript与其他语言(如java)不同,它没有传统意义上的重载(即为函数编写两个定义,只要这两个函数的参数类型或数量不同即可),在js中,后定义的函数会覆盖先前的函数.js中的参数在内部是用 ...

  3. js中重载问题

    在js中是没有重载的  但是  Arguments对象(可以实现模拟重载的效果) 利用arguments对象的length属性,可以获取函数接收的参数的个数 例如: function add(){ i ...

  4. js方法重载

    test(5); test(5,5); function test(a){ alert(a); } function test(a,b){ alert(a+b); } NaN和10,说明第二个覆盖了第 ...

  5. JS 模拟 重载

    重载:方法,根据传入的参数列表不同,执行不同的任务. 比如:functiion jz(money){ //现金结账:验钞,找零 }     function jz(cardId,pwd){ //刷卡结 ...

  6. js的重载

    1.重载 //重载(个数不同,类型不同)function prop(){var firstP = document.getElementById("p");if(arguments ...

  7. js 方法重载

    function fun1(arm1) { alert(arm1); } function fun2(arm1, arm2) { alert(arm1 + "_" + arm2); ...

  8. js 函数重载

    简单定义:根据不同参数长度来实现让同一个函数,进行不同处理. function addMethod (obj, name, fun) { let old = obj[name] obj[name] = ...

  9. js函数的伪重载

    这也是今天写东西是遇到的一个问题,导致我联想起了函数重载的问题. 在javascript中是没有函数重载机制的,对于用惯了java开发的同学可能就表示吃惊了,我屮艸芔茻,函数 没有重载?那怎么搞?!! ...

随机推荐

  1. Google BERT应用之《红楼梦》对话人物提取

    Google BERT应用之<红楼梦>对话人物提取 https://www.jiqizhixin.com/articles/2019-01-24-19

  2. Vue系列——动态设置img标签的src属性

    声明 本文转自:vue动态设置img的src路径 正文 相信开发的小伙伴已经遇到这个问题了,动态切换img标签的src时,写的路径就是不生效,原因是vue并没有把你的路径字符串当做路径来处理,而是直接 ...

  3. pom.xml activatedProperties --spring.profiles.active=uat 对应

    <profiles> <profile> <id>dev</id> <properties> <!-- 环境标识,需要与配置文件的名称 ...

  4. Linux权限:提示-bash: ./startup.sh: Permission denied的解决方案

    Linux权限:提示-bash: ./startup.sh: Permission denied的解决方案 Linux上启动Tomcat,结果弹出:-bash: ./startup.sh: Permi ...

  5. Qt编写项目作品大全(自定义控件+输入法+大屏电子看板+视频监控+楼宇对讲+气体安全等)

    一.自定义控件大全 (一).控件介绍 超过160个精美控件,涵盖了各种仪表盘.进度条.进度球.指南针.曲线图.标尺.温度计.导航条.导航栏,flatui.高亮按钮.滑动选择器.农历等.远超qwt集成的 ...

  6. k8s记录-国内下载k8s组件镜像

    #!/bin/sh ### 版本信息 K8S_VERSION=v1.13.2 ETCD_VERSION=3.2.24 DASHBOARD_VERSION=v1.8.3 FLANNEL_VERSION= ...

  7. PMP 第4章错题总结

    变更步骤: 1.配置管理活动:配置识别.配置状态记录.配置核实与审计2.项目章程中记录项目的目的和总体预算3.变更控制系统规定了变更管理流程及批准的权限4.项目章程是授权项目经理动用组织资源的文件5. ...

  8. 基于Spring Boot架构的前后端完全分离项目API路径问题

    最近的一个项目采用前后端完全分离的架构,前端组件:vue + vue-router + vuex + element-ui + axios,后端组件:Spring Boot + MyBatis.之所以 ...

  9. docker load tar.gz包失败解决方法

    执行docker load -i xxx.tar.gz时候报错 open /var/lib/docker/xxx No such file or directory 这种情况属于容器快照文件缺乏基础镜 ...

  10. F# 4.6 预览版正式公布

    1月24日,F# 4.6 预览版正式公布,与以往一样,新版本的设计与开发过程是整个 F# 开源社区共同努力的结果,这次更新的具体讨论内容可以通过下面两个链接来查看: F# 4.6 意见征求记录 FSh ...