toString()和valueOf的区别

说明

  • toString()
  • valueOf()

这两个方法几乎出现在js中所有类型对应的内置对象的原型上(null和undefined除外),它们的出现解决了值运算的问题,但也出现了隐式转换这个让人头疼的问题。

toString()

返回一个表示当前值的特殊字符串,当对象表示为文本或以期望字符串被调用时,toString()方法会自动调用,比如用console.log()或者alert()方法输出值时,就会自动调用该方法,在进行值比较时,也可能触发该方法,也有可能触发valueOf方法,关于何时触发这两个方法,下文会进行介绍。

<script>
       <script>
        var str = "1";
        var num = 1;
        var boolean = true;
        var obj = {};
        var func = function() {}
        console.log(str.toString()); //1
        console.log(num.toString()); //1
        console.log(boolean.toString()); //true
        console.log(obj.toString()); //["object Object"]
        console.log(func.toString()); //function(){}
    </script>

js中许多的类型对象上都重写了Object原型上的toString方法,我们还发现,当对象调用toString()时,它会返回一个表示当前值类型的字符串,它直接调用了Object原型上的toString。

比较对象
typeof 可以用来判断基本类型变量的类型(null除外,null会返回object,这是js历史遗留问题),当我们用typeof来比较对象时,发现并行不通,因为它都会返回一个object,比如当我们运用深拷贝copy对象时,就不能准确地判断出数组和对象,并做出相应的处理,因为它们都会返回一个object。
object原型上的toString方法,可以返回一个表示值类型的一个字符串,那么我们就可以使用object原型上的toString方法来判断对象。

  <script>
        var toString = Object.prototype.toString

        toString.call(1) // '[object Number]'
        toString.call('1') // '[object String]'
        toString.call(false) // '[object Boolean]'
        toString.call(() => {}) // '[object Function]'
        toString.call([]) // '[object Array]'
        toString.call({}) // '[object Object]'
        toString.call(null) // '[object Null]'
        toString.call(undefined) // '[object Undefined]'
        toString.call(new Date()) // '[object Date]'
        toString.call(window) // '[object Window]'
        toString.call(Math) // '[object Math]'
    </script>

运算操作时带来的隐式转换问题

var c = [1, 2, 3]
var d = {a:2}
console.log(2 + 1)  // 3
console.log('s')    // 's'
console.log('s' + 2)  // 's2'
console.log(c < 2)  // false        (一次 => 'Array')
console.log(c + c)  // "1,2,31,2,3" (两次 => 'Array')
console.log(d > d)  // false        (两次 => 'Object')

数组原型上重写了toString方法,它会把数组用join(",")方法分割,输出字符串。
toString()重写

    <script>
        String.prototype.toString = function() {
            var num = 1;
            return `myString${num}`;
        }
        var str = "2"
        console.log(str.toString()); //myString1
    </script>

valueOf()

返回当前包装对象的原始值。
这个 API 通常是由 JavaScript 引擎内部调用,开发者一般情况下不会使用,其功能和 toString 大同小异。
二者区别
共同点:两者都是在值运算的时候会自动调用,从而触发隐式转换
不同点:默认返回值不同,并且存在调用优先级关系
在二者并存的情况下,数值运算时优先使用 valueOf ,字符运算时优先使用 toString 。

class A {
  construtor(num) {
  this.num = num
  }
  
  valueOf() {
    return this.num
  }
  
  toString() {
    return `我好有钱:${this.num}`
  }
}

const a = new A(100)

console.log(String(a)) // '我好有钱:100'             -> toString
console.log(Number(a)) // 100                       -> valueOf
console.log(a + '22') // '10022'          -> valueOf
console.log(a == 100) // true            -> valueOf
console.log(a === 100) // false,全等不触发隐式转换

总结
总结可以发现,valueOf 更加偏向于运算,toString 更加偏向于调用。
在进行对象运算时,将优先调用 toString 方法,如若没有重写 toString 方法,则会调用 valueOf 方法;如果两个方法都没有重写,则会调用 Object 上面的 toString
当进行强制类型转换时,如果转换成字符串则会调用 toString ,转换成数字则会调用 valueOf
使用运算符进行运算时,valueOf 的优先级高于 toString。
原文链接

posted @ 2022-10-23 20:53  含若飞  阅读(119)  评论(0编辑  收藏  举报