Abstract Equality Comparison is the == operator, also known as loose Equality and non-strict Equality.
In the process of abstract equality comparison, the data on both sides of == will be converted to the same type as much as possible so that they can be compared. This is the implicit type conversion of.
The problem
Let’s start with the confusing examples of == :
console.log([] == ! [])// true
console.log([] == []) // false
console.log(0= ='0') // true
console.log(0= = [])// true
console.log('0'= = [])// false
console.log(Boolean(null)) // false
console.log(Boolean(undefined)) // false
console.log(null= =false) // false
console.log(undefined= =false) // false
console.log(null= =undefined) // true
const a = { toString() { return '0'}}const b = { toString() { return '0'}}console.log(a == 0) // true
console.log(b == 0) // true
console.log(a == b) // false
Copy the code
Let’s implement an Abstract Equality Comparison according to the ECMA-262 Abstract Equality Comparison to see what happens during the Comparison.
implementation
AbstractEqualityComparison( x, y )
The body of an abstract equality comparison.
Abstract Equality Comparison
If Type(x) is the same as Type(y), then
a. Return the result of performing Strict Equality Comparison x === y.
If x is null and y is undefined, return true.
If x is undefined and y is null, return true.
If Type(x) is Number and Type(y) is String, return the result of the comparison x == ! ToNumber(y).
If Type(x) is String and Type(y) is Number, return the result of the comparison ! ToNumber(x) == y.
If Type(x) is BigInt and Type(y) is String, then
a. Let n be ! StringToBigInt(y).
b. If n is NaN, return false.
c. Return the result of the comparison x == n.
If Type(x) is String and Type(y) is BigInt, return the result of the comparison y == x.
If Type(x) is Boolean, return the result of the comparison ! ToNumber(x) == y.
If Type(y) is Boolean, return the result of the comparison x == ! ToNumber(y).
If Type(x) is either String, Number, BigInt, or Symbol and Type(y) is Object, return the result of the comparison x == ToPrimitive(y).
If Type(x) is Object and Type(y) is either String, Number, BigInt, or Symbol, return the result of the comparison ToPrimitive(x) == y.
If Type(x) is BigInt and Type(y) is Number, or if Type(x) is Number and Type(y) is BigInt, then
If x or y are any of NaN, +∞, or -∞, return false.
b. If the mathematical value of x is equal to the mathematical value of y, return true; otherwise return false.
Return false.
Note: ` x = =! The ` ` ToNumber (x)! The meaning of 'is not clear to me, but the early specification is not'! ', and the logic is smooth after ignoring, the subsequent processing will be ignored.Copy the code
Abstract Equality Comparison
If Type(x) and Type(y) are the same, then
A. Return x === y
True if x is null and y is undefined.
True if x is undefined and y is null.
Return x == ToNumber(y) when Type(x) is Number and Type(y) is String.
Return ToNumber(x) == y when Type(x) is String and Type(y) is Number.
If Type(x) is BigInt and Type(y) is String
A. Run n = StringToBigInt(y).
B. Return false if n is a NaN.
C. Return x == n.
Return y == x when Type(x) is String and Type(y) is BigInt.
When Type(x) is Boolean, return ToNumber(x) == y.
Return x == ToNumbery when Type(y) is Boolean.
Return x == ToPrimitive(y) when Type(x) is String, Number, BigInt, or Symbol and Type(y) is Object.
Return ToPrimitive(x) == y when Type(x) is Object and Type(y) is String, Number, BigInt, or Symbol.
When Type(x) is BigInt and Type(y) is Number, or when Type(y) is Number and Type(y) is BigInt
A. Return false if x or y is NaN, +∞, or -∞.
B. Return true if the value of x is equal to the value of y, and false otherwise.
Returns false.
The most perplexing is ToPrimitive, we can leave it alone and finish the main body first.
Type, ToNumber, ToPrimitive, etc will be explained later.
In 12.b, the numeric comparison between Number and BigInt can be performed by converting them to binary strings via toString(2).
function AbstractEqualityComparison(x, y) {
/ * 1 * /
if (Type(x) === Type(y)) {
/* a */
return x === y
}
/ * 2 * /
if (x === null && y === undefined) return true
/ * * / 3
if (x === undefined && y === null) return true
/ * * / 4
if (Type(x) === 'number' && Type(y) === 'string') {
return AbstractEqualityComparison(x, ToNumber(y))
}
/ * * / 5
if (Type(x) === 'string' && Type(y) === 'number') {
return AbstractEqualityComparison(ToNumber(x), y)
}
/ * * / 6
if (Type(x) === 'bigint' && Type(y) === 'string') {
/* a */
const n = StringToBigInt(y)
/* b */
if (Number.isNaN(n)) return false
/* c */
return AbstractEqualityComparison(x, n)
}
/ * * / 7
if (Type(x) === 'string' && Type(y) === 'bigint') {
return AbstractEqualityComparison(y, x)
}
/* 8 */
if (Type(x) === 'boolean') {
return AbstractEqualityComparison(ToNumber(x), y)
}
/* 9 */
if (Type(y) === 'boolean') {
return AbstractEqualityComparison(x, ToNumber(y))
}
/ * 10 * /
if(['string'.'number'.'bigint'.'symbol'].includes(Type(x)) &&
Type(y) === 'object'
) {
return AbstractEqualityComparison(x, ToPrimitive(y))
}
/ * * /
if (
Type(x) === 'object' &&
['string'.'number'.'bigint'.'symbol'].includes(Type(y))
) {
return AbstractEqualityComparison(ToPrimitive(x), y)
}
/ * * / 12
if (
(Type(x) === 'bigint' && Type(y) === 'number') ||
(Type(x) === 'number' && Type(y) === 'bigint')) {/* a */
if ([x, y].some(v= > [NaN.Infinity, -Infinity].includes(v))) {
return false
}
/* b */
return x.toString(2) === y.toString(2)}/ * * / 13
return false
}
Copy the code
Note: in12.a
In theincludes
Do not useindexOf
Instead,[NaN].indexOf(NaN)
Always return- 1
.
Type( argument )
Type gets the Type of the data.
ECMAScript Language Types
An ECMAScript language type corresponds to values that are directly manipulated by an ECMAScript programmer using the ECMAScript language. The ECMAScript language types are Undefined, Null, Boolean, String, Symbol, Number, BigInt, and Object. An ECMAScript language value is a value that is characterized by an ECMAScript language type.
Typeof is roughly the same as Type,
But typeof treats null as object and functions as functions,
Adjust these two types separately.
function Type(argument) {
const type = typeof argument
if (argument === null) return 'null'
if (type === 'function') return 'object'
return type
}
Copy the code
ToNumber( argument )
ToNumber is used to convert other types to the Number type.
ToNumber( argument )
The abstract operation ToNumber converts argument to a value of type Number according to Table 11:
Table 11: ToNumber Conversions
Argument Type Result Undefined Return NaN. Null Return +0. Boolean If argument is true, return 1. If argument is false, return +0. Number Return argument (no conversion). String See grammar and conversion algorithm below. Symbol Throw a TypeError exception. BigInt Throw a TypeError exception. Object Apply the following steps:
1. Let primValue be ? ToPrimitive(argument, hint Number).
2. Return ? ToNumber(primValue).
ToNumber(argument)
ToNumber converts argument to a value of type Number according to Table 11:
Table 11: ToNumber transformations
Argument type The results of Undefined Returns NaN. Null Return + 0. Boolean Returns 1 when argument is true. When argument is false, return +0. Number Return to argument (no conversion). String See syntax and conversion algorithms below. Symbol Throw a TypeError. BigInt Throw a TypeError. Object Apply the following steps:
1. Run primValue = ToPrimitive(argument, hint Number).
2. Return ToNumber(primValue).
ToNumber is almost identical ToNumber, except that for BigInt, ToNumber throws a type error directly.
function ToNumber(argument) {
if (Type(argument) === 'bigint') {
throw new TypeError('Cannot convert bigint to number value')}return Number(argument)
}
Copy the code
StringToBigInt( argument )
StringToBigInt converts a String into a BigInt.
The String parser for BigInt would presumably use StringToBigInt.
In BigInt, if StringToBigInt returns a NaN, an error will be thrown.
Using BigInt to implement StringToBigInt catches errors and returns NaN.
function StringToBigInt(argument) {
if(Type(argument) ! = ='string') {
throw new TypeError('Only accept string')}try {
return BigInt(argument)
} catch (e) {
return NaN}}Copy the code
ToPrimitive( input [, PreferredType ] )
ToPrimitive converts an input value to a value type (the base type other than Object).
ToPrimitive( input [, PreferredType ] )
Assert: input is an ECMAScript language value.
If Type(input) is Object, then
a. If PreferredType is not present, let hint be “default”.
b. Else if PreferredType is hint String, let hint be “string”.
c. Else,
i. Assert: PreferredType is hint Number.
ii. Let hint be “number”.
d. Let exoticToPrim be ? GetMethod(input, @@toPrimitive).
e. If exoticToPrim is not undefined, then
i. Let result be ? Call(exoticToPrim, input, « hint »).
ii. If Type(result) is not Object, return result.
iii. Throw a TypeError exception.
f. If hint is “default”, set hint to “number”.
g. Return ? OrdinaryToPrimitive(input, hint).
Return input.
ToPrimitive(input [, PreferredType])
Assertion: Input is an ECMAScript language value.
If Type(input) is Object
A. If the PreferredType does not exist, run the hint = “default” command.
B. Otherwise, run the hint = “String” command when the PreferredType is String.
C. otherwise,
I. Assert: PreferredType is Number.
Hint = “number”
D. Set exoticToPrim = GetMethod(input, @@toprimitive).
E. If exoticToPrim is not undefined
I. Make result = Call(exoticToPrim, input, « hint »).
Ii. If Type(result) is not Object, result is returned.
Iii. Raise TypeError.
F. When hint is “default”, run hint = “number”.
G. Return OrdinaryToPrimitive(input, hint)
Back to the input.
function ToPrimitive(input, PreferredType) {
/* 1 does not need to implement */
/ * 2 * /
if (Type(input) === 'object') {
let hint
/* a */
if (PreferredType === undefined) {
hint = 'default'
}
/* b */
else if (PreferredType === 'string') {
hint = 'string'
}
/* c */
else {
/* i */
if(PreferredType ! = ='number') {
throw new TypeError('preferred type must be "string" or "number"')}/* ii */
hint = 'number'
}
/* d */
const exoticToPrim = GetMethod(input, Symbol.toPrimitive)
/* e */
if(exoticToPrim ! = =undefined) {
/* i */
const result = exoticToPrim.call(input, hint)
/* i */
if(Type(result) ! = ='object') return result
/* i */
throw new TypeError('Cannot convert object to primitive value')}/* f */
if (hint === 'default') hint = 'number'
/* g */
return OrdinaryToPrimitive(input, hint)
}
/ * * / 3
return input
}
Copy the code
OrdinaryToPrimitive( o, hint )
OrdinaryToPrimitive converts an Object to a value type.
The hint parameter controls whether toString or valueOf is preferred.
OrdinaryToPrimitive( O, hint )
Assert: Type(O) is Object.
Assert: Type(hint) is String and its value is either “string” or “number”.
If hint is “string”, then
A. Let methodNames be « “toString”, “valueOf” »
Else,
A. Let methodNames be « “valueOf”, “toString” »
For each name in methodNames in List order, do
a. Let method be ? Get(O, name).
b. If IsCallable(method) is true, then
i. Let result be ? Call(method, O).
ii. If Type(result) is not Object, return result.
Throw a TypeError exception.
OrdinaryToPrimitive(O, hint)
Assert: Type(O) is Object.
Assertion: Type(hint) is String and the hint value is “String” or “number”.
When hint is “string”, then
A. Make methodNames = « “toString”, “valueOf” »
Otherwise,
A. Make methodNames = « “valueOf”, “toString” »
To iterate over methodNames, execute
A. Run method = Get(O, name).
B. If IsCallable(method) is true
I. Set result = Call(method, O).
Ii. If Type(result) is not Object, result is returned.
Throw a TypeError.
function OrdinaryToPrimitive(o, hint) {
/ * 1 * /
if(Type(o) ! = ='object') throw new TypeError('Only accept Object')
/ * 2 * /
if(Type(hint) ! = ='string'| |! ['string'.'number'].includes(hint)) {
throw new TypeError('Hint value must be "string" or "number"')}let methodNames
/ * * / 3
if (hint === 'string') {
/* a */
methodNames = ['toString'.'valueOf']}/ * * / 4
else {
/* a */
methodNames = ['valueOf'.'toString']}/ * * / 5
for (const name of methodNames) {
/* a */
const method = Get(o, name)
/* b */
if (IsCallable(method)) {
/* i */
const result = method.call(o)
/* ii */
if(Type(result) ! = ='object') return result
}
}
/ * * / 6
throw new TypeError('Cannot convert object to primitive value')}Copy the code
other
function GetMethod(v, p) {
if(! isPropertyKey(p)) {throw new TypeError('Not valid property key')}const func = GetV(v, p)
if (func === undefined || func === null) return undefined
if(! IsCallable(func)) {throw new TypeError('Not callable')}return func
}
function isPropertyKey(argument) {
if (Type(argument) === 'string') return true
if (Type(argument) === 'symbol') return true
return false
}
function Get(o, p) {
if(Type(o) ! = ='object') {
throw new TypeError('Only accept object')}if(! isPropertyKey(p)) {throw new TypeError('Not valid property key')}return o[p]
}
function GetV(v, p) {
if(! isPropertyKey(p)) {throw new TypeError('Not valid property key')}const o = ToObject(v)
return o[p]
}
function IsCallable(argument) {
if(Type(argument) ! = ='object') return false
if(!!!!! argument.call)return true
return false
}
function ToObject(argument) {
switch (Type(argument)) {
case 'boolean':
return new Boolean(argument)
case 'number':
return new Number(argument)
case 'string':
return new String(argument)
case 'symbol':
case 'bigint':
case 'object':
return argument
default:
throw new TypeError(
`Cannot convert ${Type(argument)} to object value`)}}Copy the code
test
function test(x, y) {
const a = x == y
const b = AbstractEqualityComparison(x, y)
const sign = a === b ? [
'%cSuccess'.'background: green; color: white; '
] : [
'%cFailure'.'background: red; color: white; '
]
console.log(... sign,` [${a}] `, x, y)
}
Copy the code
const testData1 = [
undefined.null.true.false.'123'.Symbol('123'),
123.123n, [] and {valueOf() {
return 123}}, {toString() {
return 123}}]const testData2 = [
undefined.null.true.false.'123'.Symbol('123'),
123.123n, [] and {valueOf() {
return 123}}, {toString() {
return 123
}
}
]
testData1.forEach((a) = > {
testData2.forEach((b) = > {
test(a, b)
})
})
Copy the code
conclusion
In real development, you want to avoid using the == operator whenever possible, and some might find this article pointless. What I’m really trying to convey is a way to learn the rest of JavaScript in the same way.
Finally, the implementation of AbstractEqualityComparison or useful, I on the basis of the above code to write the abstract equal comparison process display, can see more clearly the comparison process. Enter the question at the beginning of the article to see how the comparison works.