美文网首页
a 可以同时 == 1 && == 2 && == 3吗?

a 可以同时 == 1 && == 2 && == 3吗?

作者: 代码技巧 | 来源:发表于2019-07-22 12:02 被阅读0次

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

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

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

转换为布尔值

转换为数字

转换为字符串

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

其算法如下:

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

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

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

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

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

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

​var a = {

   arr: [3, 2, 1],

   valueOf () {

       console.group('valueOf')

       console.log(this.arr)

       console.groupEnd('valueOf')

       return this.arr.pop()

   }

}

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

   console.log('biu')

}

var b = {

   arr: [3, 2, 1],

   toString () {

       console.group('toString')

       console.log(this.arr)

       console.groupEnd('toString')

       return this.arr.pop()

   }

}

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

   console.log('biu')

}

var c = {

   arr: [3, 2, 1],

   [Symbol.toPrimitive] () {

       console.group('Symbol.toPrimitive')

       console.log(this.arr)

       console.groupEnd('Symbol.toPrimitive')

       return this.arr.pop()

   }

}

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

   console.log('biu')

}

var  d = [1, 2, 3]

d.join = d.shift

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

   console.log('biu')

}

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

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

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

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

返回 Call(func,array)。

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

代码如下:

window.val = 0

Object.defineProperty(window, 'd', {

   get () {

       return ++this.val

   }

})

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

   console.log('biu')

}

const e = new Proxy({}, {

 val: 1,

 get ()  {

   return () => this.val++;

 }

});

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

   console.log('biu')

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

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

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

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

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

var f = {

   reg: /\d/g,

   valueOf () {

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

   }

}

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

   console.log('biu')

}

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

相关文章

  • a 可以同时 == 1 && == 2 && == 3吗?

    作者:鱼头的Web海洋 公号 / 陈大鱼头 此题目的答案可以分为三大类: 类型转换时的劫持首先我们要知道,在 JS...

  • 来自灵魂深处的拷问?

    1、人可以同时爱两个人吗? 2、我这辈子可以保证婚后不出轨吗? 3、我可以爱一个人直到永远吗? 我以前以为我可以爱...

  • html seu.menu~有感

    1,html网页标题及网页图标的引用 2,修饰引用的是.css而不是less 3,(1)可以同时在.less里同时...

  • a 可以同时 == 1 && == 2 &

    此题目的答案可以分为三大类: 1. 类型转换时的劫持 首先我们要知道,在 JS 中类型转换只有三种情况,分别是: ...

  • 房贷

    [机智]【一个关于房子的故事】 1、外地房产能贷吗? 可以 2、刚刚按揭房能贷吗? 可以 3、...

  • mac下的开发工具pycharm运行起python项目来

    python2 和python3是可以同时存在的 pip和pip3也是可以同时存在的 如果想用python2的命令...

  • CSS的三大特性

    1 继承性 1 什么是继承性2 所有的属性都可以继承吗3 孙子能继承爷爷的属性吗 2 层叠性 1 什么是层叠性2 ...

  • DAY11 微习惯的养成

    1一个习惯养成要多少天? 2养成习惯需要多少自律? 3我们可以同时培养多个习惯吗? 第一个问题大家怕是会不约而同喊...

  • eclipse 启动,同时启动的虚拟机吗?

    Questions 1、eclipse 启动,同时启动的虚拟机吗?2、jvm什么时候启动的,什么时候关闭的?3、e...

  • 每天郁郁寡欢,用积极心理学教你建立自我

    一、你足够正能量吗? 1.面对困难你真的可以做到迎难而上吗? 2.遇到挫折你真的可以很快调整心态,振作起来吗? 3...

网友评论

      本文标题:a 可以同时 == 1 && == 2 && == 3吗?

      本文链接:https://www.haomeiwen.com/subject/tifilctx.html