查看原文
其他

面试官:a 可以同时 == 1 && == 2 && == 3吗?

The following article is from 鱼头的Web海洋 Author 陈大鱼头

点击上方“IT平头哥联盟”,选择“置顶或者星标

一起进步~


作者:鱼头的Web海洋 公号 / 陈大鱼头

此题目的答案可以分为三大类:

1. 类型转换时的劫持

首先我们要知道,在 JS 中类型转换只有三种情况,分别是:

  • 转换为布尔值

  • 转换为数字

  • 转换为字符串

转换为原始类型

对象在转换类型的时候,会执行原生方法ToPrimitive

其算法如下:

  1. 如果已经是 原始类型,则返回当前值;

  2. 如果需要转 字符串 则先调用 toSting方法,如果此时是 原始类型 则直接返回,否则再调用 valueOf方法并返回结果;

  3. 如果不是 字符串,则先调用 valueOf方法,如果此时是 原始类型 则直接返回,否则再调用 toString方法并返回结果;

  4. 如果都没有 原始类型 返回,则抛出 TypeError类型错误。

当然,我们可以通过重写 Symbol.toPrimitive来制定转换规则,此方法在转原始类型时调用优先级最高。

所以以此定义我们可以有以下四种答案:


  1. var a = {



  2. arr: [3, 2, 1],



  3. valueOf () {



  4. console.group('valueOf')



  5. console.log(this.arr)



  6. console.groupEnd('valueOf')



  7. return this.arr.pop()



  8. }



  9. }



  10. if (a == 1 && a == 2 && a == 3) {



  11. console.log('biu')



  12. }




  13. var b = {



  14. arr: [3, 2, 1],



  15. toString () {



  16. console.group('toString')



  17. console.log(this.arr)



  18. console.groupEnd('toString')



  19. return this.arr.pop()



  20. }



  21. }



  22. if (b == 1 && b == 2 && b == 3) {



  23. console.log('biu')



  24. }




  25. var c = {



  26. arr: [3, 2, 1],



  27. [Symbol.toPrimitive] () {



  28. console.group('Symbol.toPrimitive')



  29. console.log(this.arr)



  30. console.groupEnd('Symbol.toPrimitive')



  31. return this.arr.pop()



  32. }



  33. }



  34. if (c == 1 && c == 2 && c == 3) {



  35. console.log('biu')



  36. }




  37. var d = [1, 2, 3]



  38. d.join = d.shift



  39. if (d == 1 && d == 2 && d == 3) {



  40. console.log('biu')



  41. }


注:事实上,这四种可以算是同一种。关于最后一种,我们可以来看看ECMA中的 Array.prototype.toString() 定义:

  1. 定义 array 为 ToObject(thisvalue)(原生方法,将当前数组转换成对象);

  2. 定义 func 为 Get(array,'join')(原生方法,在这一步调用 join 方法);

  3. 如果 IsCallble(func) (原生方法,判断是否有内部可调用的函数)为 false,则 设置 func 原生函数 %ObjProto_toString%(原生函数, toString 的具体实现);

  4. 返回 Call(func,array)

2. 对 getter 的劫持

所谓的 getter 就是对象属性在进行查询时会被调用的方法 get,利用此函数也可以实现题目功能。

代码如下:


  1. window.val = 0



  2. Object.defineProperty(window, 'd', {



  3. get () {



  4. return ++this.val



  5. }



  6. })



  7. if (d == 1 && d == 2 && d == 3) {



  8. console.log('biu')



  9. }




  10. const e = new Proxy({}, {



  11. val: 1,



  12. get () {



  13. return () => this.val++;



  14. }



  15. });



  16. if (e == 1 && e == 2 && e == 3) {



  17. console.log('biu')



  18. }


3. 正则表达式

JS 中的 RegExp.prototype.exec() 作用是在一个指定字符串中执行一个搜索匹配,返回一个结果数组或 null

当正则表达式使用 " g" 标志时,可以多次执行 exec 方法来查找同一个字符串中的成功匹配。当你这样做时,查找将从正则表达式的 lastIndex 属性指定的位置开始。( test() 也会更新 lastIndex 属性)。

lastIndex 是正则表达式的一个可读可写的整型属性,用来指定下一次匹配的起始索引。只有正则表达式使用了表示全局检索的 " g" 标志时,该属性才会起作用。

注:只有正则表达式使用了表示全局检索的 " g" 标志时,该属性才会起作用。

综上所述,我们可以有方案如下:


  1. var f = {



  2. reg: /\d/g,



  3. valueOf () {



  4. return this.reg.exec(123)[0]



  5. }



  6. }



  7. if (f == 1 && f == 2 && f == 3) {



  8. console.log('biu')



  9. }


注:上述方法其实也利用了类型转换的特点。然后暂时就写下以上三种答案,不知道聪明的你是否还有别的解法呢?

 end -




用心分享 一起成长 做有温度的攻城狮

每天记得对自己说:你是最棒的!


往期推荐:

精读 The Cost of JavaScript

如何写出让同事无法维护的代码?

送你一波JS 开发常用工具函数~

前后端分离实际容易产生的问题

译 | JavaScript函数的6个基本术语

编写更好的 JavaScript 条件式和匹配条件的技巧


都看到这里了,给个“在看”再走呗~

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存