图片懒加载

实现方案

  1. 在img元素时, 自定义一个属性data-src,用于存放图片的地址
  2. 获取屏幕可是区域的尺寸
  3. 获取元素到窗口边缘的距离
  4. 判断元素是否在可视区域内.在则将data-src的值赋给src,否则不执行其他操作

实质

当图片在可视区域内时,才加载,否则不加载;也可一个给个默认的图片占位

用到的api

  1. IntersectionObserver 它提供了一种异步观察目标元素与顶级文档viewport的交集中的变化的方法
  2. window.requestIdleCallback() 方法将浏览器的空闲时段内调用的函数排队. 这使开发者在主事件循环上执行后台和底优先级工作, 而不会影响延迟关键事件, 如动画和输入响应

几个细节

  1. 提前加载,可以+100 像素
  2. 滚动时只处理未加载的图片即可
  3. 函数节流

简单代码示例

判断是否是在可视区域的三种方法

  1. 屏幕可视区域的高度 + 滚动条滚动距离 > 元素到文档顶部的距离
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
    107
    108
    109
    110
    111
    112
    113
    114
    115
    116
    117
    118
    document.documentElement.clientHeight + document.documentElement.scrollTop > element.offsetTop
    ```

    2. 使用getBoundingClientRect() 获取 元素大小和位置
    3. IntersectionObserver 自动观察元素是否在可视区域内

    ```html
    <!DOCTYPE html>
    <html lang="en">
    <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>图片懒加载</title>
    <style>
    img {
    display: block;
    height: 450px;
    margin-bottom: 20px;
    border: 1px solid blue;
    width: 450px;
    }
    </style>
    </head>
    <body>
    <img data-src="./images/1.png" alt="" />
    <img data-src="./images/2.png" alt="" />
    <img data-src="./images/3.png" alt="" />
    <img data-src="./images/4.png" alt="" />
    <img data-src="./images/5.png" alt="" />
    <img data-src="./images/6.png" alt="" />
    </body>

    <script>
    var imgs = document.querySelectorAll('img')

    // 节流函数, 定时器版本
    function throttle(func, wait) {
    let timer = null

    return (...args) => {
    if (!timer) {
    func(...args)
    timer = setTimeout(() => {
    timer = null
    }, wait)
    }
    }
    }

    // H + S > offsetTop
    function lazyLoad1(imgs) {
    // offsetTop 是元素与offsetParent 的距离, 循环获取直到页面顶部
    function getTop(e) {
    var T = e.offsetTop
    while ((e = e.offsetParent)) {
    T += e.offsetTop
    }
    return T
    }

    var H = document.documentElement.clientHeight // 获取可视区域的高度
    var S =
    document.documentElement.scrollTop || document.body.scrollTop

    // + 100 像素 提前100个像素就开始加载
    // 并且只处理没有src即没有加载过的图片
    Array.from(imgs).forEach((img) => {
    if (H + S + 100 > getTop(img) && !img.src) {
    img.src = img.dataset.src
    }
    })
    }

    const throttleLazyLoad1 = throttle(lazyLoad1, 200)

    function lazyLoad2(imgs) {
    function isIn(el) {
    var bound = el.getBoundingClientRect()
    var clientHeight = window.innerHeight
    return bound.top <= clientHeight + 100
    }

    Array.from(imgs).forEach((img) => {
    if (isIn(img) && !img.src) {
    img.src = img.dataset.src
    }
    })
    }

    const throttleLazyLoad2 = throttle(lazyLoad2, 200)

    // 监听滚动事件
    window.onload = window.onscroll = () => {
    // throttleLazyLoad1(imgs)
    throttleLazyLoad2(imgs)
    }

    function lazyLoad3(imgs) {
    const io = new IntersectionObserver((ioes) => {
    ioes.forEach((ioe) => {
    const img = ioe.target
    const intersectionRatio = ioe.intersectionRatio
    if (intersectionRatio > 0 && intersectionRatio <= 1) {
    if (!img.src) {
    img.src = img.dataset.src
    }
    }

    img.onload = img.onerror = () => io.unobserve(img)
    })
    })

    imgs.forEach((img) => io.observe(img))
    }

    // lazyLoad3(imgs)
    </script>
    </html>

API说明

getBoundingClientRect

  1. 返回值是一个DOMRect对象, 这个对象是由该元素的getClientRects() 方法返回的一组矩形的集合, 就是该元素的css边框大小.返回的结果是包含完整元素的最小矩形,并且拥有 left,top,right,bottom,x, y, width, height 这几个以像素为单位的只读属性用于描述整个边框, 除了width和height以外的属性是相对于视图窗口的左上角来计算的
  2. 如果需要获得相对于整个网页左上角定位的属性,那么只要给top、left属性加上当前的滚动位置,(通过window.scrollX和window.scrollY),这样就可以获取与当前滚动位置无关的值

IntersectionObserver交叉观察器http://www.ruanyifeng.com/blog/2016/11/intersectionobserver_api.html

IntersectionObserverEntry 对象提供目标元素信息

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
{
time: 3893.92,
rootBounds: ClientRect {
bottom: 920,
height: 1024,
left: 0,
right: 1024,
top: 0,
width: 920
},
boundingClientRect: ClientRect {
// ...
},
intersectionRect: ClientRect {
// ...
},
intersectionRatio: 0.54,
target: element
}
  1. time: 可见性发生变化的时间, 是一个高精度时间戳,单位为毫秒
  2. target: 被观察的元素,是一个DOM节点对象
  3. rootBounds: 根元素的矩形区域的信息, getBoundingClientRect()方法的返回值,如果没有根元素(即直接相对于视口滚动), 则返回null
  4. boundingClientRect: 当前目标元素的矩形区域信息
  5. intersectionRect: 目标元素与视口(或根元素)的交叉区域的信息
  6. intersectionRatio: 目标元素的可见比例, 即intersectionRect占boundingClientRect的比例,完全时可为1, 完全不可见时小于等于0

上图中,灰色的水平方框代表视口, 深红色的区域代表四个被观察的元素,

vue-diff算法

处理流程

  1. 处理头部的同类型节点
  2. 处理头部的同类型节点
  3. 处理头尾/尾头同类型的节点
  4. 处理新增的节点
  5. 处理更新的节点
  6. 处理需要删除的节点

图示

1
2
3
vdom前 [ 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 ]

vdom后 [ 1, 9, 11, 7, 3, 4, 5, 6, 2, 10 ]

第一步 设置初始化值

vdom前设置指针 oldLeft 指向第一个值, oldRight 指向最后一个值

1
2
oldLeft = 1;
oldRight = 10;

vdom后设置指针 newLeft 指向第一个值, newRight 指向最后一个值

1
2
newLeft = 1;
newRight = 10;

第二步 对比头部

对比oldLeft和newLeft的值相同,则指针后移动

1
2
3
4
5
6
7
oldLeft = 2;
oldRight = 10;
newLeft = 9;
newRight = 10;

[1,2,3,4,5,6,7,8,9,10]
[ 1, 9, 11, 7, 3, 4, 5, 6, 2, 10 ]

第三步 对比尾部

  1. 对比头部发现不一致

  2. 对比尾部发现一直,这里指 oldRight = 10 与 newRight = 10, 尾部指针向前移动;

    1
    2
    3
    4
    5
    6
    oldLeft = 2;
    oldRight = 9;
    newLeft = 9;
    newRight = 2;
    [1,2,3,4,5,6,7,8,9,10]
    [ 1, 9, 11, 7, 3, 4, 5, 6, 2, 10 ]

    第四步 对比头尾

  3. 对比头部发现不一致

  4. 对比尾部发现不一致

  5. 对比头尾发现一致, 则将2号位置的dom移动到oldRight之后, oldLeft向后移动一位,newRight向前移动一位

1
2
3
4
5
6
oldLeft = 3;
oldRight = 9;
newLeft = 9;
newRight = 6;
[1,3,4,5,6,7,8,9,2, 10]
[ 1, 9, 11, 7, 3, 4, 5, 6, 2, 10 ]

第五步 对比尾头,

  1. 对比头部发现不一致
  2. 对比尾部发现不一致
  3. 对比头尾发现不一致,对比尾头发现一致, 将 oldRight 复制到 oldLeft前,oldRight向前移动一位, newLeft向前移动一位
1
2
3
4
5
6
oldLeft = 3;
oldRight = 8;
newLeft = 11;
newRight = 6;
[1,9,3,4,5,6,7,8,2,10]
[ 1, 9, 11, 7, 3, 4, 5, 6, 2, 10 ]

第六步 插入新元素

  1. 对比头部发现不一致
  2. 对比尾部发现不一致
  3. 对比头尾发现不一致
  4. 对比尾头发现不一致
  5. 发现新增节点11,则在oldLeft之前插入新元素,其他保持不变
1
2
3
4
5
6
oldLeft = 3;
oldRight = 8;
newLeft = 7;
newRight = 6;
[1, 9, 11, 3, 4, 5, 6, 7, 8, 2, 10]
1, 9, 11, 7, 3, 4, 5, 6, 2, 10 ]

第七步 删除节点

  1. 对比头部节点发现不一致
  2. 对比尾部节点发现不一致
  3. 对比头尾发现不一致
  4. 对比为头发现不一致
  5. 不是新增节点,也不需要更新, 则删除节点, oldRight向前移动一位
1
2
3
4
5
6
7
oldLeft = 3;
oldRight = 7;
newLeft = 7;
newRight = 6

[ 1, 9, 11, 3, 4, 5, 6, 7, 2, 10]
[ 1, 9, 11, 7, 3, 4, 5, 6, 2, 10 ]

第八步 为头节点相同

将尾部节点制动到oldLeft, oldLeft指向不变

1
2
3
4
5
6
oldLeft = 3;
oldRight = 6;
newLeft = 3;
newRight = 6;
[ 1, 9, 11, 7, 3, 4, 5, 6, 2, 10]
[ 1, 9, 11, 7, 3, 4, 5, 6, 2, 10 ]

第九步 对比头部,发现一致走完算法流程

参考:

Vue3的domdiff和vDOM优化

编译模版的静态标记

1
2
3
4
5
<div id="app">
<p>周一呢</p>
<p>明天就周二了</p>
<div>{{week}}</div>
</div>

vue2 会被解析成一下代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
function render() {
with (this) {
return _c(
'div',
{
attrs: {
id: 'app',
},
},
[
_c('p', [_v('周一呢')]),
_c('p', [v('明天就周二了')]),
_c('div', [_v(_s(week))]),
]
)
}
}

可以看出,两个p标签是完全静态的,以至于后续渲染中,其实没有任何变化, 但是在vue2.x中依然会使用_c新建成一个vdom.在diff的时候依然需要去比较,这就造成了一定量的性能消耗

在vue3中

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import {
createVNode as _createVNode,
toDisplayString as _toDisplayString,
} from 'vue'

export function render(_ctx, _cache) {
return (
_openBlock(),
_createVNode('div', { id: 'app' }, [
_createVNode('p', null, '周一呢'),
_createVNode('p', null, '明天就周二了'),
_createVNode(
'div',
null,
_toDisplayString(_ctx.week),
1 /* TEXT */
),
])
)
}

只有当_createVNode 的第四个参数不为空的时候,这时才会被遍历, 而静态节点就不会被遍历到

同时发现了在vue3最后一个非静态的节点编译后: 出现了 /* TEXT */, 这是为了标记当前内容的类型以便进行diff, 如果不同的标记,只需要去比较对比相同的类型,这就不会去浪费时间对其他类型进行遍历

1
2
3
4
5
6
7
8
9
10
11
12
13
export const enum PatchFlags {
TEXT = 1, // 表示具有动态textContent的元素
CLASS = 1 << 1, // 表示具有动态Class元素
STYLE = 1 << 2, // 表示动态样式 (静态如 style="color:red", 也会提升至动态)
PROPS = 1 << 3, // 表示具有非类/样式动态道具元素
FULL_PROPS = 1 << 4, // 表示带有动态键的道具的元素, 与上面的三种
HYDRARE_EVENTS = 1 << 5, // 表示带有事件监听器的元素
STABLE_FRAGMENT = 1 << 6, // 表示其子顺序不变的片段
KEYED_FRAGMENT = 1 << 7, // 表示带有key的元素片段
UNKEYED_FRAGMENT = 1 << 8, // 表示没有Key的元素片段
NEED_PATCH = 1 << 9, // 表示只需要非属性补丁的元素, 例如 ref或 hooks
DYNAMIC_SLOTS = 1 << 10, // 表示具有动态插槽的元素
}

如果存在两种类型, 那么只需要对这两个值对应的pathflag进行位或运算

如 TEXT 和 PROPS

1
2
TEXT: 1, PROPS: 1<<3 = 8, 
// 那么对1和8进行1|8 就能得到9;

事件存储

绑定事件会存储在缓存中

1
2
3
<div id="app">
<button @click="handleClick">周五拉</button>
</div>

经过转换

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
import {
createVNode as _createVNode,
toDisplayString as _toDisplayString,
} from 'vue'

export function render(_ctx, _cache) {
return (
_openBlock(),
_createVNode('div', { id: 'app' }, [
_createVNode(
'button',
{
onClick:
_cache[1] ||
(_cache[1] = ($event, ...args) =>
_ctx.handleClick($event, ...args)),
},
'周五啦'
),
])
)
}

在代码中可以看出在绑定点击事件的时候, 会生成并缓存了一个内联函数cache中,变成了一个静态的节点

静态提升

1
2
3
4
5
6
7
8
<template>
<div id="app">
<p>周一了</p>
<p>周二了</p>
<div>{{ week }}</div>
<div :class="{ red: isRed }">周三呢</div>
</div>
</template>

转换成

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
import {
createVNode as _createVNode,
toDisplayString as _toDisplayString,
} from 'vue'

const _hoisted_1 = { id: 'app' }
const _hoisted_2 = /*#__PURE__*/ _createVNode(
'p',
null,
'周一了',
-1 /* HOISTED */
)
const _hoisted_3 = /*#__PURE__*/ _createVNode(
'p',
null,
'周二了',
-1 /* HOISTED */
)

export function render(_ctx, _cache) {
return (
_openBlock(),
_createVNode('div', _hoisted_1, [
_hoisted_2,
_hoisted_3,
_createVNode(
'div',
null,
_toDisplayString(_ctx.week),
1 /* TEXT */
),
_createVNode(
'div',
{
class: { red: _ctx.isRed },
},
'周三呢',
2 /* CLASS */
),
])
)
}

在这里可以看出将一些静态的节点放在了render函数的外部,这样就避免了每次render都会生成一次静态节点

全家桶修改

vite的使用放弃了vue2.x使用的webpack

  1. 开发服务器启动后不需要进行打包操作
  2. 可以自定义开发服务器 const {createServe} = require('vite')
  3. 热模块替换的性能和模块数量无关, 替换变快, 即时热模块替换
  4. 生产环境和rollup捆绑

其他

  1. 提供了treeShaking,在打包的时候自动去除没有用到的vue模块
  2. 更好的ts支持,类型定义提示,tsx支持,class组件支持

Vue3 beta 新优势

optionsAPI –> compositionAPi

composititionAPI 字面意思就是组合API, 它是为了实现基于函数逻辑复用机制而产生的

举个简单的例子

声明变量

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
const { reactive } = Vue

const App = {
template: `
<div>
{{message}}
</div>
`,
setup() {
const state = reactive({ message: 'HelloWorld!!!' })
return {
...state,
}
},
}

Vue.createApp().mount(App, '#app')

双向绑定

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
const { reactive } = Vue

const App = {
template: `
<div>
<input type="text" v-model="state.value"/>
</div>
`,
setup() {
const state = reactive({ value: 'Hello Vue' })
return {
state,
}
},
}

Vue.createApp().mount(App, '#app')

  • setup
    • 被诟病的地方,内容要写在这个地方,setup实际上是一个组件的入口, 它运在组件被实例化的时候, props 属性被定义后,实际上等价于vue2版本的beforeCreate 和Created这两个生命周期
  • reactive
    • 创建一个响应式的状态,几乎等价于vue2.x中的Vue.observable()API, 为了避免于rxjs中的observable混淆进行了重命名

观察属性

1
2
3
4
5
6
7
8
9
10
11
12
13
14
import { reactive, watchEffect } from 'vue'

const state = reactive({
count: 0,
})

watchEffect(() => {
document.body.innerHtml = `count is ${state.count}`
})

return {
...state,
}

  • watchEffect 和 2.x 中的watch选项类似,但是它不需要把被依赖的数据源和副作用回调分开. 组合式API同样提供了一个watch函数,其行为和2.x的选项完全一致.

ref

vue3 允许用户创建单个响应对象

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
const App = {
template: `
<div>
{{value}}
</div>
`,

setup() {
const value = ref(0)
return { value }
},
}

Vue.createApp().mount(App, '#app')

计算属性

1
2
3
4
5
6
7
8
9
10
11
12
13
14
setup() {
const state = reactive({
count: 0,
double: computed(() => state.count*2)
})
function increment() {
state.count++;
}

return {
state,
increment
}
}

生命周期的变更

vue2 vue3
beforeCreate setup
created setup
beforeMount onBeforeMount
mounted onMounted
beforeUpdate onBeforeUpdate
updated onUpdated
beforeDestory onBeforeUnmount
destroyed onUnmounted
errorCaptured onErrorCaptured

生命周期使用举例

1
2
3
4
5
6
7
8
9
10
import { onMounted } from 'vue'

export default {
setup() {
onMounted(() => {
console.log(`component is mounted`)
})
},
}

preformance 优化

  1. 重构了虚拟DOM,保持了兼容性, 使dom 脱离模版渲染,提升性能
  2. 优化了模版编译过程,增加patchFlag,遍历节点的时候,会跳过静态节点
  3. 高效的组件初始化
  4. 组件upload的过程性能提升1.3~2倍
  5. SSR速读提升2~3被

参考:

<每日前端APP> 连接暂无

loader与plugin的区别

主要区别

loader

主要用于加载某些资源文件.因为webpack本省只能打包commonjs规范的js文件,对于其他资源例如css, 图片, 或者其他语法集,比如jsx,coffee, 是没有办法加载的,这就需要对应的loader将资源转换,加载进来,从字面的意思可以看出,loader是用与加载的,他的作用是一个一个文件上.

plugin

用于扩展webpack的功能,它直接作用与webpack, 扩展了它的功能,当然loader也时变相的扩展了webpack.但是它只能用于转化文件这个一个领域,而plugin的功能更加的丰富,而不仅局限于资源加载

常用的Plugin与loader

常用的Plugin

  • CommonsChunkPlugin 创建一个公用的chunk,常用于将第三坊lib抽取公用js, 例如:
1
2
3
4
5
6
7
8
9
10
entry : {
vendor: ['jquery', 'other-lib'],
app:'./entry.js'
}

new CommonsChunkPlugin({
name: 'vendor',
filename: 'vendor.js',
minChunks:Infinity
})
  • HotModuelReplacementPlugin 启动热更新

常用的loader

loader的功能就是加载资源到webpack

  1. css 和 style

css-loader 遍历所有的require的css文件,输出文件内容

style-loader 将css内容输出到页面的style标签中

所以,在webpack.config.js中.css的配置是这样的

1
2
{test: /\.css$/, loader: "style!css}

style!css 类似一种输出重定向, css-loader的输出会作为 style-loader的输入

如果使用了css预处理. 比如less , 那么只需要在最后加上less的loader

1
{test:'/\.css$/',loader: "style!css!less"}

另一种写法(推荐)

1
{test:/\.css$/,loaders:["style","css","less"]}

总结

loader用于加载待打包的资源,
plugin用于扩展webpack

参考:

https://blog.csdn.net/wp270280522/article/details/51496436

webpack插件

知识点

  1. 一个简单的插件构成
  2. webpack构建流程
  3. Tapable是如何把各个插件串联到一起的
  4. compiler以及compileation对象的使用以及它们对应的事件钩子

插件的基本构成

plugins 是可以用自身原型方法apply来实例化的对象, apply只在安装插件被webpack compiler执行一次. apply方法传入一个webpack compiler 的应用,来访问编译器回调

一个简单的插件结构:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
class HelloPlugin {
// 在构造函数中获取用户给该插件传入的配置
constructor(options) {

}

// webpack 会调用 HelloPlugin函数,用来特定时机处理额外的逻辑
apply(compiler) {
compiler.hooks.emit.tap('HelloPlugin', (compilation)=>{
// 在功能流程完成后可以调用webpack提供的回调
})

//如果事件是异步的,会带两个参数, 第二个参数为回调函数,在插件处理完任务时需要调用回调函数通知webpack处理完毕,才能进入下一个流程
compiler.plugin('emit', function(compilation, callback) {
// 支持处理逻辑
// 处理完毕后执行callback以通知webpack
// 如果不执行callback, 运行流程将会一直卡在这里不往下执行
callback()

})
}
}

插件安装时, 只需要将它的一个实例放到webpack.config.js的plugins数组里面

1
2
3
4
5
6
7
8
9
10

const HelloPlugin = require('./hello-plugin.js')

var webpackConfig = {
plugins: [
new HelloPlugin({
options:true
})
]
}

webpack Plugin的工作原理

  1. 读取配置过程中,开会先执行 new HelloPlugin(options)初始化一个HelloPlugin 获取其实例
  2. 初始化 compiler 对象后调用 HelloPlugin.apply(compiler)给插件实例传入compiler对象
  3. 插件实例在获取到compiler对象后,就可以通过compiler.plugin(事件名,回调函数)监听到webpack广播出来的事件,并且可以通过compiler对象去操作webpack

webpack 构建流程

webpack的基本构建流程如下:

  1. 校验配置文件: 读取命令传入或webpack.config.js文件,初始化本次构建配置参数
  2. 生成Compiler对象: 执行配置文件中的价差实例化语句 new MyWebpackPlugin(),为webpack事件流挂上自定义hooks
  3. 引入entryOption阶段: webpack开始读取配置的Entires,递归遍历所有的入口文件
  4. run/watch: 如果运行在watch模式则执行watch方法,否则执行run方法
  5. compileation: 创建Compilation对象回调compilation相关钩子,依次进入每一个入口文件(entry),使用loader对文件进行编译.通过compilation 我可以读取到module的resource(资源路径)、loaders(使用loader)等信息.在将编译好的文件内容使用acorn 解析成AST静态语法数.然后递归、重复的执行这个过程,所有模块和依赖分析完成后,执行compilation的seal放大对每个chunk进行整理、优化、封装__webpack_require__来模拟模块化操作
  6. emit: 所有文件的编译及转化都已经完成,包含了最终输出的资源,我梦可以在传入事件回调的compilation.asstes上拿到所需要的数据,其中包括即将输出的资源、代码块Chunk等信息
1
2
3
4
5
6
7
8
9
//修改或添加资源
compilation.assets[`new-file.js`] = {
source() {
return `var a= 1;`
}
size() {
return this.source().length;
}
}
  1. afterEmit: 文件已经写入磁盘完成
  2. done: 完成编译

webpack流程图

webpack编译

  1. webpack 配置处理
    1. 错误检查、增加默认配置等
  2. 编译前的准备工作
    1. 处理webpack配置中的plugin、webpack自己一堆的plugin、初始化compiler等等
  3. 开始编译主入口
  4. resolve阶段: 解析文件路径&loaders
    1. 将文件的绝对路径解析出;同时解析出inline和我们配置在webpack.config.js中匹配的loaders,将loaders解析为固定的格式以及loaders执行文件的路径
  5. loaders逐个执行
  6. parse阶段
    1. 将文件转换为ast树,解析出import和export等
  7. 递归处理依赖
  8. module一些优化&增加id
  9. 生成chunk,决定每个chunk中包含的module
  10. 生成文件
    1. 耕具模板生成文件名称;生成文件,这里会再次用到前面parse阶段的内容,将import xxx``export xxx 替换成webpack的写法
  11. 写文件,结束

参考

webpack插件机制

理解事件流机制tapable

webpkac本质上是一种事件流机制, 他的工作流程就是将各个插件串联起来,而实现这一切的核心就是Tapable

webpack的tapable事件流机制保证了插件的有序性, 将各个插件串联起来,webpack 在运行过程中会广播事件, 插件知识需要监听它所关心的事件,就能加入到这条webpack机制中去,去去改变webpack的运作,使得整个系统的扩展性良好

tapable 也是一个小型的library, 是webpakc的核心工具. 类似于node中的events库,核心原理就是一个订阅发布模式,作用是提供类似的插件接口

webpack中最核心的负责编译的Compiler和负责构建bundles的Compilation都是Tapable的实例,可以直接在Compiler和Compilation对象上广播和监听事件,方法如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
/**
* 广播事件
* event-name 为事件名称,注意不要和现有的事件名重复
*
*/

compiler.apply('event-name', params)
compilation.apply('event-name', params)

/**
* 监听事件
*/
compiler.plugin("event-name", function(params) {})
compilation.plugin("event-name", function(params){}),

Tabable 类暴露了 tap、tapAsync、和tapPromise 方法,可以根据钩子的同步/异步方式来选择一个函数注入逻辑

tap同步钩子

1
2
3
compiler.hooks.compile.tap("MyPlugin", params =>{
console.log('以同步方式出发compile钩子')
})

tapAsync 异步钩子, 通过 callback回调告诉webpack异步执行完毕tapPromise 异步钩子,返回一个Promise告诉webpack异步执行完毕

1
2
3
4
5
6
7
8
9
10
compiler.hooks.run.tapAsync("MyPlugin",(compiler,callback) =>{
console.log("异步方式触及 run 钩子。");
callback();
})

compiler.hooks.run.tapPromise("MyPlugin", compiler =>{
return new Promise(resolve=> setTImeojut(resolve,1000)).then=>{
console.log("以具有延迟的异步方式触及 run 钩子")
}
}),

Tapable 用法

1
2
3
4
5
6
7
8
9
10
11
const { 
SyncHook, // 同步钩子
SyncBailHook, // 同步保险钩子
SyncWaterfallHook, // 同步瀑布沟组
SyncLoopHook, // 同步循环钩子
AsyncParallelHook, // 异步
AsyncParallelBailHook, // 异步并醒保险钩子
AsyncSeriesHook, // 异步串行钩子
AsyncSeriesBailHook, // 异步串行保险钩子
AsyncSeriesWaterfallHook // 异步串行瀑布钩子
} = require("tapable")

简单实现一个SyncHook

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
class Hook {
constructor(args) {
this.taps = [];
this.interceptors = [];
this._args = args;
}

tap(name, fn) {
this.taps.push({name, fn})
}
}

class SyncHook extends Hook {
call(name, fn) {
try {
this.taps.forEach(tap =>tap.fn(name))
fn(null, name)
} catch(error) {
fn(error)
}
}
}

关联 Tapable 和 webpack 插件

Compiler.js

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
const {AsyncSeriesHook, SyncHook} = reuqire("tapable");

// 创建类
class Compiler {
constructor() {
this.hooks = {
run : new AsyncSeriesHook(["compiler“]), // 异步钩子
compile: new SyncHook(["params"]) // 同步钩子
};
}

run() {
// 执行异步钩子
this.hooks.run.callAsync(this, err => {
this.comile(onCompiled)
})
}

compile() {
// 执行同步钩子, 并传参
this.hooks.compile.call(params);
}

}

module.exports = Compiler

MyPlugin.js

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25

const Compiler = require("./Compiler");

class MyPlugin {
apply(compiler) { // 接收compiler参数
compiler.hooks.run.tap("MyPlugin", ()=> console.log("开始编译...."))
compiler.hooks.compiler.tapAsync("MyPlugin",(name,age)=>{
setTimeout(()=> {
console.log("编译中")
},1000)
})
}
}

// 这里类似webpack.config.js的plugin配置
// 向 plugins属性传入new 实例

const myPlugin = new MyPlugin();

const options = {
plugins: [myPlugin]
}

let compiler = new Compiler(options)
compiler.run();

参考

tapable 源码解析

理解Compiler(负责编译)

Compiler 对象包含了当前运行webpack的配置,包括entry、 output、loaders等配置,这个对象在启动webpack时被实例化, 而且全局唯一的,plugin可以通过对象获取到webpack配置星系进行处理

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
// Compiler 结构
Compiler {
_pluginCompat: SyncBailHook {
...
},
hooks: {
shouldEmit: SyncBailHook {
...
},
done: AsyncSeriesHook {
...
},
additionalPass: AsyncSeriesHook {
...
},
beforeRun: AsyncSeriesHook {
...
},
run: AsyncSeriesHook {
...
},
emit: AsyncSeriesHook {
...
},
assetEmitted: AsyncSeriesHook {
...
},
afterEmit: AsyncSeriesHook {
...
},
thisCompilation: SyncHook {
...
},
compilation: SyncHook {
...
},
normalModuleFactory: SyncHook {
...
},
contextModuleFactory: SyncHook {
...
},
beforeCompile: AsyncSeriesHook {
...
},
compile: SyncHook {
...
},
make: AsyncParallelHook {
...
},
afterCompile: AsyncSeriesHook {
...
},
watchRun: AsyncSeriesHook {
...
},
failed: SyncHook {
...
},
invalid: SyncHook {
...
},
watchClose: SyncHook {
...
},
infrastructureLog: SyncBailHook {
...
},
environment: SyncHook {
...
},
afterEnvironment: SyncHook {
...
},
afterPlugins: SyncHook {
...
},
afterResolvers: SyncHook {
...
},
entryOption: SyncBailHook {
...
},
infrastructurelog: SyncBailHook {
...
}
},
...
outputPath: '',//输出目录
outputFileSystem: NodeOutputFileSystem {
...
},
inputFileSystem: CachedInputFileSystem {
...
},
...
options: {
//Compiler对象包含了webpack的所有配置信息,entry、module、output、resolve等信息
entry: [
'babel-polyfill',
'/Users/frank/Desktop/fe/fe-blog/webpack-plugin/src/index.js'
],
devServer: { port: 3000 },
output: {
...
},
module: {
...
},
plugins: [ MyWebpackPlugin {} ],
mode: 'production',
context: '/Users/frank/Desktop/fe/fe-blog/webpack-plugin',
devtool: false,
...
performance: {
maxAssetSize: 250000,
maxEntrypointSize: 250000,
hints: 'warning'
},
optimization: {
...
},
resolve: {
...
},
resolveLoader: {
...
},
infrastructureLogging: { level: 'info', debug: false }
},
context: '/Users/frank/Desktop/fe/fe-blog/webpack-plugin',//上下文,文件目录
requestShortener: RequestShortener {
...
},
...
watchFileSystem: NodeWatchFileSystem {
//监听文件变化列表信息
...
}
}

Compiler 源码精简版代码解析

https://github.com/webpack/webpack/blob/master/lib/Compiler.js

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
const { SyncHook, SyncBailHook, AsyncSeriesHook } = require("tapable");
class Compiler {
constructor() {
// 1. 定义生命周期钩子
this.hooks = Object.freeze({
// ...只列举几个常用的常见钩子,更多hook就不列举了,有兴趣看源码
done: new AsyncSeriesHook(["stats"]),//一次编译完成后执行,回调参数:stats
beforeRun: new AsyncSeriesHook(["compiler"]),
run: new AsyncSeriesHook(["compiler"]),//在编译器开始读取记录前执行
emit: new AsyncSeriesHook(["compilation"]),//在生成文件到output目录之前执行,回调参数: compilation
afterEmit: new AsyncSeriesHook(["compilation"]),//在生成文件到output目录之后执行
compilation: new SyncHook(["compilation", "params"]),//在一次compilation创建后执行插件
beforeCompile: new AsyncSeriesHook(["params"]),
compile: new SyncHook(["params"]),//在一个新的compilation创建之前执行
make:new AsyncParallelHook(["compilation"]),//完成一次编译之前执行
afterCompile: new AsyncSeriesHook(["compilation"]),
watchRun: new AsyncSeriesHook(["compiler"]),
failed: new SyncHook(["error"]),
watchClose: new SyncHook([]),
afterPlugins: new SyncHook(["compiler"]),
entryOption: new SyncBailHook(["context", "entry"])
});
// ...省略代码
}
newCompilation() {
// 创建Compilation对象回调compilation相关钩子
const compilation = new Compilation(this);
//...一系列操作
this.hooks.compilation.call(compilation, params); //compilation对象创建完成
return compilation
}
watch() {
//如果运行在watch模式则执行watch方法,否则执行run方法
if (this.running) {
return handler(new ConcurrentCompilationError());
}
this.running = true;
this.watchMode = true;
return new Watching(this, watchOptions, handler);
}
run(callback) {
if (this.running) {
return callback(new ConcurrentCompilationError());
}
this.running = true;
process.nextTick(() => {
this.emitAssets(compilation, err => {
if (err) {
// 在编译和输出的流程中遇到异常时,会触发 failed 事件
this.hooks.failed.call(err)
};
if (compilation.hooks.needAdditionalPass.call()) {
// ...
// done:完成编译
this.hooks.done.callAsync(stats, err => {
// 创建compilation对象之前
this.compile(onCompiled);
});
}
this.emitRecords(err => {
this.hooks.done.callAsync(stats, err => {

});
});
});
});

this.hooks.beforeRun.callAsync(this, err => {
this.hooks.run.callAsync(this, err => {
this.readRecords(err => {
this.compile(onCompiled);
});
});
});

}
compile(callback) {
const params = this.newCompilationParams();
this.hooks.beforeCompile.callAsync(params, err => {
this.hooks.compile.call(params);
const compilation = this.newCompilation(params);
//触发make事件并调用addEntry,找到入口js,进行下一步
this.hooks.make.callAsync(compilation, err => {
process.nextTick(() => {
compilation.finish(err => {
// 封装构建结果(seal),逐次对每个module和chunk进行整理,每个chunk对应一个入口文件
compilation.seal(err => {
this.hooks.afterCompile.callAsync(compilation, err => {
// 异步的事件需要在插件处理完任务时调用回调函数通知 Webpack 进入下一个流程,
// 不然运行流程将会一直卡在这不往下执行
return callback(null, compilation);
});
});
});
});
});
});
}
emitAssets(compilation, callback) {
const emitFiles = (err) => {
//...省略一系列代码
// afterEmit:文件已经写入磁盘完成
this.hooks.afterEmit.callAsync(compilation, err => {
if (err) return callback(err);
return callback();
});
}

// emit 事件发生时,可以读取到最终输出的资源、代码块、模块及其依赖,并进行修改(这是最后一次修改最终文件的机会)
this.hooks.emit.callAsync(compilation, err => {
if (err) return callback(err);
outputPath = compilation.getPath(this.outputPath, {});
mkdirp(this.outputFileSystem, outputPath, emitFiles);
});
}
// ...省略代码
}

apply 方法中插入钩子的一般形式如下;

1
2
3
4
5
6
// compiler提供了compiler.hooks,可以根据这些不同的时刻去让插件做不同的事情
compiler.hooks.阶段.tap函数("插件名称", (阶段回调参数)=>{

})

compiler.run(callback)

理解Compilation (负责创建Bundles)

Compilation对象代表了一次资源版本构建.当运行webpack开发环境中间件时,每当检测到一个文件的变化,就会创建一个新的compilation,从而生成一组新的编译资源,一个Compilation对象表现了当前的模块资源、编译生成资源、编译文件、以及被跟踪依赖的状态信息, 简单来讲就是把本次打包编译的内容存到内存里,Compilation对象也提供了插件需要自定义功能的回调,以插件做自定义处理时选择使用拓展.

简单来说,Compilation的职责就是构建模块和Chunk, 并利用插件优化构建过程.

和Compiler用法相同,钩子类型不同名,也可以在某些钩子上访问tapAsync和taoPromsie

参考:

常见的Compilation Hooks区别

钩子 类型 什么时候调用
buildModule SyncHook 在模块开始编译前触发名可以用于修改模块
successModule SyncHook 在一个模块被修改成功编译,会执行这个钩子
finishMoudles AsyncSeriesHook 当所有模块都编译成功后调用
seal SyncHook 当一次compilation停止接收新模块时触发
optimizeDependencies SyncBailHook 当依赖优化开始执行
optimize SyncHook 在优化阶段开始执行
optimizeMoudules SyncBailHook 在模块优化阶段开始时执行, 插件可以在这个钩子颗粒执行的模块的优化,回调参数modules
optimizeChunks SyncBailHook 在代码块优化阶段开始执行, 插件可以在这个钩子里执行对代码快的优化,回调参数chunks
optimizeChunkAssets AsyncSeriesHook 优化任何代码快资源,在这些资源存放在compilation.assets上,一个chunk有一个files属性,他指向有一个chunk创建的所有文件,任何额外的chunk资源都存放在compilation.additionalChunkAssets上.回调参数: chunks
optimizeAssets AsyncSeriesHook 优化说有存放在compilation.assets的所有资源,回调参数assets

Compiler 和 Compilation 的区别

Compilaer 代表了整个webpack从启动到关闭的生命周期,而Compilation只代表了一次新的编译,只要文件有改动,compilation就会被重新创建

常用 API

插件可以用来修改输出文件、增加输出文件、甚至可以提升webpack性能、总之插件可以通过调用webpack提供的API能完成很多事情,由于webpack提供的API非常多,

读取输出资源、代码块、模块以及依赖

有些插件可能需要读取webpack的处理结果,例如输出资源、代码块、模块及其依赖,以便做下一步处理、在emit事件发生时, 代表源文件的转换和组装已经完成,在这里可以读取到最终将输出的资源、代码块、模块及其依赖,并且可以修改输出资源的内容.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
class Plugin {
apply(compiler) {
compiler.plugin('emit', function (compilation, callback) {
// compilation.chunks 存放所有代码块,是一个数组
compilation.chunks.forEach(function (chunk) {
// chunk 代表一个代码块
// 代码块由多个模块组成,通过 chunk.forEachModule 能读取组成代码块的每个模块
chunk.forEachModule(function (module) {
// module 代表一个模块
// module.fileDependencies 存放当前模块的所有依赖的文件路径,是一个数组
module.fileDependencies.forEach(function (filepath) {
});
});

// Webpack 会根据 Chunk 去生成输出的文件资源,每个 Chunk 都对应一个及其以上的输出文件
// 例如在 Chunk 中包含了 CSS 模块并且使用了 ExtractTextPlugin 时,
// 该 Chunk 就会生成 .js 和 .css 两个文件
chunk.files.forEach(function (filename) {
// compilation.assets 存放当前所有即将输出的资源
// 调用一个输出资源的 source() 方法能获取到输出资源的内容
let source = compilation.assets[filename].source();
});
});

// 这是一个异步事件,要记得调用 callback 通知 Webpack 本次事件监听处理结束。
// 如果忘记了调用 callback,Webpack 将一直卡在这里而不会往后执行。
callback();
})
}
}

监听文件变化

webpack会从配置入口模块触发, 依次找出所有依赖模块,当入口模块或则其依赖的模块发生变化时, 就会触发一次新的Compilation

在开发插件时经常需要知道哪个文件发生变化导致了新的Compilation,为此可以使用如下代码

1
2
3
4
5
6
7
8
9
10
11
// 当依赖的文件发生变化时会触发 watch-run 事件
compiler.hooks.watchRun.tap('MyPlugin', (watching, callback) => {
// 获取发生变化的文件列表
const changedFiles = watching.compiler.watchFileSystem.watcher.mtimes;
// changedFiles 格式为键值对,键为发生变化的文件路径。
if (changedFiles[filePath] !== undefined) {
// filePath 对应的文件发生了变化
}
callback();
});

默认情况下webpack只会监控入口文件及其依赖的模块是否发生变化,在有些情况下项目可能需要引入新文件, 例如引入一个HTML文件.由于javascript文件不会去导入HTML文件,webpack就不会监听HTML文件的变化,编辑HTML文件就不会重新触发新的Compilation.为了监听HTML文件的变化,我们需要把HTML文件加入到依赖列表中,

1
2
3
4
5
compiler.hooks.afterCompile.tap('MyPlugin', (compilation, callback) => {
// 把 HTML 文件添加到文件依赖列表,好让 Webpack 去监听 HTML 模块文件,在 HTML 模版文件发生变化时重新启动一次编译
compilation.fileDependencies.push(filePath);
callback();
});

修改输出资源

在有些场景下插件需要修改、增加、删除输出的资源, 要做到这点需要监听emit事件,因为发出emit事件时所有模块的转换和代码对应的文件已经生成好了,需要输出的资源即将输出,因此emit事件是修改webpack输出资源的最后的时机

所有的需要输出的资源都会存放在compilation.assets中, compilation.assets是一个键值对,键为需要输出的文件名称,值为文件对应的内容
设置Comilation.assets的代码如下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
// 设置名称为 fileName 的输出资源
compilation.assets[fileName] = {
// 返回文件内容
source: () => {
// fileContent 既可以是代表文本文件的字符串,也可以是代表二进制文件的 Buffer
return fileContent;
},
// 返回文件大小
size: () => {
return Buffer.byteLength(fileContent, 'utf8');
}
};
callback();

判断webpack使用了哪些插件

1
2
3
4
5
6
7
8
// 判断当前配置使用使用了 ExtractTextPlugin,
// compiler 参数即为 Webpack 在 apply(compiler) 中传入的参数
function hasExtractTextPlugin(compiler) {
// 当前配置所有使用的插件列表
const plugins = compiler.options.plugins;
// 去 plugins 中寻找有没有 ExtractTextPlugin 的实例
return plugins.find(plugin=>plugin.__proto__.constructor === ExtractTextPlugin) != null;
}

参考:
webpack学习-pluign

管理Warnings和 Errors

如果在apply函数内插入throw new Error(“Message”), 终端会打印出Unhandle rejection Error:Message. 然后webpack中断执行.为了不影响compilation.warnings和compilation.errors

1
2
3
compilation.warnings.push("warning");
compilation.errors.push("error");

参考

https://juejin.im/post/6844904161515929614

jenkins安装

安装

拉取镜像

1
docker pull jenkins

创建卷积

1
docker volume create --name jenkins_home

运行 docker

1
2
3
4
5
6
7
docker run \
--name jenkins \
-itd \
-p 8080:8080 \
-p 50000:50000 \
-v jenkins_home:/var/jenkins_home \
jenkins

删除容器

docker container rm jenkins

暂停服务

docker container stop jenkins

copy 文件

docker cp jenkins.war jenkins:/usr/share/jenkins/

配置

输入密码

密码保存在/var/jenkins_home/secrets/initialAdminPassword目录下

进入 docker container bash, 打印密码

1
2
3
docker exec -it jenkins /bin/bash
cat /var/jenkins_home/secrets/initialAdminPassword
//7c5f6c209bd24a57b50b1decf50564e8

创建项目

常用函数式编程库

Rxjs

事件流

1
2
3
4
5
6
7
8
9
import { range } from "rxjs";
import { map, filter } from "rxjs/operators";

range(1, 200)
.pipe(
filter(x => x % 2 === 1),
map(x => x + x)
)
.subscribe(x => console.log(x));
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
import { Subject, from } from "rxjs";
import { debounceTime, distinctUntilChanged, switchMap } from "rxjs/operators";

const searchItems = new Subject();

searchItems
.pipe(
debounceTime(300),
distinctUntilChanged(),
switchMap((val) => from(getSuggestList(val)))
)
.subscribe((x) => console.log(x));

const input = document.getElementById("input");

input.oninput = search;

function search(val) {
searchItems.next(val.target.value);
}

function getSuggestList(val) {
return new Promise((resolve) => {
console.log(val);
setTimeout(() => {
resolve([
{ id: 1, name: "zhangsan" },
{ id: 2, name: "lisi" },
]);
});
});
}


lodash