ECMAScript overview of

  • ECMAScript is a scripting language, abbreviated as ES. It is often seen as a standard specification for JAVAScript, but JAVAScript is actually an extension of ECMAScript. ECMAScript provides only the most basic syntax (i.e. how our code should be written, e.g. how variables and functions should be defined…). Just staying at the language level does not complete the actual function development in the page. JAVAScript, which we use a lot, implements the ECMAScript standard and extends it to allow us to manipulate DOM and BOM in the browser and read and write files in the Node environment.
  • In general, JAVAScript in the browser environment is equivalent to ECMAScript + Web API(BOM,DOM)

  • JAVAScript = ECMAScript + Node API used in the Node environment

  • The language itself in JAVScript is ECMAScript. With the in-depth development of the Web model, ES has maintained an iteration of one version per year since 2015, with many new features appearing successively. As a result, JAVAScript itself has become more and more advanced and convenient.

ES2015 overview

  1. Solve some problems or deficiencies in the original grammar. For example, let and const provide block-level scope.
  2. The original syntax is enhanced to make it more convenient to use. For example: deconstruction, expansion, parameter defaults, template strings…
  3. New methods, new functions, new objects. Promise, for example, the proxy Object. The assign ().
  4. New data types and data deconstruction. For example: Symbol, set, map…

ES2015 and let block-level scope

  • Before ES2015, there were only two types of scope in ES (global scope, function scope)
    • Global scope
    • Function scope
    • Block-level scope (new in ES2015)
  • Previously, blocks had no separate scope, and members defined within the block were externally accessible.

Such as:

If (true){var foo = 123} console.log(foo)//123 // is very bad for complex code and unsafeCopy the code
  • Use the new LETS in ES2015
If (true){let foo = 123} console.log(foo)//foo is not defined // Members defined in the block-level scope are not externally accessible.Copy the code

const

  • Used to declare a read-only constant (constant), with a read – only feature added to let.

Array structure

  • The structure of an array is based on position, and because the elements in the data have subscripts, it is regular.
Const arr = [100,200,300,4] const [foo,bar,baz] = arr; console.log(foo,bar,baz)//100 200 300 //----------------------------------------- const [foo,...reset] = arr; Console. log(reset)// [200, 300, 4] // reset all members from the current position and put them in an array. Note: this can only be used in the last position of our deconstruction.Copy the code
// If the number is smaller than the length of the array to be deconstructed, it is extracted from front to back. Const arr = [100,200,300,4] const [foo] = arr; Console. log(foo)//100 // Undefined const [foo,bar,baz,boo,coo] = arr if the number of structures is greater than the length of the array to be deconstructed; console.log(coo)//undefinedCopy the code

Object structure

Const obj = {name: "ykk", age:20} const {name, phone = "123000"} = obj console.log(phone)//123000 add defaultCopy the code
const obj = { name: "ykk", age:20 } let name = "kee" const { name: ObjName} = obj console.log(objName)//ykk If the structure name of the object conflicts with the external name, we can rename it such as objNameCopy the code
const obj = { name: "ykk", age:20 } let name = "kee" const { name: ObjName = "default value"} = obj console.log(objName)//ykk can also add default values for renamingCopy the code

Template string

// Traditional strings do not support line breaks. If our string has a line break we need to use the \n character to indicate it. Const STR = 'hello es2015 This is a string' console.log(STR) const STR = 'hello es2015 this is a string' console.log(STR)Copy the code
Const name = "Tom" const STR = 'HEY ${name}-- ${1+1} ----${true? 1-0} - ${Math. The random ()} ` console. The log (STR) / / HEY Tom - 2-0-0.22368499859594526Copy the code
  • Tagged template string
// The tag is actually a special function, Const STR = console.log 'hello word' //['hello word'] const name = "Tom" const gender = true function myTagFunc(strings,name,gender){ // console.log(strings)//[ 'HEY, ', ' IS A ', // console.log(name,gender)// Tom true const sex = gender? "man" : "woman" return strings[0] + name +strings[1]+ sex + strings[2] } const result = myTagFunc`HEY, ${name} IS A ${gender}. 'console.log(result) // the function of the tag can also accept the return values of all the template string expressions, such as name genderCopy the code

ES2015 string extension method

// String extension method const message = "Error: Foo is not defined" console.log(// message.startsWith("Error")// Whether to start with Error // message.endsWith("defined")// Whether to end with defined // message.includes("foo")Copy the code

Parameter Default Value

/ / function parameter default / / function foo (enable) {/ / / / enable = enable | | true / / enable enable = = = = undefined? true : enable // console.log(enable) // } // foo(false) function foo (enable = true){ console.log(enable)//false } foo(false)Copy the code

The remaining array

// function foo(){// console.log(arguments) //} // foo(1,2,3,4) function foo(num,... Args){// must be placed last and console.log(args) //[2,3,4]} foo(1,2,3,4)Copy the code

An array of

Const arr = [1,2,3] console.log(... arr)Copy the code

Arrow function

  • The arrow function makes our code shorter and easier to read

Object literal

Const bar = 1 const obj = {bar, method1(){console.log("11")}, [bar]:123, [Math.random()]:456 } console.log(obj) obj.method1()Copy the code

Object extension method

Object.assign() assigns the attributes of multiple source objects to a target Object

const source1 = { a:123, b:456 } const target = { a:999, b:888 } const source2 = { a:1000, c:666 } const result = Object.assign(target,source1,source2)//{ a: 1000, b: 456, c: 666} console.log(result === target)//true // If the source object and the target object have the same attributes, the source object will overwrite the target object,Copy the code
Function func(obj){// obj.name = "ykk" // Use object.assign () to assign a new target Object, Const result = object. assign({},obj) result.name = "ykk" result.aa = "123"} const obj = {name: "uuu" } func(obj) console.log(obj)Copy the code

Object.is() is used to determine whether two values are equal

  • Two equal operators that automatically convert data types before comparison
    console.log(
        0==false,//true
        0===false,//false
        NaN===NaN,//false
        +0 === -0,//true
        Object.is(+0 , -0),//false
        Object.is(NaN , NaN)//true
    )
Copy the code

Proxy

  • If we want to monitor the read and write of an Object, we can use object.defineProperty provided by ES5.
    const person = {
        name:"uu",
        age:30
    }

    const personProxy = new Proxy(person,{
        get(target,property){
        return  property in target ? target[property] : undefined
        },
        set(target,property,value){
            target[property] = value
        }
    })
    console.log(personProxy.xx)
    console.log(personProxy.age)
Copy the code

Proxy VS Object.defineProperty()

  • Object.defineproperty () can only monitor read and write of attributes
  • The Proxy advantages
    1. The Proxy can monitor operations on more objects, such as delete operations and calls to methods on objects
        const person = {
            name:"uu",
            age:30
        }
        const personProxy = new Proxy(person,{
            deleteProperxy(target,property){
                console.log(target,property)
            delete target[property]
            }
        })
    
        delete person.name
        console.log(person)//{ age: 30 }
    Copy the code
    1. Proxy better supports array object monitoring
        const list = []
        const result = new Proxy(list,{
            set (target,property,value){
            console.log('set',property,value)
            target[property] = value
            return true
            }
        })
        result.push(100)
    Copy the code
    1. Proxy regulates reads and writes of objects in a non-intrusive manner. A defined object can monitor its internal reads and writes without doing anything to the object. Object.defineproperty () needs to define the attributes of the Object that need to be monitored in a specific way. For an existing Object, you need to do a lot of extra work on it.

Reflect unified object manipulation API

  • Reflect is a static class and cannot pass new Reflect().
  • Reflect encapsulates a series of low-level operations on objects
  • The Reflect member method is the default implementation of Proxy handling objects
const person = { name:"ykk", age:20 } const p = new Proxy(person,{ get(target,property){ console.log("wtach log") return Reflect.get(target,property)  } }) console.log(p.name)Copy the code
  • The highlight of the Reflect object is that it provides a unified set of apis for manipulating objects. In the past, when manipulating objects, the Object method might be used, as well as operators such as in or delete, which was too chaotic for some novices and had no rules. The Reflect object solves this problem by unifying the way objects operate
const obj = { name:"23", Age :12} // console.log("name" in obj) whether an attribute exists // delete obj["name"] Deletes an attribute of the object // console.log(obj) // Console. log(object.keys (obj)) gets all properties of the Object console.log(reflect.has (obj,'name')) console.log(reflect.deleteProperty (obj,'age')) console.log(Reflect.ownKeys(obj))Copy the code

Promise

  • A new asynchronous programming solution, through the chain call method, solves the traditional asynchronous programming callback function nested too deep.

class

    // function Person(name){
    //     this.name = name
    // }

    // Person.prototype.say = function (){
    //      console.log(`hi my name is ${this.name}`)
    // }

    // let p = new Person("tom")

    // p.say()

    class Person{
        constructor(name){
        this.name = name
        }

        say(){
            console.log(`hi my name is ${this.name}`)
        }
    }

    let p = new Person("jack")
    p.say()
Copy the code
  • This independently defined syntax is easier to understand and has a clearer structure than previous functions

ES2015 Static method

Person {constructor(name){this.name = name} say(){console.log(this.name) Static create(name){return new person (name)}} let p = person.create (" Tom ") p.say() static create(name){return new person (name)} let p = person.create (" Tom ") p.say()Copy the code
  • Note: Because static methods are mounted to types, inside static methods their this does not refer to an instance object, but to the current type

ES2015 class inheritance

  • Prior to ES2015, inheritance was mostly implemented in a prototypical way. In 2015, a keyword extends specifically implements type inheritance.
  • Extends is more convenient and clear than archetypal inheritance
Class Person{constructor(name){this.name = name} say(){console.log(" Person ",this.name)}} //Student inherits Person. Class Student extends Person{constructor(name,number){// Super always points to the parent class. Calling it is like calling the superclass constructor super(name) this.number = number} hello(){super.say() console.log("Student",this.number)}} let student = new Student("jack",2) student.hello()Copy the code

ES2015 Set

  • A set data structure, known as a set, is similar to a traditional array, except that the internal members of a set are not allowed to be duplicated, and each value is unique within the same set.
Let s = new set () //add method adds some data, because this method returns the collection itself, Add (1).add(2).add(3).add(4).add(5) Log (I =>console.log(I)) // for(let val of s){// console.log(val) //} Console. log(s.haas (1))// Whether there is a specific value console.log(s.delte (1))// delete a value // s.clear()// Clear all the contents of the current collection // console.log(s) // array deduplicating // let result = array. from(new Set(arr)) let result = [...new Set(arr)] console.log(result)Copy the code

ES2015 Map

/ / / / let the map data structure obj = {} / / obj [true] = "value" / / obj [123] = 1 / / obj [{a: 1}] = "value" / / obj [' [object object] '] = "value" // console.log(obj) // { // true: 'value', // '[object Object]': 'value', // '[object Object ]': 'value' // } // console.log(Object.keys(obj))//[ '123', 'true', '[object Object]', '[object object]'] // Any object that is not a string is converted to a string (toString). The toString structure of each object is the same by default and cannot be distinguished. To solve this problem, we have data structures such as map, which is strictly a set of key-value pairs, used to map the relationship between two sets of any type. let map = new Map () let tom = { name: 'jack'} map.set(tom,20) map.set(true,1).set("num","123") // console.log(map.has(tom)) // console.log(map.delete(tom)) //  map.clear() map.forEach((value,key) =>{ console.log(value,key) })Copy the code
  • The biggest difference between a map and an object is that it can use any type of data as a key, whereas an object can only use strings as keys

ES2015 Symbol

  • Before ES2015, attribute names of objects were strings, and strings could be duplicated, which would cause conflicts.
  • Each value created through Symbol is unique and is never repeated
// const s = Symbol() // console.log(s) // console.log(typeof s)// Symbol // print Symbol through typeof, indicating that it is indeed a new data type. // console.log(//Symbol () === Symbol() false //) //Symbol type is unique, The types created by him are never duplicated // console.log(Symbol("foo")) // console.log(Symbol("bar")) // console.log(Symbol("baz")) // Symbol Since ES2015, objects can use the value of Symbol type as attribute names. Now object attribute names can be of two types, one is a string. Obj = {} // obj[Symbol()] = 123 // obj[Symbol()] = 456 // console.log(obj) So you don't have to worry about possible conflicts. // Use the method of calculating the attribute name, // const obj = {// [Symbol()] :123 //} // console.log(obj) //Symbol We can also use the Symbol type feature to simulate private members of objects. //a.js============================== const name = Symbol() const person = { [name]:123, say(){ console.log(this[name]) } } //b.js============================== // Console. log(person[name])// Since we cannot create an identical Symbol again, we cannot access this member. The person. Say () //Symbol member can only be called to add unique attribute names to objectsCopy the code

ES2015 Symbol added

. / / the console log (/ / Symbol () = = = Symbol (), / / false / / Symbol (" foo ") = = = Symbol (" foo ") / / false / / / / by creating value is the only Symbol function, Each call to the Symbol function yields a new structure, regardless of whether our incoming description text is the same or not. // If we want to reuse the same Symbol globally, we can use global variables or static methods provided by the Symbol type. // The for method can take a string as an argument. The same string returns the same Symbol value. This method internally maintains a global registry, // const s1 = symbol.for ("foo") // const s2 = symbol.for ("foo") // console.log(// s1 === s2) //true //) // Note: the for method maintains the relation between the string and the Symbol. If we pass in a string that is not a string, the method will convert it to a string. This will result in the same Boolean true as the string "true". // console.log(// symbol.for (true) === symbol.for ("true")//true //)// There are also many built-in Symbol constants in the Symbol type that are used to identify internal methods. These identifiers allow custom objects to implement some of the built-in js interfaces // console.log(symbol.iterator) // console.log(symbol.hasinstance) // toString tags for custom objects // Const obj = {// // If string tags are used to add identifiers, // [symbol.toStringTag]:"xObject" //} // ECMAScript requires us to implement an interface for this using Symbol console.log(obj.toString())//[object xObject] const obj = { [Symbol()]:"symbol value", foo:123 } for(let val in obj){ console.log(val) //foo } console.log(Object.keys(obj))//[ 'foo' ] Console. log(json.stringify (obj))//{"foo":123} // Use the Symbol value for the Object's property name, which is not available through traditional for in loops and object.keys. Serializing the object to a JSON string Symbol property via json.stringify is also ignored. None of the above methods can get the Symbol property name. These attributes make Symbol properties particularly suitable as private properties of objects. / / if you want to obtain Symbol attribute of the Object is not no way, we can use the Object. The getOwnPropertySymbols method, similar to the Object of the Object. The keys () method. / / Object. The keys () method can only obtain all the string objects in your property name / / Object. GetOwnPropertySymbols () to obtain the Symbol type of attribute names console.log(Object.getOwnPropertySymbols(obj))//[ Symbol() ]Copy the code

The for of circulation

  • There are many ways to iterate over data in ECMAScript. The basic for loop is a good way to iterate over ordinary arrays. Then there are the for in loop, which is a good way to iterate over key/value pairs. For example, the forEach method on an array object. The various ways of traversing data have certain limitations. So ES2015 borrowed a lot of other languages and introduced a new way of traversal for of loop.
  • The for of loop serves as a unified way to traverse all data structures.
Const arr = [100, 200, 300, 400] //for (let item of arr){// console.log(item) //} //for of takes each element of the array, not the corresponding index. For (let item of arr){console.log(item) if(item>100){break}} //for of can jump loop, // arr.foreach ()// arr.some() returns true to break out of the loop // arr.every() returns false to break out of the loop // const s = new Set(['foo','bar']) // for(const item of s){ // console.log(item) // } // const m = new Map() // m.set("foo",123).set("boo",456) // for(const [key,value] of m){ // console.log(key,value) // } const obj = { name:'kk', age: 19} for(const item of obj){console.log(item) //obj is not iterable} //for of loop can be used as a uniform way to iterate over all data structures, but even the most basic ordinary objects can not iterate over? Why is that?Copy the code

Iterable interface

  • for… The OF loop is a uniform way of traversing data, but after testing, it is found that it can only traverse data structures such as groups of numbers. For ordinary objects, traversing directly will report an error.
  • ES can represent an increasing number of structured data types, from arrays and objects to sets and maps. Developers can also combine these types to define data structures that fit their business needs. In order to provide a unified way of traversal, ES2015 proposed an Iterable interface, Iterable.
  • An iterable interface is one that can be used for… The specification standard for unified traversal access for the OF loop. As long as this data implements an iterable interface, it can be used by for.. Of loops to iterate over. Before trying set map, array can be for… Of traversal data because they have implemented the interface internally

const set = new Set(["foo","bar","coo"]) let interator = set[Symbol.iterator]() console.log(interator.next()) console.log(interator.next()) console.log(interator.next()) console.log(interator.next()) console.log(interator.next()) // print the following: // {value: 'foo', done: false} // {value: 'bar', done: false} // {value: 'coo', done: false} // {value: 'foo', done: false} undefined, done: true } // { value: undefined, done: true }Copy the code

Implement iterable interfaces

// const obj = {// //iterable // [symbol.iterator]:function(){// // must have a next method inside iterator // return {// Next :function(){// //iterationResult // // this interface specifies that there must be a value attribute inside the object that represents the data being iterated to. The value can be of any type. // return {// value: "ykk", // done: true // } // } // } // } // } const obj = { store:["foo","bar","boo"], [Symbol.iterator]:function(){ let self = this let index = 0 return { next:function(){ const result = { value: self.store[index], done: index>=self.store.length } index ++ return result } } } } for(const item of obj){ console.log(item) }Copy the code

ES2015 iterator pattern

// Iterator design pattern // scenario: Two people together to open a task list/code/xiao Ming = = = = = = = = = = = = = = = = = = = = = = const todos = {life: [" shop taobao, "" money", "clothing", "eat small lobster"], learn: [" language ", "mathematics", "English"]. Work: (" drink tea "), each:function(callback){ let arr = [].concat(this.life,this.learn,this.work) for(const item of arr){ callback(item) } },  [Symbol.iterator]:function(){ const arr = [...this.life,...this.learn,...this.work] let index = 0 return { next:function(){ const result = { value:arr[index], Done: index++ > = arr. Length} return result}}}} / / little red code = = = = = = = = = = = = = = = = = = = = = = = = = = = / / for (const item of todos. Life) { // console.log(item) // } // for(const item of todos.learn){ // console.log(item) // } // for(const item of todos.work){ // console.log(item) //} // If the data structure changes and a new class is added, but currently Xiao Hong's code and Xiao Ming's code data structure are heavily coupled, Todos. each(function(item){console.log(item)}) for(const item of todos){ Console. log(" iterator ",item)}Copy the code
  • Iterator his core meaning is external to provide a unified traversal interface, external need not go to the internal structure of the data is what kind of relationship,ES2015 iterator implementation of the language level of the iterator pattern, can be applied to any data structure, just need your code to implement iterator method, and his iteration logic.

ES2015 generator

  • The purpose of the Generator was to reduce the problem of nested callback functions in complex asynchronous code and provide a better asynchronous programming solution
// function *foo(){ // console.log("log") // return 100 // } // const result = foo() // console.log(result.next())//{ value: 100, done: The return value of the next method has the same structure as the return value of the iterator method, indicating that generator also implements the iterator interface function * foo(){console.log(11) yield 100 console.log(22) yield 200 console.log(33) yield 300 } const result = foo() console.log(result.next()) Console.log (result.next()) console.log(result.next()) console.log(result.next()) console.log(result.next()) When a yield keyword is paused during execution, the yield value is returned as the result, and the cycle continues. // The biggest characteristic of generator functions is lazy execution.Copy the code
  • The generator application
/ / the const iterator method using the generator function todos = {life: [" shop taobao, "" money", "clothing", "eat small lobster"], learn: [" language ", "mathematics", "English"], work: [" drink tea "], [Symbol.iterator]:function * (){ const arr = [...this.life,...this.learn,...this.work] for (const item of arr){ yield item } } } for (const item of todos){ console.log(item) }Copy the code

ES2016 overview

  • Compared to ES2015, ES2016 is a small version with only two small features.
//ES2016 // Array.prototype.includes---------------- const arr = ["foo", NaN, false] console.log(arr.indexOf("foo")) console.log(arr.indexOf(NaN))//false console.log(arr.includes(NaN))//true // Before ES2015 we used indexOf to check if there was a value in the array. IndexOf did not find NaN in the array. / / index operator -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- the console. The log (2, 10)) (math.h pow console. The log (2 * 10)Copy the code

ES2017 overview

const obj = { foo:"value1", Bar: "value2"} / / Object. Values () returns an array of all the values of Object of -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - / / the console. The log (Object) values (obj)) / / Object. Entries () in the form of an array to return to the Object's key values to -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- / / console log (Object. Entries (obj)) / / / 'foo', 'value1', [ 'bar', 'value2' ] ] // for (const [key,value] of Object.entries(obj)){ // console.log(key,value) // } console.log(new Map(Object.entries(obj)))//Map(2) { 'foo' => 'value1', 'bar' => 'value2' } // Object.getOwnPropertyDescriptors---------------------------------- // String.prototype.padEnd------------------------------------ // String.prototype.padStart---------------------------------- //Async/AwaitCopy the code