JS中 toString() & valueOf()


數據的轉換

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

第一個是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 數組的元素被轉換為字符串,這些字符串由逗號分隔,連接在一起。其操作與 Array.toString 和 Array.join 方法相同。
Boolean Boolean 值。
Date 存儲的時間是從 1970 年 1 月 1 日午夜開始計的毫秒數 UTC。
Function 函數本身。
Number 數字值。
Object 對象本身。這是默認情況。
String 字符串值。

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

 

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

 

更詳細了使用例子:

(摘自:http://www.jb51.net/article/32327.htm

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