10分鍾搞懂toString和valueOf函數(詳細版)


首先要說明的是這兩種方法是toPrimitive抽象操作里會經常用到的。

默認情況下,執行這個抽象操作時會先執行valueOf方法,如果返回的不是原始值,會繼續執行toString方法,如果返回的還不是原始值,那么會報錯,如果有指定轉換類型時,情況又會有所不同,詳細解析請繼續往下看。

(注意:valueOf和toString方法在Date,array等對象中有些是被重寫過的,所以不同對象調用此方法可能產生的操作不同,如果沒有這些方法,會調用最原始的Object.prototype上的valueOf和toString方法)

數據的轉換

所有對象繼承了兩個轉換方法:

第一個是toString(),它的作用是返回一個反映這個對象的字符串

第二個是valueOf(),它的作用是返回它相應的原始值

toString()

toString()可以看做是把一個數據轉換成了相應字符串的形式,安照這個轉換規則中

案例:
//返回相應的字符串 console.log( ({x:1, y:1 }).toString() ); // [object Object] console.log([1,2,3].toString()); // 1,2,3 console.log((function(x){f(x); }).toString()); //function (x){f(x); } console.log(/\d+/g.toString()); // /\d+/g console.log(new Date(2015,4,4).toString()); // Mon May 04 2015 00:00:00 GMT+0800 console.log(new Date(2015,4,4).valueOf()); // 1430668800000


valueOf()

 
        

每個JavaScript固有對象的 valueOf 方法定義不同。

 
        
對象 返回值
Array 數組沒有valueOf方法,繼承的是Object.prototype.valueOf的原始方法,返回數組本身
Boolean Boolean 值。
Date 存儲的時間是從 1970 年 1 月 1 日午夜開始計的毫秒數 UTC。
Function 同Array
Number 數字值。
Object 對象本身。這是默認情況。
String 字符串值。
 

Array,Function,Math 和 Error 對象沒有 valueOf 方法。

------------------------------------------------------------------------------------------------------------------------------

一般來說,對象到字符串的轉換經過了如下步驟:

1.如果對象具有toString()方法,則調用這個方法。如果它返回一個原始值,js將這個值轉換成字符串,並返還這個字符串結果。

2.如果對象沒有toString()方法,或者這個方法並不返回一個原始值,那么js將調用valueOf()方法。

3.否則,js無法從toString()或者valueOf()獲得一個原始值,因此這時它將拋出一個類型錯誤異常。

 

一般來說,對象到數字的轉換過程中,js做了同樣類似的事情,但這里它會首先嘗試使用valueOf()方法:

1.如果對象具有valueOf()方法,后者返回一個原始值,則js將這個原始值轉換成數字,並返回這個數字。

2.否則,如果對象具有toString()方法,后者返回一個原始值,則js將轉換並返回。

(首先js轉換成相應的字符串原始值,再繼續將這個原始值轉換成相應的數字類型,再返回數字)

3.否則,js拋出一個類型錯誤異常。

 

 對象通過toString或valueOf方法轉換為原始值,JS語言核心的內置類首先嘗試使用valueOf(),再嘗試使用toString()

一個小李子

“1” == true;

將返回true,轉換形式是:true首先轉換為1,然后再執行比較。接下來字符串“1”也轉換成了數字1,相等,所以返回true

另外如:

var str = new String('hello,world');
console.log(typeof str); //'object'
console.log(typeof str.valueOf()); //'string'

對於所有非日期對象來說,對象到原始值的轉換基本上是對象到數字的轉換

(首先調用valueOf,但日期對象則使用對象到字符串的轉換模式,但這種轉換只執行一次就立即被使用,不會像上面所說的那般 先轉成字符串再轉成相應的數字類型)

比如說,js中“+"運算符可以進行數學加法和字符串連接操作。

如果他它的其中一個操作數是對象,則js將使用特殊的方法將對象轉換成原始值,而不是使用其他算術運算符的方法執行對象到數字的轉換,”==“運算符類似

和”==“一樣,”<"與其他運算符也會做對象到原始值的轉換,但要出去日期對象的特殊情形

“-“減號運算符把兩個操作數都轉換成數字

比如:

var now = new Date();
console.log(now);  // Date {Sat Apr 04 2015 13:21:08 GMT+0800}
console.log(typeof (now+1));  //string
console.log((now+1));  //Sat Apr 04 2015 13:21:08 GMT+08001
console.log(typeof (now-1));  //number
console.log((now-1));  // 1428124868474
console.log(now == now.toString());  //true
console.log(now > now -1); //true
var date = new Date(); 
var date_string = date.toString(); 
var date_value = date.valueOf(); 
alert(date == date_string); //true
alert(date == date_value); //false
var aaa = { 
i: 10, 
valueOf: function() { return this.i+30; }, 
toString: function() { return this.valueOf()+10; } 
} 
alert(aaa > 20); // true 
alert(+aaa); // 40 
alert(aaa); // 50 

之所以有這樣的結果,因為它們偷偷地調用valueOf或toString方法。更進一步測試

var bbb = { 
i: 10, 
toString: function() { 
console.log('toString'); 
return this.i; 
}, 
valueOf: function() { 
console.log('valueOf'); 
return this.i; 
} 
} 
alert(bbb);// 10 toString 
alert(+bbb); // 10 valueOf 
alert(''+bbb); // 10 valueOf 
alert(String(bbb)); // 10 toString 
alert(Number(bbb)); // 10 valueOf 
alert(bbb == '10'); // true valueOf 
alert(bbb === '10'); // false

乍一看結果,大抵給人的感覺是,如果轉換為字符串時調用toString方法,如果是轉換為數值時則調用valueOf方法,但其中有兩個很不和諧。

一個是alert(''+bbb),字符串合拼應該是調用toString方法……另一個我們暫時可以理解為===操作符不進行隱式轉換,因此不調用它們。

為了追究真相,我們需要更嚴謹的實驗。

var aa = { 
i: 10, 
toString: function() { 
console.log('toString'); 
return this.i; 
} 
} 
alert(aa);// 10 toString 
alert(+aa); // 10 toString 
alert(''+aa); // 10 toString 
alert(String(aa)); // 10 toString 
alert(Number(aa)); // 10 toString 
alert(aa == '10'); // true toString 
再看valueOf。 
var bb = { 
i: 10, 
valueOf: function() { 
console.log('valueOf'); 
return this.i; 
} 
} 
alert(bb);// [object Object] 
alert(+bb); // 10 valueOf 
alert(''+bb); // 10 valueOf 
alert(String(bb)); // [object Object] 
alert(Number(bb)); // 10 valueOf 
alert(bb == '10'); // true valueOf 
發現有點不同吧?!它沒有像上面toString那樣統一規整。對於那個[object Object],我估計是從Object那里繼承過來的,我們再去掉它看看。 
Object.prototype.toString = null; 
var cc = { 
i: 10, 
valueOf: function() { 
console.log('valueOf'); 
return this.i; 
} 
} 
alert(cc);// 10 valueOf 
alert(+cc); // 10 valueOf 
alert(''+cc); // 10 valueOf 
alert(String(cc)); // 10 valueOf 
alert(Number(cc)); // 10 valueOf 
alert(cc == '10'); // true valueOf 

總結起來就是 如果只重寫了toString,對象轉換時會無視valueOf的存在來進行轉換。

但是,如果只重寫了valueOf方法,在要轉換為字符串的時候會優先考慮valueOf方法。在不能調用toString的情況下,只能讓valueOf上陣了

 [-_-]眼睛累了吧,注意勞逸結合呀[-_-]

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM