HTML5技术

一道面试题引发的对javascript类型转换的思考 - ChokCoco(2)

字号+ 作者:H5之家 来源:H5之家 2017-03-06 17:00 我要评论( )

var obj = {valueOf: function() {console.log('调用 valueOf');return {};},toString: function() {console.log('调用 toString');return 10;}}console.log(obj + 1); // 调用 valueOf// 调用 toString// 11 var o

var obj = { valueOf: function() { console.log('调用 valueOf'); return {}; }, toString: function() { console.log('调用 toString'); return 10; } } console.log(obj + 1); // 调用 valueOf // 调用 toString // 11

var obj = { valueOf: function() { console.log('调用 valueOf'); return {}; }, toString: function() { console.log('调用 toString'); return {}; } } console.log(obj + 1); // 调用 valueOf // 调用 toString // Uncaught TypeError: Cannot convert object to primitive value

  

Boolean 转换

什么时候会进行布尔转换呢:

  • 布尔比较时
  • if(obj) , while(obj) 等判断时
  • 简单来说,除了下述 6 个值转换结果为 false,其他全部为 true:

    Boolean(undefined) // false Boolean(null) // false Boolean(0) // false Boolean(NaN) // false Boolean('') // false

     

    Function 转换

    好,最后回到我们一开始的题目,来讲讲函数的转换。

    我们定义一个函数如下:

    function test() { var a = 1; console.log(1); }

    如果我们仅仅是调用 test 而不是 test() ,看看会发生什么?

    image


    可以看到,这里把我们定义的 test 函数的重新打印了一遍,其实,这里自行调用了函数的 valueOf 方法:

    image

    我们改写一下 test 函数的 valueOf 方法。

    test.valueOf = function() { console.log('调用 valueOf 方法'); return 2; } test; // 输出如下: // 调用 valueOf 方法 // 2

    与 Number 转换类似,如果函数的 valueOf 方法返回的不是一个原始类型,会继续找到它的 toString 方法:

    test.valueOf = function() { console.log('调用 valueOf 方法'); return {}; } test.toString= function() { console.log('调用 toString 方法'); return 3; } test; // 输出如下: // 调用 valueOf 方法 // 调用 toString 方法 // 3

     

    破题

    再看回我正文开头那题的答案,正是运用了函数会自行调用 valueOf 方法这个技巧,并改写了该方法。我们稍作改变,变形如下:

    function add () { console.log('进入add'); var args = Array.prototype.slice.call(arguments); var fn = function () { var arg_fn = Array.prototype.slice.call(arguments); console.log('调用fn'); return add.apply(null, args.concat(arg_fn)); } fn.valueOf = function () { console.log('调用valueOf'); return args.reduce(function(a, b) { return a + b; }) } return fn; }

    当调用一次 add 的时候,实际是是返回 fn 这个 function,实际是也就是返回 fn.valueOf();

    add(1); // 输出如下: // 进入add // 调用valueOf // 1

    其实也就是相当于:

    [1].reduce(function(a, b) { return a + b; }) // 1

    当链式调用两次的时候:

    add(1)(2); // 输出如下: // 进入add // 调用fn // 进入add // 调用valueOf // 3

    当链式调用三次的时候:

    add(1)(2)(3); // 输出如下: // 进入add // 调用fn // 进入add // 调用fn // 进入add // 调用valueOf // 6

    可以看到,这里其实有一种循环。只有最后一次调用才真正调用到 valueOf,而之前的操作都是合并参数,递归调用本身,由于最后一次调用返回的是一个 fn 函数,所以最终调用了函数的 fn.valueOf,并且利用了 reduce 方法对所有参数求和。

    除了改写 valueOf 方法,也可以改写 toString 方法,所以,如果你喜欢,下面这样也可以:

    function add () { var args = Array.prototype.slice.call(arguments); var fn = function () { var arg_fn = Array.prototype.slice.call(arguments); return add.apply(null, args.concat(arg_fn)); } fn.toString = function() { return args.reduce(function(a, b) { return a + b; }) } return fn; }

    这里有个规律,如果只改写 valueOf() 或是 toString() 其中一个,会优先调用被改写了的方法,而如果两个同时改写,则会像 String 转换规则一样,优先查询 valueOf() 方法,在 valueOf() 方法返回的是非原始类型的情况下再查询 toString() 方法。

    如果你能认真读完,相信会有所收获。

    到此本文结束,如果还有什么疑问或者建议,可以多多交流,原创文章,文笔有限,才疏学浅,文中若有不正之处,万望告知。

     

    1.本站遵循行业规范,任何转载的稿件都会明确标注作者和来源;2.本站的原创文章,请转载时务必注明文章作者和来源,不尊重原创的行为我们将追究责任;3.作者投稿可能会经我们编辑修改或补充。

    相关文章
    • 记一次和阿里某总监对话引发的思考:说说你框架的设计思路和优点亮点! - 路过秋天

      记一次和阿里某总监对话引发的思考:说说你框架的设计思路和优点亮点

      2016-12-20 11:00

    • Web前端面试题集锦 - wangwen896

      Web前端面试题集锦 - wangwen896

      2016-10-04 17:00

    • 很强大的HTML+CSS+JS面试题(附带答案) - Angelasp

      很强大的HTML+CSS+JS面试题(附带答案) - Angelasp

      2016-10-04 16:00

    • 前端开发面试题 - wangwen896

      前端开发面试题 - wangwen896

      2016-10-03 14:00

    网友点评