javascript中你可能遇到的隐式调用
前言
不知道用隐式调用来形容是否确切,其行为总是隐藏在背后,时不时出来露脸一下,作用貌似不大,但是了解一下还是有用处的,保不准在你的使用下大有作为。
所谓的隐式调用简单来说就是自动调用一些方法,而这些方法像钩子一样可以在外部修改,从而改变既定行为。
下面我会列举一些最近看到的隐式调用,例子都是点到即止,欢迎补充
数据类型转换 toSting 和 valueOf
var obj = {
a: 1,
toString: function () {
console.log('toString')
return '2'
},
valueOf: function () {
console.log('valueOf')
return 3
}
}
console.log(obj == '2'); //依次输出 'valueOf' false
console.log(String(obj));//依次输出 'toString' '2'
var obj = {
a: 1,
toString: function () {
console.log('toString')
return '2'
},
valueOf: function () {
console.log('valueOf')
return {} //修改为对象
}
}
console.log(obj == '2'); //依次输出 'valueOf' 'toString' true
console.log(Number(obj));//依次输出 'valueOf' 'toString' 2
在相等运算符的操作中,对象会先调用 valueOf 如果返回的值是一个对象, 就会调用 toSting, null除外,然后用返回的值进行比较,第一个例子 相当于 3 == '2' 输出 false, 第二个例子由于执行valueOf 返回的是一个对象, 然后执行 toString, 最后相当于 '2' == '2' 输出true
在 Number 和 String 方法中 Number会先调用 valueOf, 后调用 toString, String方法中是相反的。
数据类型的转换除了上面的两个例子外,还存在在各种其他操作中,如数值运算,当涉及到引用类型时,都会调用valueOf 或 toString 方法,只要是对象都会继承这两个方法,我们可以重新覆盖这两个方法,从而影响数据类型转换的行为
DOM2事件中的 handleEvent
var eventObj = {
a: 1,
handleEvent: function (e) {
console.log(this, e);//返回 eventObj 和 事件对象
alert(this.a)
}
}
document.addEventListener('click', eventObj)
你没有看错,addEventListener 第二个参数除了函数外还可以是一个对象, 事件触发后会执行对象的handleEvent方法,方法执行时的this指向eventObj, 你可以把想传入的数据绑定在eventObj对象上
JSON对象 toJSON
var Obj = {
a: 10,
toJSON: function () {
return {
a: 1,
b: function () {
},
c: NaN,
d: -Infinity,
e: Infinity,
f: /\d/,
g: new Error(),
h: new Date(),
i: undefined,
}
}
}
console.log(JSON.stringify(Obj));
//{"a":1,"c":null,"d":null,"e":null,"f":{},"g":{},"h":"2018-02-09T19:29:13.828Z"}
如果JSON的stringify方法传入的对象有toJSON方法,那么该方法执行的对象会转为toJSON执行后返回的对象,有一点要注意的是,如下面代码
var Obj1 = {
a: 10,
toJSON: function () {
console.log(this === Obj1);//true
return this
}
}
console.log(JSON.stringify(Obj1));//{"a":10}
var Obj2 = {
a: 10,
toJSON: function () {
console.log(this === Obj2);//true
return {
a: this
}
}
}
console.log(JSON.stringify(Obj2));//报错 Maximum call stack size exceeded
如果按上面的说法很明显报错是我们所预期的,但是当直接 return this 根本没有报错,不妨可以大胆猜测一下其内部对toJSON返回的对象和原对像进行比较,如果相等就直接使用原对象
promise对象的 then
var obj = {
then: function (resolve, reject) {
setTimeout(function () {
resolve(1000);
}, 1000)
}
}
Promise.resolve(obj).then(function (data) {
console.log(data);// 延迟1秒左右输出 1000
})
当Promise.resolve方法传入对象时,如果存在 then 方法会立即执行then方法,相当于把方法放入new Promise中,除了Promise.resolve有这个行为外,Promise.all也有这个行为
var timePromise = function (time) {
return new Promise(function (resolve) {
setTimeout(function () {
resolve(time);
}, time)
})
}
var timePromise1 = timePromise(1000);
var timePromise2 = timePromise(2000);
var timePromise3 = timePromise(3000);
Array.prototype.then = function (resolve) {
setTimeout(function () {
resolve(4000);
}, 4000)
}
Promise.all([timePromise1, timePromise2, timePromise3]).then(function (time) {
console.log(time);// 等待4秒左右输出 4000
})
对象属性存取器 get 和 set
var obj = {
_age: 100,
get age() {
return this._age < 18 ? this._age : 18;
},
set age(value) {
this._age = value;
console.log(`年龄设置为${value}岁`);
}
}
obj.age = 1000; //年龄设置为1000岁
obj.age = 200; //年龄设置为200岁
console.log(obj.age);// 18
obj.age = 2; ////年龄设置为2岁
console.log(obj.age); // 2
可以看到不管把年龄设置为多少,我的年龄都是18岁或以下的,当执行属性存取时实际上是调用对象属性相应的 get set函数,除了以上写法还有下面的写法
var input = document.createElement('input');
var span = document.createElement('span');
document.body.appendChild(input);
document.body.appendChild(span);
var obj = {
_age:''
}
var obj = Object.defineProperty(obj, 'age', {
get: function () {
return this._age;
},
set: function (value) {
this._age = value;
input.value = value;
span.innerHTML = value;
}
});
input.onkeyup = function (e) {
if (e.keyCode === 37 || e.keyCode === 39) {
return;
}
obj.age = this.value
}
现在input的value值和obj.age的属性值span的innerHTML值都绑定在一起了
遍历器接口 Symbol.iterator
var arr = [ 1, 2 , 3];
arr[Symbol.iterator] = function () {
const self = this;
let index = 0;
return {
next () {
if(index < self.length) {
return {
value: self[index] ** self[index++],
done: false
}
} else {
return {
value: undefined,
done: true
}
}
}
}
}
console.log([...arr, 4]);//返回 [1, 4, 27, 4]
for(let value of arr) {
console.log(value); //依次返回 1 4 27
}
可以看到凡是调用扩展运算符,或者使用 for...of 循环遍历对象都会调用对象的遍历器接口,像Array,String,Map,Set,TypedArray还有一些类数组对象如arguments,NodeList原生具备了遍历器接口,而普通对象没有部署这个接口,如果要让对象能够使用扩展运算符或者for...of循环可以在对象上添加该方法,也可以在原来具备接口的对象上重写方法,从而改变其行为
最后
暂时就想到这么多,欢迎大家补充
javascript中你可能遇到的隐式调用的更多相关文章
- JavaScript中valueOf、toString的隐式调用
今天在群上有人问这样一个问题: 函数add可以实现连续的加法运算函数add语法如下add(num1)(num2)(num3)...;//注意这里是省略号哟,无限使用举例如下:add(10)(10)=2 ...
- JavaScript隐藏的坑一,隐式调用toString
最近在重新学习JavaScript,看动态原型对象的时候,打印了两个用同一个构造函数生成的对象,但是打印结果却不一样,请看代码: var box1=new Box(); console.log(box ...
- 「译」JavaScript 的怪癖 1:隐式类型转换
原文:JavaScript quirk 1: implicit conversion of values 译文:「译」JavaScript 的怪癖 1:隐式类型转换 译者:justjavac 零:提要 ...
- Intent(二)隐式调用intent
在上一节我们一起学习了显示调用Intent,这一节我们来学习如何隐式调用Ingtent.有了这个我们就可以调用其他的线程,或者程序,可以让我们的应用程序变得多彩,如打开网页,拨打电话等. 接下来让我们 ...
- Windows下动态库的隐式调用
多年的工作经验告诉我Windows下使用动态库最简单的方法:使用def导出函数,然后隐式调用. 具体做法如下: (1)首先使用visual studio 创建“Win32项目”,如下图: (2)然后在 ...
- C++类构造函数、拷贝构造函数、复制构造函数、复制构造函数、构造函数显示调用和隐式调用
一. 构造函数是干什么的 class Counter { public: // 类Counter的构造函数 // 特点:以类名作为函数名,无返回 ...
- C++ 中operator用法:隐式类型转换
[转]C++ operator两种用法 C++,有时它的确是个耐玩的东东,就比如operator,它有两种用法,一种是operator overloading(操作符重载),一种是operator c ...
- C++函数模板的显示调用与隐式调用
C++函数模板可以显示调用与可以隐式调用 首先定义函数模板: template <class T> inline const T& c_max (const T& a, c ...
- JavaScript的六种数据类型与隐式转换
一.六种数据类型 javascript的数据类型包括: (1)基本数据类型:number.string.boolean.null.undefined (2)对象:object object又包括Fun ...
随机推荐
- tap点击一次,内部程序执行两次,多次
调试过程发现,使用 $(document).on('tap', '.children2', function () { //内部程序 }) 点击children2的时候,程序在里面执行了两次.百度得到 ...
- 直击LG曲面OLED首发现场,高端品质更出众
简直是太棒了,我可以去看LG曲面OLED电视新品发布会了.这可是LG向中国首次推出的曲面OLED电视.在网上我就已经看到其实曲面OLED电视已经在韩国.美国还有欧洲都上市了,听说现在反响还挺不错.真没 ...
- 阿里巴巴IconFont的使用方式
一.解释一下为什么要使用IconFont? IconFont顾名思义就是把图标用字体的方式呈现. 其优点在于以下几个方面: 1.可以通过css的样式改变其颜色:(最霸气的理由) 2.相对于图片来说,具 ...
- Ribbon XML Editor 2019.01.23-Setup.zip(支持64位)
RibbonXMLEditor是一款用于书写和测试customUI XML代码的国产工具,具有XML验证.Office文档压入.回调函数查询等多种功能. 可作为定制Office自定义界面有关程序开发的 ...
- Xcode查看iOS崩溃与崩溃日志分析
一.造成崩溃的原因 1.代码中存在bug 2.Watchdog 超时机制 3.用户强制退出 4.低内存终止 5.其他违法系统规则的操作,大部分是内存问题 二.崩溃的类型 1.信号错误类 (1)EXC_ ...
- python3下scrapy爬虫(第十三卷:scrapy+scrapy_redis+scrapyd打造分布式爬虫之配置)
之前我们的爬虫都是单机爬取,也是单机维护REQUEST队列, 看一下单机的流程图: 一台主机控制一个队列,现在我要把它放在多机执行,会产生一个事情就是做重复的爬取,毫无意义,所以分布式爬虫的第一个难点 ...
- IDEA Maven项目中添加tomcat没有无artifact选项
IntelliJ使用 ##使用IntelliJ IDEA配置web项目时,选择Edit Configration部署Tomcat的Deployment可能会出现以下情况: 导致新手部署过程中摸不着头脑 ...
- iOS 去掉导航栏最下面线的方法
导航栏透明,但是字体就不显示了,所以不可行. 下面两种方法是让导航栏下面的线不显示. // 方法1: [[self.navigationController.navigationBar.subview ...
- python语法基础-面向对象-基础-长期维护
############### 类的基本操作 ############## """ 类的基本认识: 1,类就是一个模子 2,dict,list都是类,具体的一 ...
- Ubuntu16.04使用sublime text3编写C语言后,实现编译并自动调用bash终端运行程序
实现编译并自动调用bash运行程序只需要新建自己的.build文件就OK 依次打开: tools->building system->new building system 后,把下面的内 ...