[转载]浅谈JavaScript函数重载
原文地址:浅谈JavaScript函数重载 作者:ChessZhang
上个星期四下午,接到了网易的视频面试(前端实习生第二轮技术面试)。面了一个多小时,自我感觉面试得很糟糕的,因为问到的很多问题都很难,根本回答不上来。不过那天晚上,还是很惊喜的接到了HR面电话。现在HR面试的结果还没有出来,听说要等到下周二才出,所以再好好等几天哈。

- function overLoading() {
- // 根据arguments.length,对不同的值进行不同的操作
- switch(arguments.length) {
- case 0:
- /*操作1的代码写在这里*/
- break;
- case 1:
- /*操作2的代码写在这里*/
- break;
- case 2:
- /*操作3的代码写在这里*/
- //后面还有很多的case......
- }
- }

我把代码发给了面试官。
- var people = {
- values: ["Dean Edwards", "Sam Stephenson", "Alex Russell", "Dean Tom"]
- };
我们希望people对象拥有一个find方法,当不传任何参数时,就会把people.values里面的所有元素返回来;当传一个参数时,就把first-name跟这个参数匹配的元素返回来;当传两个参数时,则把first-name和last-name都匹配的才返回来。因为find方法是根据参数的个数不同而执行不同的操作的,所以,我们希望有一个addMethod方法,能够如下的为people添加find的重载:
- addMethod(people, "find", function() {}); /*不传参*/
- addMethod(people, "find", function(a) {}); /*传一个*/
- addMethod(people, "find", function(a, b) {}); /*传两个*/
这时候问题来了,这个全局的addMethod方法该怎么实现呢?John Resig的实现方法如下,代码不长,但是非常的巧妙:

- function addMethod(object, name, fn) {
- var old = object[name]; //把前一次添加的方法存在一个临时变量old里面
- object[name] = function() { // 重写了object[name]的方法
- // 如果调用object[name]方法时,传入的参数个数跟预期的一致,则直接调用
- if(fn.length === arguments.length) {
- return fn.apply(this, arguments);
- // 否则,判断old是否是函数,如果是,就调用old
- } else if(typeof old === "function") {
- return old.apply(this, arguments);
- }
- }
- }

现在,我们一起来分析一个这个addMethod函数,它接收3个参数,第一个为要绑定方法的对象,第二个为绑定的方法名称,第三个为需要绑定的方法(一个匿名函数)。函数体的的分析已经在注释里面了。

- //addMethod
- function addMethod(object, name, fn) {
- var old = object[name];
- object[name] = function() {
- if(fn.length === arguments.length) {
- return fn.apply(this, arguments);
- } else if(typeof old === "function") {
- return old.apply(this, arguments);
- }
- }
- }
- var people = {
- values: ["Dean Edwards", "Alex Russell", "Dean Tom"]
- };
- /* 下面开始通过addMethod来实现对people.find方法的重载 */
- // 不传参数时,返回peopld.values里面的所有元素
- addMethod(people, "find", function() {
- return this.values;
- });
- // 传一个参数时,按first-name的匹配进行返回
- addMethod(people, "find", function(firstName) {
- var ret = [];
- for(var i = 0; i < this.values.length; i++) {
- if(this.values[i].indexOf(firstName) === 0) {
- ret.push(this.values[i]);
- }
- }
- return ret;
- });
- // 传两个参数时,返回first-name和last-name都匹配的元素
- addMethod(people, "find", function(firstName, lastName) {
- var ret = [];
- for(var i = 0; i < this.values.length; i++) {
- if(this.values[i] === (firstName + " " + lastName)) {
- ret.push(this.values[i]);
- }
- }
- return ret;
- });
- // 测试:
- console.log(people.find()); //["Dean Edwards", "Alex Russell", "Dean Tom"]
- console.log(people.find("Dean")); //["Dean Edwards", "Dean Tom"]
- console.log(people.find("Dean Edwards")); //["Dean Edwards"]

好啦,实现JS函数重载就写这么多啦,如果亲们有更好的实现方法,欢迎评论交流哈~
[转载]浅谈JavaScript函数重载的更多相关文章
- 浅谈JavaScript函数重载
上个星期四下午,接到了网易的视频面试(前端实习生第二轮技术面试).面了一个多小时,自我感觉面试得很糟糕的,因为问到的很多问题都很难,根本回答不上来.不过那天晚上,还是很惊喜的接到了HR面电话.现在HR ...
- 浅谈javascript函数节流
浅谈javascript函数节流 什么是函数节流? 函数节流简单的来说就是不想让该函数在很短的时间内连续被调用,比如我们最常见的是窗口缩放的时候,经常会执行一些其他的操作函数,比如发一个ajax请求等 ...
- [转]浅谈javascript函数劫持
转自:Ph4nt0m Security Team 这么多年了,现在学习依然还是有很多收货,向前辈致敬.转载一方面是自己存档一份,另一方面是让更多喜欢安全的人一同学习. ================ ...
- 浅谈JavaScript 函数作用域当中的“提升”现象
在JavaScript当中,定义变量通过var操作符+变量名.但是不加 var 操作符,直接赋值也是可以的. 例如 : message = "hello JavaScript ! " ...
- 浅谈javascript函数,变量声明及作用域
javascript函数跟变量的声明.作用域这些概念网上都已经讲烂了. 这里写个博客,也相当于做个笔记. 变量声明 首先看个例子: var globalVar = "gv"; fu ...
- 浅谈javascript函数执行过程
javascript函数执行过程: 1. 为函数创建一个执行环境 2. 复制函数的 [[scopes]] 属性中的对象构建起执行环境的作用链域 3. 创建函数活动对象并推入执行环境作用链域的前端 4. ...
- 浅谈JavaScript函数
JavaScript作为一种基于对象(非严格面向对象)的语言,函数在JS中的地位非同一般:用函数声明类和对象.甚至函数本身也是对象. 一.函数的三种声明方式辨析. 1.function命令 funct ...
- 浅谈JavaScript中的闭包
浅谈JavaScript中的闭包 在JavaScript中,闭包是指这样一个函数:它有权访问另一个函数作用域中的变量. 创建一个闭包的常用的方式:在一个函数内部创建另一个函数. 比如: functio ...
- 浅谈JavaScript浮点数及其运算
原文:浅谈JavaScript浮点数及其运算 JavaScript 只有一种数字类型 Number,而且在Javascript中所有的数字都是以IEEE-754标准格式表示的.浮点数的精度问题 ...
随机推荐
- C#删除文件夹的文件
using System.IO; //判断文件是不是存在if(File.Exists(@"文件路径")){//如果存在则删除File.Delete(@"文件路径" ...
- Nginx动静分离-tomcat
一.动静分离 1.通过中间件将动态请求和静态请求分离. 2.为什么? 分离资源,减少不必要的请求消耗,减少请求延时. 3.场景 还可以利用php,fastcgi,python 等方式 处理动态请求 # ...
- TYPORA的使用手册
Typora可以根据当前文档的标题层级,自动生成并显示大纲,窗口的右下角并有字数显示. 1.标题的使用标题的使用格式# 一阶标题 或者快捷键Ctrl+1 ##二阶标题 或者快捷键Ctrl+2 ###三 ...
- angular6的响应式表单
1:在AppModule模块里面引入 ReactiveFormsModule 要使用响应式表单,就要从@angular/forms包中导入ReactiveFormsModule,并把它添加到你的NgM ...
- mongoexport导出记录到csv文件
root@service:~# mongoexport -d prod -c employees -f _id,platform,phone --csv -o /opt/employees.csv 2 ...
- a daemon 守护进程
w Cron and Crontab usage and exampleshttps://www.pantz.org/software/cron/croninfo.html
- oracle查询语句,根据中文的拼音排序
SELECT * FROM USER t ORDER BY nlssort(FIRSTNAME, 'NLS_SORT=SCHINESE_PINYIN_M')
- WebSocket知识、轮询、长轮询、长连接
一.WebSocket理论知识 1.什么是websocket WebSocket是HTML5新增的协议,它的目的是在浏览器和服务器之间建立一个不受限的双向通信的通道,比如说,服务器可以在任意时刻发送消 ...
- WPF使用Mutex创建单实例程序失效
vs2019 1.引入名称空间 using System.Threading; using System.Runtime.InteropServices; 2.导入dll并声明方法 [DllImpor ...
- 【转载】研发应该懂的binlog知识(上)
---------------------------------------------------------------------------------------------------- ...