手写简单call,apply,bind
分析一下call的使用方法:call是显示绑定this指向,然后第一个参数是你所指向的this对象,后面跟着多个参数,以逗号隔开
function sum(num1,num2){
return num1 + num2
}
sum.call({},1,2) // 3
上面是一个最简单的call使用方法,call的第一个参数,可以是对象,可以是数字,可以是空对象
当call的第一个参数是undefined或者是null时,他的指向是window
接下来就一步步的开始手写一个简单的call
首先第一步,我们需要在Function内添加一个原型对象hycall
Function.prototype.hycall=function(){
console.log('我是新的hycall对象');
}
sum.hycall() // 我是新的hycall对象
然后我们需要在内部执行sum的函数,但是由于这个hycall是通用式,不能写死内部的函数调用
这时候会有个问题:如何知道是谁调用了hycall??
其实很清楚的一个事情,对于this指向熟悉的都知道,sum.hycall是一个隐式绑定,此时的hycall内部的this就是当前的sum对象,我们用一个参数去接收this,再将其运行即可
Function.prototype.hycall=function(thisArg){
var fn = this // 外部的对象 sum,或者{}等等
fn()
}
function foo(){
console.log('我是foo对象',this);
}
sum.hycall({})
这里实现了一个最简单的call调用方式,接下来第二步,call的第一个参数是存在this指向的,此时的hycall需要接收一个参数对象thisArg,又因为这个参数是this的指向
问题二:如何让传来的this指向绑定到对象上?
对于传来的this指向,可以给其本身添加一个函数对象,再让其自身运行,相当于间接性的调用了函数
Function.prototype.hycall=function(thisArg){
var fn = this // 外部的对象 sum,或者{}等等
thisArg.fn=fn // thisArg身上添加一个fn
thisArg.fn() // 再将其调用即可
delete thisArg.fn // 调用完成之后再将其删除即可
}
function foo(){
console.log('我是foo对象',this);
}
sum.hycall({})
这里看似完成了对this参数的调用,但是此处又有问题出现:call是可以传入number等参数进入的,但是代码中不可能给一个数字添加函数例如123.fn(),此处又该如何解决?
这里有一个例子,当123要变成对象类型时,只需要new Number(123) ,便成为了一个对象。
以上例子说明,我们可以对thisArg进行对象转化
thisArg=Object(thisArg)
再考虑到call传入undefined或者null的时候,指向的this是window
我们对thisArg做出判断
thisArg=(thisArg !== null && thisArg !== undefined) ? thisArg=Object(thisArg) : window
第三步,对于传值,call除了能传递this指向,后方还可以跟多个传值,对于我们的hycall的接收,此处可以使用es6的rest参数语法 ...
Function.prototype.hycall=function(thisArg,...args){
var fn = this // 外部的对象 sum,或者{}等等
thisArg=(thisArg !== null && thisArg !== undefined) ? thisArg=Object(thisArg) : window // 判断thisArg是否存在,存在将其转换,不存在让其指向window
thisArg.fn=fn // thisArg身上添加一个fn
thisArg.fn(...args) // 再将其调用即可 spread--展开运算符
delete thisArg.fn // 调用完成之后再将其删除即可
}
function sum(a,b){
return a+b
}
sum.hycall({},1,2)
最后一步,接收返回值并返回出去
Function.prototype.hycall=function(thisArg,...args){
var fn = this // 外部的对象 sum,或者{}等等
thisArg=(thisArg !== null && thisArg !== undefined) ? thisArg=Object(thisArg) : window // 判断thisArg是否存在,存在将其转换,不存在让其指向window
thisArg.fn=fn // thisArg身上添加一个fn
var result = thisArg.fn(...args) // 再将其调用即可 spread--展开运算符
delete thisArg.fn // 调用完成之后再将其删除即可
return result
}
function sum(a,b){
return a+b
}
sum.hycall({},1,2)
以上是对call的一个简单的手写,如果对call的手写熟悉了,apply和bind其实也大同小异,apply第一个参数和call一样,但是第二个参数是以数组的方式进行传递
function sum(num1,num2){
console.log("sum被调用",this,num1,num2);
return num1+num2
}
// 系统调用
var result=sum.apply("abc",[20,30])
console.log(result);
上方是一个正常的系统调用,apply第二个参数不传值时会和call有些不一样
Function.prototype.hyapply=function(thisArg,argArray){
var fn = this // 外部的对象 sum,或者{}等等
thisArg=(thisArg !== null && thisArg !== undefined) ? thisArg=Object(thisArg) : window // 判断thisArg是否存在,存在将其转换,不存在让其指向window
argArray=argArray || []
thisArg.fn=fn // thisArg身上添加一个fn
var result = thisArg.fn(...argArray) // 再将其调用即可 spread--展开运算符
delete thisArg.fn // 调用完成之后再将其删除即可
return result
}
function sum(a,b){
return a+b
}
sum.hyapply({},1,2)
这里又有人要提问了:为什么call那里就可以不用判断第二个参数的值,你这里就需要呢?
因为在hycall中的第二个参数是...args 倘若我们未传值过来,...args的值=[],所以不需要对call的第二参数做出判断
下面开始对bind进行分析,bind是以返回一个函数的方法进行使用的
function sum(num1,num2,num3,num4){
console.log(num1,num2,num3.num4)
}
function bar(){
console.log("我是bar")
}
// 方式一
var newBar=bar.bind("aaa")
newBar()
// 方式二
var newSum=sum.bind("aaa",10,20,30,40)
newSum()
// 方式三
var newSum=sum.bind("aaa",10)
newSum(20,30,40)
手写bind函数
Function.prototype.hybind=function(thisArg,...argArray){
var fn = this // 外部的对象 sum,或者{}等等
thisArg=(thisArg !== null && thisArg !== undefined) ? thisArg=Object(thisArg) : window
function proxyFn(){
thisArg.fn=fn
var result=thisArg.fn()
deletr thisArg.fn
return result
}
return proxyFn
}
function sum(a,b){
return a+b
}
var newSum=sum.hybind("aaa")
newSum()
接下来需要考虑上方的一种情况
// 方式三
var newSum=sum.bind("aaa",10)
newSum(20,30,40)
由于我们的hybind只有两个参数,但你传第三个参数的时候,就需要内部调用
Function.prototype.hybind=function(thisArg,...argArray){
var fn = this // 外部的对象 sum,或者{}等等
thisArg=(thisArg !== null && thisArg !== undefined) ? thisArg=Object(thisArg) : window
function proxyFn(...args){ //此处接收第三个参数
thisArg.fn=fn
var finalArgs=[...argArray,...args]//拼接两个数组
var result=thisArg.fn(...finalArgs)
deletr thisArg.fn
return result
}
return proxyFn
}
function sum(a,b){
return a+b
}
var newSum=sum.hybind("aaa",10)
newSum(20,30,40)
以上就是对三个显示绑定的简单手写,其实对于上面的边界判断其实还不是很完善,例如=0的时候(优化边界判断),例如fn本身就存在然后给覆盖(使用ES6-symbol语法定义解决)等等
如有更好的意见和完善的地方欢迎补充留言,感谢coderwhy老师的指导
手写简单call,apply,bind的更多相关文章
- 不使用Tomcat,手写简单的web服务
背景: 公司使用的YDB提供了http的查询数据库服务,直接通过url传入sql语句查询数据-_-||.ydb的使用参照:https://www.cnblogs.com/hd-zg/p/7115112 ...
- 手写call、apply、bind
区别&联系 三者都是指定函数执行时的上下文,第一个参数都是上下文: call从第二个参数开始,后续所有的参数传递给函数执行: apply第二个参数是一个数组,传递给函数执行: bind返回一个 ...
- 手写简单的jq雪花飘落
闲来无事,准备写个雪花飘落的效果,没有写太牛逼的特效,极大的简化了代码量,这样容易读取代码,用起来也很简单,对于那些小白简直是福利啊,简单易读易学.先直接上代码吧,然后再一一讲解,直接复制粘贴就可以拿 ...
- 利用Java手写简单的httpserver
前言: 在看完尚学堂JAVA300中讲解如何实现一个最简单的httpserver部分的视频之后, 一.前置知识 1.HTTP协议 当前互联网网页访问主要采用了B/S的模式,既一个浏览器,一个服务器,浏 ...
- 手写简单PE
环境工具:Windows 10 010Editor 目标程序功能: 调用MessageBoxA弹出消息框. 1.构造DOS头 typedef struct _IMAGE_DOS_HEADER { // ...
- 通过反射来手写简单的ORM SQlserver
不说废话,直接上干货,如发现问题,欢迎大家指出,谢谢! //------------------------------------MySQlServerORM [简单 CURD] using Sys ...
- 手写简单的promise
function Promise(fn) { var that = this; this.status = "pedding"; this.value = undefined; / ...
- node搭环境(四)--webpack启服务运行VUE模块文件(手写简单脚手架)
webpack启服务步骤: 1.新建空文件夹webpack-vue.在空文件夹右键点击- GIt Bath here--输入cnpm init--按程序走完会生成package.json文件 2.打开 ...
- 手写简单的php生成Html网页
这个是基本功,以前用到laravel及thinkphp时,这一步,都被设置好了吧. 这里只依靠纯的php环境,而没有任何框架, 而框架,只是将这一切规范化,加快代码效率及减小沟通成本,维护升级也方便, ...
随机推荐
- CS5212Capstone|DP转VGA方案设计芯片|CS5212设计资料
CS5212是台湾CAPSTONE最新开发出一款用于DP转VGA方案设计芯片,其周围器件较少,设计简单,整体BOM成本低,市场性价比较高. CS5212内置MCU是工业标准8051核心单片机处理器.C ...
- .NET6: 三分钟搭建WPF三维应用
要运行本文中的示例,请先安装Vistual Studio 2022,社区版就可以了. 1 创建项目 选择创建WPF应用 给程序起一个酷酷的名字,选一个酷酷的位置: 选一下.NET6 2 配置项目 从n ...
- Java中的关键字有哪些?「Java中53个关键字的意义及使用方法」
Java中的关键字有哪些? 1)48个关键字:abstract.assert.boolean.break.byte.case.catch.char.class.continue.default.do. ...
- Mysql 设计超市经营管理系统,包括员工信息表(employee)和 员工部门表(department)
互联网技术学院周测机试题(二) 一.需求分析 为进一步完善连锁超市经营管理,提高管理效率,减少管理成本,决定开发一套商品管理系统,用于日常的管理.本系统分为商品管理.员工管理.店铺管理,库存管理等功能 ...
- 初识MASA Blazor
MASA Blazor是一个Blazor的UI组件库.就像大家写前端熟知的Bootstrap, Ant Design一样. MASA Blazor官网地址:https://blazor.masasta ...
- spring security 动态 修改当前登录用户的 权限
1.前言 spring security 可以获取当前登录的用户信息,同时提供了接口 来修改权限列表信息 , 使用这个方法 ,可以动态的修改当前登录用户权限. 那么问题来了... 如果我是管理员 ,如 ...
- centos7 alias别名永久生效
进入/etc/profile.d/目录 cd /etc/profile.d/ 在profile.d目录随意创建一个sh文件,例如alias_test.sh vi alias_test.sh##里面的内 ...
- CentOS6.9部署Redis3.2.9+FastDFS_4.06+Nginx1.5.0
CentOS6.9部署Redis3.2.9+FastDFS_4.06+Nginx1.5.0 原文链接:https://www.toutiao.com/i6481931577499582990/ 一.上 ...
- PAT 乙级 1002. 写出这个数 (20)(C语言描述)
读入一个自然数n,计算其各位数字之和,用汉语拼音写出和的每一位数字. 输入格式:每个测试输入包含1个测试用例,即给出自然数n的值.这里保证n小于10100. 输出格式:在一行内输出n的各位数字之和的每 ...
- kibana7.x安装配置操作elasticsearch
什么是Kibana? Kibana是一个基于Node.js的Elasticsearch索引库数据统计工具,可以利用Elasticsearch的聚合功能,生成各种图表,如柱形图,线状图,饼图等. 而且还 ...