首页 > 编程语言 >令人头疼的Javascript隐式&强制转换

令人头疼的Javascript隐式&强制转换

时间:2022-11-18 13:56:47浏览次数:70  
标签:头疼 const Javascript NaN Number toString true 隐式

Javascript 的隐式强制只是指 Javascript 试图将意外的值类型强制为预期的类型。因此,您可以在需要数字的地方传递一个字符串,在需要字符串的地方传递一个对象等,它会尝试将其转换为正确的类型。这是最好避免的 Javascript 功能。

3 * "3" //9
1 + "2" + 1 //121

true + true //2
10 - true //9


const foo = {
  valueOf: () => 2
}
3 + foo // 5
4 * foo // 8

const bar = {
  toString: () => " promise is a boy :)"
}
1 + bar // "1 promise is a boy :)"


4 * [] // 0
4 * [2] // 8
4 + [2] // "42"
4 + [1, 2] // "41,2"
4 * [1, 2] // NaN

"string" ? 4 : 1 // 4
undefined ? 4 : 1 // 1

数字表达式中的非数字值

字符串

每当您在涉及以下任一运算符的数字表达式中将字符串作为操作数传递时:-, *, /, %,数字的转换过程类似于对值调用内置Number函数。这非常简单,任何仅包含数字字符的字符串都将转换为其等效的数字,但包含非数字字符的字符串将返回NaN. 如下图,

3 * "3" // 3 * 3
3 * Number("3") // 3 * 3
Number("5") // 5

Number("1.") // 1
Number("1.34") // 1.34
Number("0") // 0
Number("012") // 12

Number("1,") // NaN
Number("1+1") // NaN
Number("1a") // NaN
Number("one") // NaN
Number("text") // NaN

+ 运算符的情况

  • 运算符与其他数学运算符不同,执行两个功能:
  1. 数学加法
  2. 字符串连接

当字符串是 + 运算符的操作数时,Javascript 不是将字符串转换为数字,而是将数字转换为字符串。

// concatenation
1 + "2" // "12"
1 + "js" // "1js"

// addition
1 + 2 // 3
1 + 2 + 1 // 4

//addition, then concatenation
1 + 2 + "1" // "31"
(1 + 2) + "1" // "31"

//concatenation all through
1 + "2" + 1 // "121"
(1 + "2") + 1 // "121"

对象

大多数 Javascript 对象转换通常会导致[object Object],例如

"name" + {} // "name[object Object]

每个 javascript 对象都继承了一个toString方法,每当将对象转换为字符串时都会调用该方法。该toString方法的返回值用于字符串连接和数学表达式等操作。

const foo = {}
foo.toString() // [object Object]

const baz = {
  toString: () => "I'm object baz"
}

baz + "!" // "I'm object baz!"

当它是数学表达式时,Javascript 将尝试将返回值转换为数字,如果不是。

const foo = {
  toString: () => 4
}

2 * foo // 8
2 / foo // 0.5
2 + foo // 6
"four" + foo // "four4"

const baz = {
  toString: () => "four"
}

2 * baz // NaN
2 + baz // 2four

const bar = {
  toString: () => "2"
}

2 + bar // "22"
2 * bar // 4

数组对象

toStringArrays的继承方法有点不同,它的工作方式类似于join不带任何参数调用数组的方法。

[1,2,3].toString() // "1,2,3"
[1,2,3].join() // "1,2,3"
[].toString() // ""
[].join() // ""

"me" + [1,2,3] // "me1,2,3"
4 + [1,2,3] // "41,2,3"
4 * [1,2,3] // NaN

因此,当您在需要字符串的地方传递数组时,Javascript 会将toString方法的返回值与第二个操作数连接起来。如果它需要一个数字,它会尝试将返回值转换为一个数字。

4 * [] // 0
4 / [2] // 2

//similar to
4 * Number([].toString())
4 * Number("")
4 * 0

//

4 / Number([2].toString())
4 / Number("2")
4 / 2

真、假和“”

Number(true) // 1
Number(false) // 0
Number("") // 0

4 + true // 5
3 * false // 0
3 * "" // 0
3 + "" // "3"

参考视频讲解:进入学习

valueOf方法

也可以定义一个valueOf方法,当您传递一个需要字符串或数字值的对象时,Javascript 将使用该方法。

const foo = {
  valueOf: () => 3
}

3 + foo // 6
3 * foo // 9

当对象上同时定义了toStringvalueOf方法时,Javascript 将使用该valueOf方法。

const bar = {
  toString: () => 2,
  valueOf: () => 5
}

"sa" + bar // "sa5"
3 * bar // 15
2 + bar // 7

valueOf 方法适用于应该表示数值的对象。

const two = new Number(2)

two.valueOf() // 2

虚假与真实

我真的想让它变得虚假和真实

每个 Javascript 值都可以被强制为 true 或 false。强制转换为布尔值true意味着该值是真实的。强制转换为布尔值false意味着该值为假。

Javascript 中有一些返回假值的值,它们是:

  1. 错误的
  2. 0
  3. 空值
  4. 不明确的
  5. “”
  6. NaN
  7. -0

其他的都是真的

if (-1) // truthy
if ("0") // truthy
if ({}) // truthy

上面的代码片段是可以的,但是在尝试确定值的真实性时,最好是明确的。基本上,不要依赖 Javascript 的隐式强制,即使您觉得自己非常了解它们。
而不是下面的代码片段,

const counter = 2

if (counter)

根据您的要求,以下任何一种都是更好的做法

if (counter === 2)

//or

if (typeof counter === "number")

这是因为例如,您定义了一个应该处理数字的函数

const add = (number) => {
  if (!number) new Error("Only accepts arguments of type: number")
  //your code
}

所以如果我用 0 调用 add 函数,我总是会得到一个意想不到的错误

add(0) // Error: Only accepts arguments of type: number

//better check

const add = (number) => {
  if (typeof number !== "number") new Error("Only accepts arguments of type: number")
  //your code
}

add(0) // no error

NaN

NaN 是一个不等于自身的特殊数值。

NaN === NaN // false

const notANumber = 3 * "a" // NaN

notANumber == notANumber // false
notANumber === notANumber // false

NaN是唯一不等于自身的 Javascript 值。因此,您可以NaN通过将其与自身进行比较来检查。

if (notANumber !== notANumber) // true

ECMAScript 6 引入了一种检查 NaN 的方法, Number.isNaN

Number.isNaN(NaN) // true
Number.isNaN("name") // false

当心全局isNaN函数,它会在检查它是否为NaN. 例如,

isNaN("name") // true
isNaN("1") // false

isNaN应该避免使用全局函数,它的工作方式类似于下面的函数

const coerceThenCheckNaN = (val) => {
  const coercedVal = Number(val)
  return coercedVal !== coercedVal ? true : false
}

coerceThenCheckNaN("1a") // true
coerceThenCheckNaN("1") // false
coerceThenCheckNaN("as") // true
coerceThenCheckNaN(NaN) // true
coerceThenCheckNaN(10) // false

这是大部分隐性强制。如果我错过了什么,请在下面的评论中加入,感谢您一路阅读。

标签:头疼,const,Javascript,NaN,Number,toString,true,隐式
From: https://www.cnblogs.com/hellocoder2029/p/16902988.html

相关文章

  • JavaScript语法_变量和JavaScript语法_变量typeof
    JavaScript语法_变量:1.变量:变量:小块存储数据的内存空间Java语言是强类型语言,而Javascript是弱类型语言。强类型:在开辟变量存储空间时,定义了空间将来存储的数据的数据......
  • JavaScript_语法_特殊语法与JavaScript_语法_流程控制语句
    JavaScript_语法_特殊语法JS特殊语法:1.语句以;结尾,如果一行只有一语句则;可以省略(不建议)2.变量的定义使用var关键字,也可以不使用......
  • JavaScript语法_与html结合方式和JavaScript语法_注释&数据类型
    JavaScript语法_与html结合方式:ECMAscript:客户端脚本语言的标准1.基本语法∶与html结合方式1.内部JS:定义<script>,标签体内容就是js代码2.外部JS:定义<script>,通过s......
  • JavaScript_语法_一元运算符与JavaScript_语法_算数&比较运算符
    JavaScript_语法_一元运算符运算符:1.一元运算符:只有一个运算数的运算符++、--、+(正号)、-(负号)......
  • javascript-代码随想录训练营day2
    59.螺旋矩阵Ⅱ题目链接:https://leetcode.cn/problems/spiral-matrix-ii/题目描述:给你一个正整数n,生成一个包含1到n2所有元素,且元素按顺时针顺序螺旋排列的nxn......
  • JavaScript对象-RegExp1、RegExp2
    JavaScript对象-RegExp1 RegExp:正则表达式对象1、正则表达式:定义字符串的组成规则1.单个字符:[]如:[a][ab][a-zA-Z][0-9_]特殊符号代表特殊含义的单个字符:\ld:......
  • JavaScript对象-Global
    1.特点:全局对象这个Global中封装的方法不需要对象就可以直接调用方法名2.方法encodeURI():url编码decodeURI():url解码encodeURIComponent():url编码,编码的字符更......
  • javascript js 对象数组 转json 数组解构
           ......
  • JavaScript对象-Function、Array
    JavaScript对象-FunctionFunction:函数(方法对象):1、创建:1.varfun=newfunction(形式参数列表,方法体);2.function方法名称(形式参数列表){方法体}3.var......
  • JavaScript字符串MD5
    进行HTTP网络通信的时候,调用API向服务器请求数据,有时为了防止API调用过程中被黑客恶意篡改,所请求参数需要进行MD5算法计算,得到摘要签名。服务端会根据请求参数,对签名进行验......