掌握JavaScript中的迭代器和生成器,顺便了解一下async、await的原理
掌握JavaScript中的迭代器和生成器,顺便了解一下async、await的原理
前言
相信很多人对迭代器和生成器都不陌生,当提到async和await的原理时,大部分人可能都知道async、await是Promise+生成器的语法糖,其原理具体是怎么做的呢?下面通过这篇文章带你详细了解一下迭代器和生成器,以及带你从生成器一步步推导到async和await。
1.迭代器(Iterator)
1.1.什么是迭代器?
迭代器是确使用户在容器对象(链表或数组)上遍访的对象,使用该接口无需关心对象的内部实现细节。
迭代器的定义可能比较抽象,简单来说迭代器就是一个对象,可用于帮助我们对某个数据结构(链表、数组)进行遍历;
在JavaScript中,迭代器也是一个具体的对象,并且这个对象必须符合迭代器协议(iterator protocol);
什么是迭代器协议?
- 在JavaScript中就是指这个对象必须实现一个特定的next方法,并且next方法有如下要求;
- next方法可接收0个或者1个参数(在生成器中next可以接收1个参数),并且需返回一个对象,对象包含以下两个属性:
- done:值为Boolean,如果迭代器可以迭代产生下一个值,就为false,如果已经迭代完毕,就为true;
- value:迭代器返回的值,如果done为false,value一般为undefined;
编写一个最简单的迭代器:
const iterator = {
next: function() {
return { done: false, value: 123 }
}
}
1.2.迭代器的基本使用
明白了迭代器的基本定义,下面就来实现一下符合迭代器协议的对象吧,并且看看其它的一些基本用法。比如,需要通过迭代器访问一个数组:
(1)创建一个迭代器对象
const names = ['curry', 'kobe', 'klay']
let index = 0 // 通过一个index来记录当前访问的位置
const iterator = {
next() {
if (index < names.length) {
return { done: false, value: names[index++] }
} else {
return { done: true, value: undefined }
}
}
}
console.log(iterator.next()) // { done: false, value: 'curry' }
console.log(iterator.next()) // { done: false, value: 'kobe' }
console.log(iterator.next()) // { done: false, value: 'klay' }
console.log(iterator.next()) // { done: true, value: undefined }
(2)实现生成迭代器的函数
- 如果每次需要去访问一个数组就去编写一个对应的迭代器对象肯定是很麻烦的;
- 可以封装一个函数,用于生成一个访问数组的迭代器;
function createIterator(arr) {
let index = 0
return {
next() {
if (index < arr.length) {
return { done: false, value: arr[index++] }
} else {
return { done: true, value: undefined }
}
}
}
}
const names = ['curry', 'kobe', 'klay']
// 调用createIterator函数,生成一个访问names数组的迭代器
const namesIterator = createIterator(names)
console.log(namesIterator.next()) // { done: false, value: 'curry' }
console.log(namesIterator.next()) // { done: false, value: 'kobe' }
console.log(namesIterator.next()) // { done: false, value: 'klay' }
console.log(namesIterator.next()) // { done: true, value: undefined }
1.3.可迭代对象
1.3.1.什么是可迭代对象?
上面提到了迭代器是一个对象,并且符合迭代器协议,那么什么是可迭代对象呢?它与迭代器又有什么区别?
- 迭代器是一个符合迭代器协议(iterator protocol)的对象,对象内实现了一个特定的next方法;
- 而可迭代对象是一个符合可迭代协议(iterable protocol)的对象,对象内实现了一个
Symbol.iterator
方法,并且该方法返回一个迭代器对象; - 所以,可以说可迭代对象包含了迭代器对象,可迭代对象中实现了一个特定方法用于返回迭代器对象;
如下,iteratorObj
就是一个可迭代对象:
const iteratorObj = {
names: ['curry', 'kobe', 'klay'],
[Symbol.iterator]: function() {
let index = 0
return {
// 注意:这里的next需要使用箭头函数,否则this访问不到iteratorObj
next: () => {
if (index < this.names.length) {
return { done: false, value: this.names[index++] }
} else {
return { done: true, value: undefined }
}
}
}
}
}
// 调用iteratorObj中的Symbol.iterator得到一个迭代器
const iterator = iteratorObj[Symbol.iterator]()
console.log(iterator.next()) // { done: false, value: 'curry' }
console.log(iterator.next()) // { done: false, value: 'kobe' }
console.log(iterator.next()) // { done: false, value: 'klay' }
console.log(iterator.next()) // { done: true, value: undefined }
1.3.2.JS内置的可迭代对象
上面的可迭代对象都是由自己实现的,其实在JavaScript中为我们提供了很多可迭代对象,如:String、Array、Map、Set、arguments对象、NodeList(DOM集合)等。
// 1.String
const str = 'abc'
const strIterator = str[Symbol.iterator]()
console.log(strIterator.next()) // { value: 'a', done: false }
console.log(strIterator.next()) // { value: 'b', done: false }
console.log(strIterator.next()) // { value: 'c', done: false }
console.log(strIterator.next()) // { value: undefined, done: true }
// 2.Array
const names = ['curry', 'kobe', 'klay']
console.log(names[Symbol.iterator])
const namesIterator = names[Symbol.iterator]()
console.log(namesIterator.next()) // { value: 'curry', done: false }
console.log(namesIterator.next()) // { value: 'kobe', done: false }
console.log(namesIterator.next()) // { value: 'klay', done: false }
console.log(namesIterator.next()) // { value: undefined, done: true }
// 3.Map/Set
const set = new Set
set.add(10)
set.add(20)
set.add(30)
const setIterator = set[Symbol.iterator]()
console.log(setIterator.next()) // { value: 10, done: false }
console.log(setIterator.next()) // { value: 20, done: false }
console.log(setIterator.next()) // { value: 30, done: false }
console.log(setIterator.next()) // { value: undefined, done: true }
1.3.3.可迭代对象应用场景
可迭代对象在实际应用中特别常见,像一些语法的使用、创建一些对象和方法调用都用到了可迭代对象。
JS中的语法:for...of、展开语法、解构等。
for...of可用于遍历一个可迭代对象,其原理就是利用迭代器的next函数,如果done为false,就从返回的对象中拿到value返回给我们,而对象不是一个可迭代对象,所以对象不能使用for...of遍历;
const num = [1, 2, 3]
for (const item of num) {
console.log(item) // 1 2 3
}
// 遍历上面自己定义的可迭代对象iteratorObj也是可以的
for (const item of iteratorObj) {
console.log(item) // curry kobe klay
}
const obj = { name: 'curry', name: 30 }
for (const key of obj) {
console.log(key)
}
为什么数组能使用展开语法,其原理也是用到了迭代器,在使用
...
对数组进行展开时,也是通过迭代器的next去获取数组的每一项值,然后存放到新数组中;const names = ['james', 'green']
// 将数组和iteratorObj通过扩展进行合并
const newNames = [...names, ...iteratorObj]
console.log(newNames) // [ 'james', 'green', 'curry', 'kobe', 'klay' ]
可迭代对象都是可以使用解构语法的,像数组、字符串为什么可以使用解构,其原因就在这,原理也是通过迭代器一个个取值然后再赋值给对应变量;
const str = 'abc'
const nums = [1, 2, 3] const [str1, str2, str3] = str
console.log(str1, str2, str3) // a b c const [num1, num2, num3] = nums
console.log(num1, num2, num3) // 1 2 3 const [name1, name2, name3] = iteratorObj
console.log(name1, name2, name3) // curry kobe klay
注意:在扩展语法和解构语法中,我们知道数组可以使用,但是对象也可以使用呀,为什么没有提到对象呢?因为对象的扩展和解构是在ES9中新增的特性,其原理并不是使用迭代器实现的,只是ECMA提供给我们的一种操作对象的新语法而已;
JS创建对象:new Map([Iterable])、new WeakMap([Iterable])、new Set([Iterable])、new WeakSet([Iterable])等。
// 1.Set
const set = new Set(iteratorObj)
console.log(set) // Set(3) { 'curry', 'kobe', 'klay' } // 2.Array.from
const names = Array.from(iteratorObj)
console.log(names) // [ 'curry', 'kobe', 'klay' ]
JS方法调用:Promise.all(Iterable)、Promise.race(Iterable)、Array.from(Iterable)等。
// 传入的可迭代对象中的每个值,会使用Promise.resolve进行包裹
Promise.all(iteratorObj).then(res => {
console.log(res) // [ 'curry', 'kobe', 'klay' ]
})
扩展:现在我们都知道了for...of可用于遍历一个可迭代对象,如果在遍历过程中终端了呢?因为使用break、continue、return、throw
都是可以中断遍历的,既然for...of遍历的原理是基于迭代器的,那么在for...of中进行中断操作,一定是可以被迭代器监听到的,上面说了,在迭代器中有一个next方法,其实还可以指定一个return方法,如果遍历过程中断了,就会去调用return方法,注意return方法也要返回和next方法一样的对象。这种情况就称之为迭代器的中断。
const iteratorObj = {
names: ['curry', 'kobe', 'klay'],
[Symbol.iterator]: function() {
let index = 0
return {
next: () => {
if (index < this.names.length) {
return { done: false, value: this.names[index++] }
} else {
return { done: true, value: undefined }
}
},
return() {
console.log('哎呀,我被中断了!')
return { done: true, value: undefined }
}
}
}
}
for (const item of iteratorObj) {
console.log(item)
if (item === 'kobe') break
}
1.3.4.自定义可迭代类
上面提到了对象不是一个可迭代对象,所以对象不能使用for...of遍历,如果我们想要实现通过for...of遍历对象呢?那么可以自己实现一个类,这个类的实例化对象是可迭代对象。
- 实现一个Person类,并且Person类中实现了
Symbol.iterator
方法用于返回一个迭代器; - Person类的实例化对象
p
中包含一个friends数组,通过for...of遍历p
对象时,可以将friends数组的每一项遍历出来;
class Person {
constructor(name, age, friends) {
this.name = name
this.age = age
this.friends = friends
}
[Symbol.iterator]() {
let index = 0
return {
next: () => {
if (index < this.friends.length) {
return { done: false, value: this.friends[index++] }
} else {
return { done: true, value: undefined }
}
}
}
}
}
简单看一下效果:
const p = new Person('curry', 30, ['kobe', 'klay', 'green'])
for (const name of p) {
console.log(name) // kobe klay green
}
2.生成器(Generator)
2.1.什么是生成器?
生成器是ES6中新增的一种控制函数执行的方案,它可以帮助我们控制函数的暂停和执行。生成器是一种特殊的迭代器,所以生成器也是一个对象,并且可以调用next方法。那么怎么创建一个生成器对象呢?
创建生成器对象需要使用生成器函数,生成器函数和普通函数不一样,主要有以下特点:
- 生成器函数的声明需要在function后加上一个符号
*
; - 在生成器函数中可以使用
yield
关键字来分割函数体代码,控制函数的执行; - 生成器函数调用的返回值就是生成器对象了;
2.2.生成器的基本使用
实现一个生成器函数,该函数的执行可以通过返回的生成器对象进行控制。
function* generatorFn() {
console.log('函数开始执行~')
console.log('函数第一段代码执行...')
yield
console.log('函数第二段代码执行...')
yield
console.log('函数第三段代码执行...')
yield
console.log('函数第四段代码执行...')
console.log('函数执行结束~')
}
// 调用generatorFn获取生成器
const generator = generatorFn()
generator.next()
console.log('------------------------')
generator.next()
console.log('------------------------')
generator.next()
console.log('------------------------')
generator.next()
2.3.生成器next方法的返回值
上面说到了生成器是一种特殊的迭代器,那么调用next方法肯定也是有返回值的,并且返回值是一个包含done、value属性的对象。
function* generatorFn() {
console.log('函数第一段代码执行...')
yield
console.log('函数第二段代码执行...')
yield
console.log('函数第三段代码执行...')
yield
console.log('函数第四段代码执行...')
}
// 调用generatorFn获取生成器
const generator = generatorFn()
console.log(generator.next())
console.log('------------------------')
console.log(generator.next())
console.log('------------------------')
console.log(generator.next())
console.log('------------------------')
console.log(generator.next())
从打印结果可以看出来,next返回的对象中value是没有值的,当执行到最后一段代码后,done的值就为true了:
如果需要指定next返回值中的value,那么可以通过在yield
后面跟上一个值或者表达式,就可以将对应的值传递到next返回对象value中了。
function* generatorFn() {
console.log('函数第一段代码执行...')
yield 10
console.log('函数第二段代码执行...')
yield 20
console.log('函数第三段代码执行...')
yield 30
console.log('函数第四段代码执行...')
}
// 调用generatorFn获取生成器
const generator = generatorFn()
console.log(generator.next())
console.log('------------------------')
console.log(generator.next())
console.log('------------------------')
console.log(generator.next())
console.log('------------------------')
console.log(generator.next())
观察以上打印结果,在执行完第四段代码后,调用的next返回值为{ value: undefined, done: true }
,原因是后面已经没有yield
了,而且当函数没有指定返回值时,最后会默认执行return undefined
。
2.4.生成器next方法的参数传递
在前面介绍迭代器定义时,提到迭代器的next可以传递0个或1个参数,而可以传递1个参数的情况就是生成器的next可以传递一个参数,而给每一段代码传递过去的参数可以通过yield来接收。
function* generatorFn(value) {
console.log('函数第一段代码执行...', value)
const value1 = yield 10
console.log('函数第二段代码执行...', value1)
const value2 = yield 20
console.log('函数第三段代码执行...', value2)
const value3 = yield 30
console.log('函数第四段代码执行...', value3)
}
// 调用generatorFn获取生成器
const generator = generatorFn('参数0')
console.log(generator.next())
console.log('------------------------')
console.log(generator.next('参数1'))
console.log('------------------------')
console.log(generator.next('参数2'))
console.log('------------------------')
console.log(generator.next('参数3'))
next参数传递解释:
- next中传递的参数是会被上一个
yield
接收的,这样可以方便下面代码使用这个参数,所以给next传递参数,需要从第二个next开始传递; - 如果第一段代码需要使用参数呢?可以在调用生成器函数时传递参数过去,供第一段代码使用;
2.5.生成器的return方法
return方法也可以给生成器函数传递参数,但是调用return后,生成器函数就会中断,之后再调用next就不会再继续生成值了。
function* generatorFn() {
console.log('函数第一段代码执行...')
yield
console.log('函数第二段代码执行...')
yield
console.log('函数第三段代码执行...')
yield
console.log('函数第四段代码执行...')
}
// 调用generatorFn获取生成器
const generator = generatorFn()
console.log(generator.next())
console.log('------------------------')
console.log(generator.next())
console.log('------------------------')
console.log(generator.return(123))
console.log('------------------------')
console.log(generator.next())
console.log(generator.next())
console.log(generator.next())
console.log(generator.next())
console.log(generator.next())
上面的执行return方法,相当于函数内部执行了return:
function* generatorFn() {
console.log('函数第一段代码执行...')
yield
console.log('函数第二段代码执行...')
const value = yield
return value
console.log('函数第三段代码执行...')
yield
console.log('函数第四段代码执行...')
}
// 调用generatorFn获取生成器
const generator = generatorFn()
console.log(generator.next())
console.log('------------------------')
console.log(generator.next())
console.log('------------------------')
console.log(generator.next(123))
console.log('------------------------')
console.log(generator.next())
console.log(generator.next())
console.log(generator.next())
console.log(generator.next())
console.log(generator.next())
2.6.生成器的throw方法
throw方法可以给生成器函数内部抛出异常。
- 生成器调用throw方法抛出异常后,可以在生成器函数中进行捕获;
- 通过
try catch
捕获异常后,后续的代码还是可以正常执行的;
function* generatorFn() {
console.log('函数第一段代码执行...')
yield 10
console.log('函数第二段代码执行...')
try {
yield 20
} catch (err) {
console.log(err)
}
console.log('函数第三段代码执行...')
yield 30
console.log('函数第四段代码执行...')
}
// 调用generatorFn获取生成器
const generator = generatorFn()
console.log(generator.next())
console.log('------------------------')
console.log(generator.next())
console.log('------------------------')
console.log(generator.throw('err message'))
console.log('------------------------')
console.log(generator.next())
2.7.生成器替换迭代器
在前面实现了一个生成迭代器的函数,实现过程还需要进行判断,并自己返回对应的对象,下面就用生成器来实现一个生成迭代器的函数。
方式一:根据数组元素的个数,执行yield;
function* createIterator(arr) {
let index = 0
yield arr[index++]
yield arr[index++]
yield arr[index++]
}
方式二:遍历数组,执行yield;
function* createIterator(arr) {
for (const item of arr) {
yield item
}
}
方式三:执行
yield*
,后面可以跟上一个可迭代对象,它会依次迭代其中每一个值;function* createIterator(arr) {
yield* arr
}
测试一下以上三种方法,执行结果都是一样的:
const names = ['curry', 'kobe', 'klay']
const iterator = createIterator(names)
console.log(iterator.next()) // { value: 'curry', done: false }
console.log(iterator.next()) // { value: 'kobe', done: false }
console.log(iterator.next()) // { value: 'klay', done: false }
console.log(iterator.next()) // { value: undefined, done: true }
3.异步请求的处理方案
在进行异步请求时,如果出现了这样一个需求,下一次的请求需要拿到上一次请求的结果。如下是使用Promise封装的一个request方法。
function request(url) {
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve(url)
}, 300)
})
}
实现上面的需求可以怎么做呢?
3.1.方案一:使用Promise的then进行嵌套调用
- 在then中拿到结果后,再去做下一次请求,以此类推;
- 缺点:形成了回调地狱;
request('/aaa').then(res => {
request(res + '/bbb').then(res => {
request(res + '/ccc').then(res => {
console.log(res) // /aaa/bbb/ccc
})
})
})
3.2.方案二:使用Promise的then的返回值
- 虽然可以解决回调地狱问题,但是阅读性不佳;
request('/aaa').then(res => {
return request(res + '/bbb')
}).then(res => {
return request(res + '/ccc')
}).then(res => {
console.log(res) // /aaa/bbb/ccc
})
3.3.方案三:使用Promise和Generator处理
function* getRequestData() {
const res1 = yield request('/aaa')
const res2 = yield request(res1 + '/bbb')
const res3 = yield request(res2 + '/ccc')
console.log(res3)
}
手动执行生成器的next方法;
const generator = getRequestData()
generator.next().value.then(res => {
generator.next(res).value.then(res => {
generator.next(res).value.then(res => {
generator.next(res) // /aaa/bbb/ccc
})
})
})
自动执行生成器的next方法:如果手动执行嵌套层级过多的话是不方便的,那么可以借助递归的思想实现一个自动执行生成器的函数;
function autoGenerator(generatorFn) {
const generator = generatorFn() function recursion(res) {
const result = generator.next(res)
// 如果done值为true,说明结束了
if (result.done) return result.value
// 没有结束,继续调用Promise的then
result.value.then(res => {
recursion(res)
})
} recursion()
} autoGenerator(getRequestData) // /aaa/bbb/ccc
使用第三方库来执行生成器:像自动执行生成器函数,早就已经有第三方库帮助我们实现了,如
co
;const co = require('co')
co(getRequestData) // /aaa/bbb/ccc
3.4.方案四:使用async和await
async和await是我们解决异步回调的最终解决方案,它可以让我们异步的代码,看上去是同步执行的。
async function getRequestData() {
const res1 = await request('/aaa')
const res2 = await request(res1 + '/bbb')
const res3 = await request(res2 + '/ccc')
console.log(res3)
}
getRequestData() // /aaa/bbb/ccc
4.async和await的原理
相信从上面讲述的四个异步请求的处理方案中,就可以看出来async、await和生成器的关系了。
- 将生成器函数的
*
换成async
; - 将生成器函数中的
yield
换成await
; - 两种方案所体现出的效果和代码书写形式几乎差不多;
总结:
- async和await的原理其实就是Promise+生成器实现的;
- 为什么async、await能够让异步代码看上去是同步执行的,其原因就在于生成器的next方法可以对函数内代码的执行进行控制,当上一次请求拿到结果后,再去执行下一次next;
- 所以为什么说async和await只是Promise+生成器的语法糖,其原理就在这;
掌握JavaScript中的迭代器和生成器,顺便了解一下async、await的原理的更多相关文章
- JavaScript中的迭代器和生成器[未排版]
JavaScript中的迭代器 在软件开发领域,"迭代"的意思是按照顺序反复多次执行一段程序,通常会有明确的终止条件. ECMAScript 6规范新增了两个高级特性:迭代器和生成 ...
- python is、==区别;with;gil;python中tuple和list的区别;Python 中的迭代器、生成器、装饰器
1. is 比较的是两个实例对象是不是完全相同,它们是不是同一个对象,占用的内存地址是否相同 == 比较的是两个对象的内容是否相等 2. with语句时用于对try except finally 的优 ...
- pytorch :: Dataloader中的迭代器和生成器应用
在使用pytorch训练模型,经常需要加载大量图片数据,因此pytorch提供了好用的数据加载工具Dataloader. 为了实现小批量循环读取大型数据集,在Dataloader类具体实现中,使用了迭 ...
- Python中的迭代器和生成器
本文以实例详解了python的迭代器与生成器,具体如下所示: 1. 迭代器概述: 迭代器是访问集合元素的一种方式.迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束.迭代器只能往前不会后 ...
- 终于理解Python中的迭代器和生成器了!
迭代器和生成器 目录 迭代器和生成器 可迭代对象和迭代器 基础概念 判断 for循环本质 不想用for循环迭代了,如何使用迭代器? 列表推导式 生成器Generator 概念 如何实现和使用? 生成器 ...
- 学习迭代器实现C#异步编程——仿async/await(一)
.NET 4.5的async/await真是个神奇的东西,巧妙异常以致我不禁对其实现充满好奇,但一直难以窥探其门径.不意间读了此篇强文<Asynchronous Programming in C ...
- javascript中的迭代器
1.forEach迭代器 forEach方法接收一个函数作为参数,对数组中每个元素使用这个函数,只调用这个函数,数组本身没有任何变化 //forEach迭代器 function square(num) ...
- Python中的迭代器、生成器
from collections import Iterable, Iterator 1. 可迭代(iterable)对象 参考官网链接 class I: def __init__(self, v): ...
- python中的迭代器和生成器学习笔记总结
生成器就是一个在行为上和迭代器非常类似的对象. 是个对象! 迭代,顾名思意就是不停的代换的意思,迭代是重复反馈过程的活动,其目的通常是为了逼近所需目标或结果.每一次对过程的重复称为一次“迭代”,而 ...
随机推荐
- web Javascript360°全景实现
360 全景浏览是一种性价比很高的虚拟现实解决方案,给人一种全新的浏览体验,让你足不出户就能身临其境地感受到现场的环境.该技术被广泛地应用在房产.酒店.家居等领域. 下面我们使用三种方法讨论一个 36 ...
- 【自动化基础】allure描述用例详细讲解及实战
前言 allure可以输出非常精美的测试报告,也可以和pytest进行完美结合,不仅可以渲染页面,还可以控制用例的执行.下面就对allure的使用进行一个详细的介绍和总结. 需要准备的环境: pyth ...
- VUE3 之 状态动画 - 这个系列的教程通俗易懂,适合新手
1. 概述 老话说的好:不用羡慕别人,每个人都有属于自己的人生道路,重要的是在前进道路上遇见阻碍时,如何去积极的面对并解决. 言归正传,今天我们来聊聊 VUE 的状态动画. 2. 状态动画 2.1 数 ...
- Docker——镜像讲解
镜像是什么 镜像是一种轻量级,可执行的独立软件包,用来打包软件运行环境和基于运行环境开发的软件,它包含运行某个软件所需的所有内容,包括代码.运行时.库.环境变量和配置文件. 所有的应用,直接打包doc ...
- CVE-2017-7269(IIS远程代码执行)
利用CVE-2017-7269让IIS向自己的msf反弹一个shell 漏洞编号:CVE2017-7269 服务器版本:Windows server 2003 中间件:IIS6.0 攻击工具:meta ...
- 【Vulnhub】LazySysAdmin
下载链接 https://download.vulnhub.com/lazysysadmin/Lazysysadmin.zip 运行环境 Virtualbox Vnware Workstation p ...
- 如何在 Java 中实现二叉搜索树
二叉搜索树 二叉搜索树结合了无序链表插入便捷和有序数组二分查找快速的特点,较为高效地实现了有序符号表.下图显示了二叉搜索树的结构特点(图片来自<算法第四版>): 可以看到每个父节点下都可以 ...
- (数据科学学习手札135)tenacity:Python中最强大的错误重试库
本文示例代码及文件已上传至我的Github仓库https://github.com/CNFeffery/DataScienceStudyNotes 1 简介 我们在编写程序尤其是与网络请求相关的程序, ...
- 前端工程化 Webpack基础
前端工程化 模块化 (js模块化,css模块化,其他资源模块化) 组件化 (复用现有的UI结构.样式.行为) 规范化 (目录结构的划分.编码规范化.接口规范化.文档规范化.Git分支管理) 自动化 ( ...
- 解决IDEA的maven项目 添加依赖后Reimport无反应
解决IDEA的maven项目 添加依赖后Reimport无反应 如果重启项目和编译器都不管用的话, 找到项目在硬盘上的位置 把该项目的.idea文件夹和xxx.iml文件删除 打开IDEA ...