Javascript的数据类型

JS中数据类型详解

Posted by RyanYang on November 24, 2020

Javascript语言详解,第一篇开始了

1.数据类型

Null, undefined, boolean, number, string, symbol(es6),bigint(es6),object

1.1 number和bigint

number

js以64位浮点数(IEEE754)的形式储存number

618px-IEEE_754_Double_Floating_Point_Format.svg

转换规则如下: \(6.5 = 110.1_2 = 1.101*2^4 = -1^0*(1+0.101)*2^{1025-1023}=(-1)^{sign}*(1+fraction)*2^{exponent-1023}\) 所以$sign=0,exponent=1025=10000000001_2,fraction=1010000..000_2$

为什么是$exponent-1023$ ?

因为$2^{11} = 2048$表示范围为$[0,2047]$,而指数部分可能为负,所以使用$e-1023$表示范围转换为$[-1022,1023]$,其中$e \in (0,2047)$,因此js能表示的normal的范围为$[2^{-1022},2^{1023}]$

当$e == 0$

  • $f!=0$,表示subnormal,即比$2^{-1022}$更小的正常数,计算方法为:
\[{\displaystyle (-1)^{\text{sign}}\times 2^{1-1023}\times 0.{\text{fraction}}=(-1)^{\text{sign}}\times 2^{-1022}\times 0.{\text{fraction}}}\]
  • $f==0$,表示$0   -0$,符号表示趋向的方向

当$e==2047$时

  • $f == 0$,表示Infinity-Infinity
  • $f!=0$表示NaN

bigint

Es6出现的bigint解决了number只能准确表示$[-(2^{53}-1),2^{53}-1]$之间的整数的问题。

当超过这个范围时,number会出现下面这种情况

截屏2020-11-18 22.15.20

这是因为fraction只有52位,当超过这个范围时,低位会由于空间不足被舍去。

1.2 null 和 undefined

undefined 在一个变量声明且未赋值时由编译器自动赋值,null需要手动赋值表示故意缺少的对象值

1
2
3
4
null == undefined//true
null === undefined//false
typeof(null)//object
typeof(undefined)//undefined

About typeof()

First we know the value of js is stored with a type tag(at the lowest bits) in memory:

  • 000: object. The data is a reference to an object.
  • 1: int. The data is a 31 bit signed integer.
  • 010: double. The data is a reference to a double floating point number.
  • 100: string. The data is a reference to a string.
  • 110: boolean. The data is a boolean.

Two values were special:

  • undefined (JSVAL_VOID) was the integer $-2^{30}$ (a number outside the integer range).
  • null (JSVAL_NULL) was the machine code NULL pointer. Or: an object type tag plus a reference that is zero.我们可以认为null是引用次数为0的对象

Then the engine’s code for typeof:

截屏2020-11-18 17.10.03

在检查(1)undefined的时候

1
  #define JSVAL_IS_VOID(v)  ((v) == JSVAL_VOID)

在(2)object的时候,检查type tag,由于null pointer在内存中以$0x00$的形式存放,且不可被调用,因此判断为object

同时可以得到typeof的输出值:

  • number
  • string
  • boolean
  • undefined
  • object
  • function
  • symbol(es6)
  • Bitint(es6)

补充:

我们经常会这样使用if...else

1
2
if(obj){doSomething}
if(!obj){doOtherThing}

为什么不管是null还是undefined都不会调用呢?

这里涉及到js中的真假值和强制类型转换

JS中7个falsy:

false(boolean),+0(number),-0(number),null(null),undefined(null),’‘(string),NaN(number)

除此以外全部为真值

if语句判断表达式的结果是True还是false,若传入非boolean类型,根据其真/假值强制转换为boolean

因此我们可以用!!obj根据真假值性质强制转换为boolean型

1.3 Symbol

用于创建匿名且唯一的对象键值

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
//创建
let a = Symbol('this is a')//描述可选
a.toString()//Symbol('this is a')
let b = Symbol('this is b')

//使用
let obj = {
    [a]:foo,
    [b]:bar,
    c:"this is c"
}//必须使用[],否则当作字符串
obj[a]//foo,只能用[]访问,因为.后当作字符串处理

//
Symbol.for('foo') === Symbol.for('foo')//检查全局中是否有key === 'foo'的Symbol,如果有,返回它,没有则创建并登记
Symbol.keyFor(a)//undefined返回登记的key

//遍历
Object.getOwnPropertySymbols(obj);
//[Symbol(this is a),Symbol(this is b)]
//使用for.. in 得不到Symblo
Reflect.ownKeys(obj)
//[Symbol(this is a),Symbol(this is b),"c"]

1.4 Object

Set

set可以储存任何类型的唯一的值,包括原始值或对象引用,比较方法类似于===,特殊在NaN

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
//创建
let s = new Set()
let s1 = new Set([1,1,1,2,2,3,NaN,NaN])//[1,2,3,NaN]
//CRUD
s.add(1)
s.delete(1)
s.has(1)
s.clear()
//遍历
for (let item of s.values()){
    //do something
}
s.keys()
//set没有keys,实际返回
s.values()
//也可以直接for...of 遍历 set,因为和数组一样默认可遍历
s.entries()

数组的mapfilter都可以用于set,因此可以和简单实现交并差集合

WeakSet

weakSet只能储存对象,特点在于不计入对象的引用,因此不存在内存泄漏问题,适合临时储存一些对象比如DOM节点:当这些节点被文档移除时,不会发生内存泄漏。

同时由于垃圾回收机制的不可预测,weakset不可遍历

1
2
3
4
5
//任何可迭代对象都作为参数初始化weakset,该对象的所有成员(也必须对象),都会自动成为 WeakSet 实例对象的成员。
const a = [[1, 2], [3, 4]];
const ws = new WeakSet(a);
// WeakSet {[1, 2], [3, 4]}
//add,delete,has

Map

内置的object,本质上就是一种键值对的hash结构,但key值只能使用字符串。

map的key可以是任何类型(包括对象)

1
2
3
4
let obj = {o:"this is o"}
let map = new Map()
map.set(obj,"this is also o")//返回map本身
map.get(obj)//this is also o 返回value

若key重复(以内存地址为准)添加,覆盖。

WeakMap

WeakMap的专用场合就是,它的key所对应的对象,可能会在将来消失。

一个典型应用场景是,在网页的 DOM 元素上添加数据,就可以使用WeakMap结构。当该 DOM 元素被清除,其所对应的WeakMap记录就会自动被移除。

WeakMap 的另一个用处是部署私有属性。

但是,WeakMap 弱引用的只是key,而不是value。键值依然是正常引用。