JS中令人發(fā)指的valueOf方法介紹
彭老濕近期月報里提到了valueOf方法,興致來了翻了下ECMA5里關于valueOf方法的介紹,如下:
15.2.4.4 Object.prototype.valueOf ( )
When the valueOf method is called, the following steps are taken:
1. Let O be the result of calling ToObject passing the this value as the argument.
2. If O is the result of calling the Object constructor with a host object (15.2.2.1), then
a. Return either O or another value such as the host object originally passed to the constructor. The specific result that is returned is implementation-defined.
3. Return O.
規(guī)范里面的對于valueOf的解釋很短,大致為:調用ToObject方法(一個抽象方法,后面會講到),并將this的值作為參數(shù)傳入。
針對調用ToObject時傳入的不同參數(shù)(this),返回值分別如下:
1、this為宿主對象時,返回值取決于瀏覽器的實現(xiàn),即不同瀏覽器的返回可能不同(關于宿主對象,可參考http://www.w3school.com.cn/js/pro_js_object_types.asp)
2、this不是宿主對象,則返回ToObject(this)的值
參數(shù)類型 | 返回結果 |
Undefined | 拋出TypeError異常 |
Null | 拋出TypeError異常 |
Number | 創(chuàng)建一個Number對象,它內部的初始值為傳入的參數(shù)值 |
String | 創(chuàng)建一個String對象,它內部的初始值為傳入的參數(shù)值 |
Boolean | 創(chuàng)建一個Boolean對象,它內部的初始值為傳入的參數(shù)值 |
Object | 返回傳入的參數(shù)(無轉換) |
根據(jù)Object.prototype.valueOf的定義,以及抽象方法ToObject的描述,可得下表
obj類型 | Object.prototype.valueOf.call(obj)返回結果 |
Undefined | 拋出TypeError異常 |
Null | 拋出TypeError異常 |
Number | Number類型的對象,值等于obj |
String | String類型的對象,值等于obj |
Boolean | Boolean類型的對象,值等于obj |
Object | obj對象本身 |
舉幾個具體的例子:
var num = 123;
console.log(num.valueOf()); //輸出:123
console.log(num.valueOf()); //輸出:'number'
var unde = undefined;
console.log(Object.prototype.valueOf.call(unde)); //輸出:'TypeError: Cannot convert null to object'
var obj = {name:'casper'};
var linkObj = obj.valueOf();
linkObj.name = 'change';
console.log(linkObj.name); //輸出:'change' ...說明obj.valueOf()返回的是對象自身
實際上,上面沒有提到Array、Function對象,根據(jù)下面代碼可以猜想,當Object.prototype.valueOf調用時,參數(shù)為Array、Function類型的對象時,返回的結果也為對象自身:
var arr = [1, 2 ,3];
var linkArr = arr.valueOf();
linkArr[0] = ['casper'];
console.log(linkArr); //輸出:['casper', 2, 3]
var foo = function(){ return 1; };
var linkFoo = foo.valueOf();
linkFoo.test = 'casper';
console.log(linkFoo.test); //輸出:'casper'
看完上面的描述,是不是有種恍然大悟的感覺?如果是的話,恭喜你,可能你跟我一樣其實還沒完全理解透徹。
簡單舉個例子,當調用Object.prototype.valueOf的對象為數(shù)值類型時,假設該對象是名稱為num,num很有可能通過下面兩種方式聲明:
var num = 123; //通過對象字面量聲明console.log(typeof num); //輸出:'number'
var num = new Number(123); //通過構造方法聲明console.log(typeof num); //輸出:'object'
關于返回值的說明,ECMA5里面原文如下:
Create a new Number object whose [[PrimitiveValue]] internal property is set to the value of the argument. See 15.7 for a description of Number objects.
按照這段文字的說明,似乎num.valueOf()返回的應該是個Number對象(非字面量聲明的那種),但實際上:
var num = 123;
var tmp = num.valueOf();
console.log(typeof tmp); //輸出: 'number'
5.7.4.4 Number.prototype.valueOf ( )
Returns this Number value.
The valueOf function is not generic; it throws a TypeError exception if its this value is not a Number or a Number object. Therefore, it cannot be transferred to other kinds of objects for use as a method.
原來Number有屬于自身的原型valueOf方法,不是直接從Object.prototype上繼承下來,類似的,Boolean、String也有自己的原型valueOf方法,歸納如下:
類型 | 是否有屬于自己的原型valueOf方法 |
Undefined | 無 |
Null | 無 |
Number | 有,Number.prototype.valueOf |
String | 有,String.prototype.valueOf |
Boolean | 有,Boolean.prototype.valueOf |
Object | - |
此處之外,Array、Function并沒有自己的原型valueOf方法,見規(guī)范說明:
NOTE The Array prototype object does not have a valueOf property of its own; however, it inherits the valueOf property from the standard built-in Object prototype Object.
The Function prototype object does not have a valueOf property of its own; however, it inherits the valueOf property from the Object prototype Object.
補充說明:Number.prototype.valueOf的內部轉換規(guī)則比想的要略復雜些,此處不展開。
啰啰嗦嗦說了一大通,現(xiàn)在還有兩個問題存在疑惑:
1.關于ToObject,當參數(shù)為Function對象時,返回對象作何處理似乎沒見到規(guī)范里明確說明,當前僅靠實驗猜測(也有可能是我沒找到)
2.valueOf的使用場景,實際開發(fā)中尚未見到有兄弟用過
最后的最后:
文中示例如有錯漏,請指出;如覺得文章對您有用,可點擊“推薦” :)
相關文章
JS為什么說async/await是generator的語法糖詳解
這篇文章主要給大家介紹了關于JS為什么說async/await是generator的語法糖的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用JS具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧2019-07-07JavaScript計算字符串中特定字符出現(xiàn)次數(shù)的實例詳解
在JavaScript編程中,經常會遇到需要計算字符串中特定字符出現(xiàn)次數(shù)的情況,在本文中,我將分享兩個簡單的JavaScript函數(shù),用于計算字符串中特定字符出現(xiàn)的次數(shù),需要的朋友可以參考下2023-11-11javascript實現(xiàn)動態(tài)導入js與css等靜態(tài)資源文件的方法
這篇文章主要介紹了javascript實現(xiàn)動態(tài)導入js與css等靜態(tài)資源文件的方法,基于回調函數(shù)實現(xiàn)該功能,具有一定參考借鑒價值,需要的朋友可以參考下2015-07-07JavaScript實現(xiàn)動態(tài)添加Form表單元素的方法示例
這篇文章主要介紹了JavaScript實現(xiàn)動態(tài)添加Form表單元素的方法,結合實例形式分析了javascript表單元素操作相關函數(shù)使用方法與相關注意事項,需要的朋友可以參考下2017-08-08