0%

前言

Vue3 官网中有下面这样一张图,基本展现出了 Vue3 的渲染原理:

alt text

本文会从源码角度来简单的看一下 Vue3 的运行全流程,旨在加深对上图的理解。

初始化渲染

从下面这个很简单的使用示例开始:

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

createApp({
template: `
<div class="card">
<button type="button" @click="count++">count is {{ count }}</button>
</div>
`,
setup() {
return {
count: ref(0),
}
},
}).mount('#app')

首先调用了 createApp 方法:

1
2
3
4
5
export const createApp = ((...args) => {
const app = ensureRenderer().createApp(...args)
// ...
return app
})

我们可以看到重点在于 ensureRenderer

1
2
3
4
5
6
7
8
9
const rendererOptions = /*#__PURE__*/ extend({ patchProp }, nodeOps)
// 延时创建渲染器,当用户只依赖响应式包的时候,可以通过 tree-shaking 移除核心渲染逻辑相关的代码
// 因为 ensureRenderer 是在执行 createApp 的时候调用的,如果你不执行 createApp 而只使用 vue 的一些响应式 API,就不会创建这个渲染器,所以说延时渲染。
function ensureRenderer() {
return (
renderer ||
(renderer = createRenderer<Node, Element | ShadowRoot>(rendererOptions))
)
}

rendererOptions是一个对象,渲染相关的一些配置,比如更新属性的方法,操作 DOM 的方法。

这么做主要是方便跨平台,比如在其他非浏览器环境,可以替换成对应的节点操作方法。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
// rendererOptions
{
insert: (child, parent, anchor) => {
parent.insertBefore(child, anchor || null)
},

remove: child => {
const parent = child.parentNode
if (parent) {
parent.removeChild(child)
}
},
// ...
}
1
2
3
function createRenderer(options: RendererOptions) {
return baseCreateRenderer(options)
}

调用 baseCreateRenderer, 这个函数简直可以用庞大来形容,patch、mount、diff均在这个方法中实现,回头我们再来细看实现,现在我们只需要关心他最后返回的什么

1
2
3
4
5
6
7
8
9
10
11
function baseCreateRenderer(
options: RendererOptions,
createHydrationFns?: typeof createHydrationFunctions,
) {
// ...此处省略2000行
return {
render,
hydrate,
createApp: createAppAPI(render, hydrate),
}
}

我们发现createApp是通过createAppAPI方法调用返回的,下面我们看createAppAPI。

createAppAPI

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
export function createAppAPI<HostElement>(
render: RootRenderFunction<HostElement>,
hydrate?: RootHydrateFunction,
): CreateAppFunction<HostElement> {
return function createApp(rootComponent, rootProps = null) {
if (!isFunction(rootComponent)) {
rootComponent = extend({}, rootComponent)
}

if (rootProps != null && !isObject(rootProps)) {
__DEV__ && warn(`root props passed to app.mount() must be an object.`)
rootProps = null
}

const context = createAppContext()
const installedPlugins = new WeakSet()

let isMounted = false

const app: App = (context.app = {
_uid: uid++,
_component: rootComponent as ConcreteComponent,
_props: rootProps,
_container: null,
_context: context,
_instance: null,

version,

get config() {
return context.config
},

set config(v) {
if (__DEV__) {
warn(
`app.config cannot be replaced. Modify individual options instead.`,
)
}
},
// 都是一些眼熟的方法
// 注册plugin
use(plugin: Plugin, ...options: any[]) {
//...
},
// 全局混入
mixin(mixin: ComponentOptions) {
//...
},
// 注册全局组件
component(name: string, component?: Component): any {
//...
},
// 注册全局指令
directive(name: string, directive?: Directive) {
//...
},
// 挂载根组件
mount(
rootContainer: HostElement, // 挂载容器
isHydrate?: boolean,
namespace?: boolean | ElementNamespace,
): any {
//...
},
// 卸载
unmount() {
//...
},
// 分享数据
provide(key, value) {
//...
}
//...
})
//...
return app
}
}

这个就是最终的 createApp方法,接受了 rootComponent 和 rootProps 两个参数,我们在应用层面执行 createApp(App) 方法时,会把 App 组件对象作为根组件传递给 rootComponent。这样,createApp 内部就创建了一个 app 对象。

所谓的应用实例app其实就是一个对象,我们传进去的组件选项作为根组件存储在_component属性上,另外还可以看到应用实例提供的一些方法,比如注册插件的use方法,挂载实例的mount方法等。

createAppContext 实现

context其实也是一个普通对象:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
export function createAppContext(): AppContext {
return {
app: null as any,
config: {
isNativeTag: NO,
performance: false,
globalProperties: {},
optionMergeStrategies: {},
errorHandler: undefined,
warnHandler: undefined,
compilerOptions: {},
},
mixins: [],
components: {},
directives: {},
provides: Object.create(null),
optionsCache: new WeakMap(),
propsCache: new WeakMap(),
emitsCache: new WeakMap(),
}
}

这部分是vue的初次渲染逻辑,首先官方解构了mount方法, 然后又重写了app.mount,并调用normalizeContainer校验挂载元素,临时保存了需要渲染的内容。并对vue2的写法做了兼容处理,挂载元素进行渲染。

在以上整个 app 对象创建过程中,Vue.js 利用闭包和函数柯里化的技巧,很好地实现了参数保留。比如,在执行 app.mount 的时候,并不需要传入渲染器 render,这是因为在执行 createAppAPI 的时候渲染器 render 参数已经被保留下来了。

至此,createApp的流程大概到此结束,下一部分来分析mount挂载部分。

mounted挂载

重写app.mount方法

createApp 函数中,首先取出 app 对象中的 mount 函数,然后通过 app.mount = () => {} 对 mount 函数进行重写:

  1. 首先调用 normalizeContainer 函数来获取container节点;
  2. 判断该节点是否存在,若不存在,则直接返回;
  3. 清空container的innerHTML;
  4. 调用mount函数。
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
export const createApp = ((...args) => {
const app = ensureRenderer().createApp(...args)

const { mount } = app

/*
(1)这里重写的目的是考虑到跨平台(app.mount里面只包含和平台无关的代码)
(2)这些重写的代码是完善web平台下的渲染逻辑(比如其他平台也可以进行类似的重写)
*/

app.mount = (containerOrSelector: Element | ShadowRoot | string): any => {
// 根组件容器 选择器字符串兼容
const container = normalizeContainer(containerOrSelector)
if (!container) return
// rootComponent
const component = app._component
// 如组件对象没有定义 render 函数和 template 模板,则取容器的 innerHTML 作为组件模板内容
if (!isFunction(component) && !component.render && !component.template) {
// __UNSAFE__
// Reason: potential execution of JS expressions in in-DOM template.
// The user must make sure the in-DOM template is trusted. If it's
// rendered by the server, the template should not contain any user data.
component.template = container.innerHTML
// 2.x compat check
/**
* __COMPAT__ 是启动的时候通过rollup去注入进去的
* 用来判断是否向下兼容
*/
if (__COMPAT__ && __DEV__) {
for (let i = 0; i < container.attributes.length; i++) {
const attr = container.attributes[i]
if (attr.name !== 'v-cloak' && /^(v-|:|@)/.test(attr.name)) {
compatUtils.warnDeprecation(
DeprecationTypes.GLOBAL_MOUNT_CONTAINER,
null,
)
break
}
}
}
}

// clear content before mounting
container.innerHTML = ''
// 调用初始app.mount,挂载元素进行渲染
const proxy = mount(container, false, resolveRootNamespace(container))
if (container instanceof Element) {
// vue2的话,会给#app设置一个v-cloak属性,在render的时候清空掉
container.removeAttribute('v-cloak')
container.setAttribute('data-v-app', '')
}
return proxy
}

return app
})

首先是通过 normalizeContainer 标准化容器(这里可以传字符串选择器或者 DOM 对象,但如果是字符串选择器,就需要把它转成 DOM 对象,作为最终挂载的容器),然后做一个 if 判断,如果组件对象没有定义 render 函数和 template 模板,则取容器的 innerHTML 作为组件模板内容;接着在挂载前清空容器内容,最终再调用 app.mount 的方法走标准的组件渲染流程。

在这里,重写的逻辑都是和 Web 平台相关的,所以要放在外部实现。此外,这么做的目的是既能让用户在使用 API 时可以更加灵活,也兼容了 Vue.js 2.x 的写法,比如 app.mount 的第一个参数就同时支持选择器字符串和 DOM 对象两种类型。

从 app.mount 开始,才算真正进入组件渲染流程。

真正的挂载mounted

核心流程:

  1. 根据传入的根组件App创建vnode
  2. 渲染vnode。

mount方法

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
mount(
rootContainer: HostElement,
isHydrate?: boolean,
namespace?: boolean | ElementNamespace,
) {
if (!isMounted) {
// #5571
if (__DEV__ && (rootContainer as any).__vue_app__) {
warn(
`There is already an app instance mounted on the host container.\n` +
` If you want to mount another app on the same host container,` +
` you need to unmount the previous app by calling \`app.unmount()\` first.`,
)
}
const vnode = createVNode(rootComponent, rootProps)
// store app context on the root VNode.
// this will be set on the root instance on initial mount.
vnode.appContext = context

if (namespace === true) {
namespace = 'svg'
} else if (namespace === false) {
namespace = undefined
}

// HMR root reload
if (__DEV__) {
context.reload = () => {
// casting to ElementNamespace because TS doesn't guarantee type narrowing
// over function boundaries
render(
cloneVNode(vnode),
rootContainer,
namespace as ElementNamespace,
)
}
}

if (isHydrate && hydrate) {
hydrate(vnode as VNode<Node, Element>, rootContainer as any)
} else {
render(vnode, rootContainer, namespace)
}
isMounted = true
app._container = rootContainer
// for devtools and telemetry
;(rootContainer as any).__vue_app__ = app

if (__DEV__ || __FEATURE_PROD_DEVTOOLS__) {
app._instance = vnode.component
devtoolsInitApp(app, version)
}

return getExposeProxy(vnode.component!) || vnode.component!.proxy
} else if (__DEV__) {
warn(
`App has already been mounted.\n` +
`If you want to remount the same app, move your app creation logic ` +
`into a factory function and create fresh app instances for each ` +
`mount - e.g. \`const createMyApp = () => createApp(App)\``,
)
}
}

createVNode

createVNode方法会根据组件的类型生成一个标志,后续会通过这个标志做一些优化处理。我们传的是一个组件选项,也就是一个普通对象,shapeFlag的值为4。后续我们会重点关注我们的主线组件 vnode 和普通元素 vnode。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
const createVNode = _createVNode;
function _createVNode(type, props = null, children = null, patchFlag = 0, dynamicProps = null, isBlockNode = false) {
// ...
const shapeFlag = isString(type)
? 1 /* ShapeFlags.ELEMENT */
: isSuspense(type)
? 128 /* ShapeFlags.SUSPENSE */
: isTeleport(type)
? 64 /* ShapeFlags.TELEPORT */
: isObject(type)
? 4 /* ShapeFlags.STATEFUL_COMPONENT */
: isFunction(type)
? 2 /* ShapeFlags.FUNCTIONAL_COMPONENT */
: 0;
// ...
return createBaseVNode(type, props, children, patchFlag, dynamicProps, shapeFlag, isBlockNode, true);
}

然后调用了createBaseVNode方法:

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
function createBaseVNode(
type: VNodeTypes | ClassComponent | typeof NULL_DYNAMIC_COMPONENT,
props: (Data & VNodeProps) | null = null,
children: unknown = null,
patchFlag = 0,
dynamicProps: string[] | null = null,
shapeFlag = type === Fragment ? 0 : ShapeFlags.ELEMENT,
isBlockNode = false,
needFullChildrenNormalization = false,
) {
const vnode = {
__v_isVNode: true,
__v_skip: true,
type,
props,
key: props && normalizeKey(props),
ref: props && normalizeRef(props),
scopeId: currentScopeId,
slotScopeIds: null,
children,
component: null,
suspense: null,
ssContent: null,
ssFallback: null,
dirs: null,
transition: null,
el: null,
anchor: null,
target: null,
targetAnchor: null,
staticCount: 0,
shapeFlag,
patchFlag,
dynamicProps,
dynamicChildren: null,
appContext: null,
ctx: currentRenderingInstance,
} as VNode

// ...

return vnode
}

可以看到返回的虚拟DOM也是一个普通对象,我们传进去的组件选项会存储在type属性上。

render

虚拟DOM创建完后就会调用render方法将虚拟DOM渲染为实际的DOM节点,render方法就是在baseCreateRenderer中创建的,通过参数传给createAppAPI的:

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
const render: RootRenderFunction = (vnode, container, namespace) => {
// 新vnode不存在
if (vnode == null) {
// 卸载
if (container._vnode) {
unmount(container._vnode, null, null, true)
}
} else {
// 创建或者更新组件
patch(
container._vnode || null, // 如果是首次渲染,container._vnode不存在,那么直接将新VNode渲染为DOM元素即可
vnode,
container,
null,
null,
null,
namespace,
)
}
if (!isFlushing) {
isFlushing = true
flushPreFlushCbs()
flushPostFlushCbs()
isFlushing = false
}
// 缓存 vnode 节点,表示已经渲染
container._vnode = vnode
}

如果要渲染的新VNode不存在,那么从容器元素上获取之前VNode进行卸载,否则调用patch方法进行打补丁,如果是首次渲染,container._vnode不存在,那么直接将新VNode渲染为DOM元素即可,否则会对比新旧VNode,使用diff算法进行打补丁,Vue2中使用的是双端diff算法,Vue3中使用的是快速diff算法。

patch

看看render方法里调用的patch方法:

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
const patch: PatchFn = (
n1, // n1 表示旧的vnode,当n1为null时就表示是一次挂载(挂载or更新由n1决定)
n2, // n2 表示新的vnode,根据n2的type进行不同的处理
container, // 渲染后会将vnode渲染到container上
anchor = null,
parentComponent = null,
parentSuspense = null,
namespace = undefined,
slotScopeIds = null,
optimized = __DEV__ && isHmrUpdating ? false : !!n2.dynamicChildren,
) => {
if (n1 === n2) {
return
}

// patching & not same type, unmount old tree
// 如果新的节点和旧的节点类型不同,则销毁旧节点
if (n1 && !isSameVNodeType(n1, n2)) {
anchor = getNextHostNode(n1)
unmount(n1, parentComponent, parentSuspense, true)
n1 = null
}

if (n2.patchFlag === PatchFlags.BAIL) {
optimized = false
n2.dynamicChildren = null
}

const { type, ref, shapeFlag } = n2
switch (type) {
case Text: //... 处理文本节点
case Comment: //... 处理注释节点
case Static: //... 处理静态节点
case Fragment: //... 处理Fragment组件节点
default:
if (shapeFlag & ShapeFlags.ELEMENT) {
// 处理普通 DOM 元素
processElement(
n1,
n2,
container,
anchor,
parentComponent,
parentSuspense,
namespace,
slotScopeIds,
optimized,
)
} else if (shapeFlag & ShapeFlags.COMPONENT) {
// 处理组件
processComponent(
n1,
n2,
container,
anchor,
parentComponent,
parentSuspense,
namespace,
slotScopeIds,
optimized,
)
} else if (shapeFlag & ShapeFlags.TELEPORT) {
// 处理 TELEPORT
// ...
} else if (__FEATURE_SUSPENSE__ && shapeFlag & ShapeFlags.SUSPENSE) {
// 处理 SUSPENSE
// ...
} else if (__DEV__) {
warn('Invalid VNode type:', type, `(${typeof type})`)
}
}

// set ref
if (ref != null && parentComponent) {
setRef(ref, n1 && n1.ref, parentSuspense, n2 || n1, !n2)
}
}

patch这个函数有两个功能,一个是根据 vnode 挂载 DOM,一个是根据新旧 vnode 更新 DOM。

patch的多个参数中,重点关注前三个:

  1. 第一个参数 n1 表示旧的 vnode,当 n1 为 null 的时候,表示是一次挂载的过程;

  2. 第二个参数 n2 表示新的 vnode 节点,后续会根据这个 vnode 类型执行不同的处理逻辑;

  3. 第三个参数 container 表示 DOM 容器,也就是 vnode 渲染生成 DOM 后,会挂载到 container 下面。

switch里面根据VNode的类型不同做的处理也不同,因为我们的例子传的是一个组件选项对象,所以会走processComponent处理分支:

processComponent

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
const processComponent = (
n1: VNode | null,
n2: VNode,
container: RendererElement,
anchor: RendererNode | null,
parentComponent: ComponentInternalInstance | null,
parentSuspense: SuspenseBoundary | null,
namespace: ElementNamespace,
slotScopeIds: string[] | null,
optimized: boolean,
) => {
n2.slotScopeIds = slotScopeIds
// n1等于null,表示挂载组件
if (n1 == null) {
if (n2.shapeFlag & ShapeFlags.COMPONENT_KEPT_ALIVE) {
;(parentComponent!.ctx as KeepAliveContext).activate(
n2,
container,
anchor,
namespace,
optimized,
)
} else {
// 调用mountComponent挂载组件
mountComponent(
n2,
container,
anchor,
parentComponent,
parentSuspense,
namespace,
optimized,
)
}
} else { // n1不为null,表示更新组件
updateComponent(n1, n2, optimized)
}
}

根据是否存在旧的VNode判断是调用挂载方法还是更新方法,先看mountComponent方法:

mountComponent
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
const mountComponent: MountComponentFn = (
initialVNode,
container,
anchor,
parentComponent,
parentSuspense,
namespace: ElementNamespace,
optimized,
) => {
// 2.x compat may pre-create the component instance before actually
// mounting
const compatMountInstance =
__COMPAT__ && initialVNode.isCompatRoot && initialVNode.component

// 1.调用ComponentInternalInstance创建组件的实例
const instance: ComponentInternalInstance =
compatMountInstance ||
// 调用createComponentInstance函数创建一个实例对象,其属性皆为没有值
(initialVNode.component = createComponentInstance(
initialVNode,
parentComponent,
parentSuspense,
))

// ...

// resolve props and slots for setup context
if (!(__COMPAT__ && compatMountInstance)) {
if (__DEV__) {
startMeasure(instance, `init`)
}
// 2.设置组件实例
// 初始化组件, 主要是对组件的props/slots进行初始化处理
// 执行setup 生成render函数(所以setup是在所有选项式API钩子之前调用 包括beforeCreate)
setupComponent(instance)
if (__DEV__) {
endMeasure(instance, `init`)
}
}

// setup() is async. This component relies on async logic to be resolved
// before proceeding
if (__FEATURE_SUSPENSE__ && instance.asyncDep) {
// ...
} else {
// 3.调用设置和运行有副作用的渲染函数

// 1 创建一个组件更新函数
// 1.1 render获得vnode
// 1.2 patch(oldVnode, newVnode)
// 2 创建更新机制 new ReactiveEffect(更新函数)
// 执行渲染副作用函数
setupRenderEffect(
instance,
initialVNode,
container,
anchor,
parentSuspense,
namespace,
optimized,
)
}

if (__DEV__) {
popWarningContext()
endMeasure(instance, `mount`)
}
}
  1. 调用ComponentInternalInstance创建组件的实例
  2. 设置组件实例,初始化组件 处理setup的两个参数, 执行setup 生成render函数(所以setup是在所有选项式API钩子之前调用 包括beforeCreate)
  3. 调用设置和运行有副作用的渲染函数

下面我们依次来看这三个方法:

1. createComponentInstance
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
export function createComponentInstance(
vnode: VNode,
parent: ComponentInternalInstance | null,
suspense: SuspenseBoundary | null,
) {
const type = vnode.type as ConcreteComponent
// inherit parent app context - or - if root, adopt from root vnode
const appContext =
(parent ? parent.appContext : vnode.appContext) || emptyAppContext

const instance: ComponentInternalInstance = {
uid: uid++,
vnode,
type,
parent,
appContext,
// 还有非常多属性
// ...
}
// ...

return instance
}
2. setupComponent
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
export function setupComponent(
instance: ComponentInternalInstance,
isSSR = false,
) {
isSSR && setInSSRSetupState(isSSR)

// vnode的props和子元素
const { props, children } = instance.vnode
// 是否是有状态的组件
const isStateful = isStatefulComponent(instance)
// 初始化props
initProps(instance, props, isStateful, isSSR)
// 初始化slots
initSlots(instance, children)
// 执行setupStatefulComponent获取setupResult
const setupResult = isStateful
? setupStatefulComponent(instance, isSSR) // 执行setup
: undefined

isSSR && setInSSRSetupState(false)
return setupResult
}

可以看到setupComponent会初始化props和slots,然后执行setupStatefulComponent,这里主要是执行setup函数,并返回结果

setupStatefulComponent
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
function setupStatefulComponent(
instance: ComponentInternalInstance,
isSSR: boolean,
) {
// 组件描述对象
const Component = instance.type as ComponentOptions

// ...

// 0. create render proxy property access cache
// 创建render proxy属性访问缓存 作用是缓存访问过的属性
instance.accessCache = Object.create(null)
// 1. create public instance / render proxy
// also mark it raw so it's never observed
// 代理ctx,拦截ctx的属性访问 从而实现取值的优先级:setupState > data > props > ctx
instance.proxy = markRaw(new Proxy(instance.ctx, PublicInstanceProxyHandlers))
if (__DEV__) {
exposePropsOnRenderContext(instance)
}
// 2. call setup()
const { setup } = Component
if (setup) {
// 创建setupContext
const setupContext = (instance.setupContext =
setup.length > 1 ? createSetupContext(instance) : null)

// instance赋值给currentInstance
// 设置当前实例为instance 为了在setup中可以通过getCurrentInstance获取到当前实例
// 同时开启instance.scope.on()
const reset = setCurrentInstance(instance)
// 暂停tracking 暂停收集副作用函数
pauseTracking()
// 执行setup
const setupResult = callWithErrorHandling(
setup,
instance,
ErrorCodes.SETUP_FUNCTION,
[
__DEV__ ? shallowReadonly(instance.props) : instance.props,
setupContext,
],
)
// 重新开启副作用收集
resetTracking()
// currentInstance置为空
// activeEffectScope赋值为instance.scope.parent
// 同时instance.scope.off()
reset()

if (isPromise(setupResult)) {
setupResult.then(unsetCurrentInstance, unsetCurrentInstance)
if (isSSR) {
// return the promise so server-renderer can wait on it
return setupResult
.then((resolvedResult: unknown) => {
handleSetupResult(instance, resolvedResult, isSSR)
})
.catch(e => {
handleError(e, instance, ErrorCodes.SETUP_FUNCTION)
})
} else if (__FEATURE_SUSPENSE__) {
// async setup returned Promise.
// bail here and wait for re-entry.
instance.asyncDep = setupResult
if (__DEV__ && !instance.suspense) {
const name = Component.name ?? 'Anonymous'
warn(
`Component <${name}>: setup function returned a promise, but no ` +
`<Suspense> boundary was found in the parent component tree. ` +
`A component with async setup() must be nested in a <Suspense> ` +
`in order to be rendered.`,
)
}
} else if (__DEV__) {
warn(
`setup() returned a Promise, but the version of Vue you are using ` +
`does not support it yet.`,
)
}
} else {
handleSetupResult(instance, setupResult, isSSR)
}
} else {
finishComponentSetup(instance, isSSR)
}
}

在这个方法里会调用组件选项的setup方法,这个函数中返回的对象会暴露给模板和组件实例,看一下handleSetupResult方法:

handleSetupResult
1
2
3
4
5
6
7
8
function handleSetupResult(instance, setupResult, isSSR) {
if (isFunction(setupResult)) {
instance.render = setupResult;
} else if (isObject(setupResult)) {
instance.setupState = proxyRefs(setupResult);
}
finishComponentSetup(instance, isSSR);
}

如果setup返回的是一个函数,那么这个函数会直接被作为渲染函数。否则如果返回的是一个对象,会使用proxyRefs将这个对象转为Proxy代理的响应式对象。

finishComponentSetup

最后又调用了finishComponentSetup方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
function finishComponentSetup(instance, isSSR) {
const Component = instance.type;
if (!instance.render) {
if (!isSSR && compile && !Component.render) {
const template = Component.template ||
resolveMergedOptions(instance).template;
if (template) {
const { isCustomElement, compilerOptions } = instance.appContext.config;
const { delimiters, compilerOptions: componentCompilerOptions } = Component;
const finalCompilerOptions = extend(extend({
isCustomElement,
delimiters
}, compilerOptions), componentCompilerOptions);
Component.render = compile(template, finalCompilerOptions);
}
}
instance.render = (Component.render || NOOP);
}
}

这个函数主要是判断组件是否存在渲染函数render,如果不存在则判断是否存在template选项,我们传的组件选项显然是没有render属性,而是传的模板template,所以会使用compile方法来将模板编译成渲染函数。

3. setupRenderEffect
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
const setupRenderEffect: SetupRenderEffectFn = (
instance,
initialVNode, // 组件 vnode
container,
anchor,
parentSuspense,
namespace: ElementNamespace,
optimized,
) => {
// 组件更新方法
const componentUpdateFn = () => {}

// create reactive effect for rendering
// 创建一个effect, 将componentUpdateFn更新方法传入响应式更新方法
const effect = (instance.effect = new ReactiveEffect(
componentUpdateFn,
NOOP,
() => queueJob(update),
instance.scope, // track it in component's effect scope
))

// 组件更新函数
const update: SchedulerJob = (instance.update = () => {
if (effect.dirty) {
effect.run()
}
})
update.id = instance.uid
// allowRecurse
// #1801, #2043 component render effects should allow recursive updates
toggleRecurse(instance, true)

if (__DEV__) {
effect.onTrack = instance.rtc
? e => invokeArrayFns(instance.rtc!, e)
: void 0
effect.onTrigger = instance.rtg
? e => invokeArrayFns(instance.rtg!, e)
: void 0
update.ownerInstance = instance
}
// 执行组件更新
update()
}

这一步就涉及到Vue3的响应式原理了,核心就是使用Proxy拦截数据,然后在属性读取时将属性和读取该属性的函数(称为副作用函数)关联起来,然后在更新该属性时取出该属性关联的副作用函数出来执行。

简化后的ReactiveEffect类就是这样的:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
let activeEffect;
class ReactiveEffect {
constructor(fn, scheduler = null, scope) {
this.fn = fn;
}
run() {
activeEffect = this;
try {
return this.fn();
} finally {
activeEffect = null
}
}
}

执行它的run方法时会把自身赋值给全局的activeEffect变量,然后执行副作用函数时如果读取了Proxy代理后的对象的某个属性时就会将对象、属性和这个ReactiveEffect示例关联存储起来,如果属性发生改变,会取出关联的ReactiveEffect实例,执行它的run方法,达到自动更新的目的。

componentUpdateFn
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
const componentUpdateFn = () => {
// 判断组件是否已经挂载
if (!instance.isMounted) {
let vnodeHook: VNodeHook | null | undefined
const { el, props } = initialVNode
// 生命周期和父instance
const { bm, m, parent } = instance
const isAsyncWrapperVNode = isAsyncWrapper(initialVNode)
toggleRecurse(instance, false)
// beforeMount hook
if (bm) {
invokeArrayFns(bm)
}
// onVnodeBeforeMount
// ...
if (el && hydrateNode) {
// ssr 相关
} else {
if (__DEV__) {
startMeasure(instance, `render`)
}
// 执行render函数获得subTree(也是一个vnode) 将subTree挂载到instance上 以供更新使用
const subTree = (instance.subTree = renderComponentRoot(instance)) // 整个组件渲染生成 DOM 对应的 vnode 树
// ...
// patch subTree初次挂载
patch(
null,
subTree,
container,
anchor,
instance,
parentSuspense,
isSVG
)
if (__DEV__) {
endMeasure(instance, `patch`)
}
// el同步到initialVNode
initialVNode.el = subTree.el
}
// mounted hook
if (m) {
queuePostRenderEffect(m, parentSuspense)
}
// onVnodeMounted
// ...
// 组件已经挂载
instance.isMounted = true

if (__DEV__ || __FEATURE_PROD_DEVTOOLS__) {
devtoolsComponentAdded(instance)
}

// #2458: deference mount-only object parameters to prevent memleaks
initialVNode = container = anchor = null as any
} else {
// 组件更新
let { next, bu, u, parent, vnode } = instance
let originNext = next
let vnodeHook: VNodeHook | null | undefined

if (next) {
next.el = vnode.el
// 更新组件 vnode 节点信息
updateComponentPreRender(instance, next, optimized)
} else {
next = vnode
}

// 执行render函数获得nextTree
const nextTree = renderComponentRoot(instance)

// 获取老的subTree
const prevTree = instance.subTree
instance.subTree = nextTree

// patch新旧节点更新组件
patch(
prevTree,
nextTree,
// parent may have changed if it's in a teleport
hostParentNode(prevTree.el!)!,
// anchor may have changed if it's in a fragment
getNextHostNode(prevTree),
instance,
parentSuspense,
isSVG
)
// ...
}
}

这里的patch操作其实就是调用本章开头的那个patch,可以看到patch其实是一个递归操作,这里patch subtree如果根组件的根元素是组件则会继续执行processComponent,如果是一个element元素则会执行processElement,processElement中会处理children,又会调用patch,如此递归直到整个组件挂载完成。

组件无论是首次挂载,还是更新,做的事情核心是一样的,先调用renderComponentRoot方法生成组件模板的虚拟DOM,然后调用patch方法打补丁。

renderComponentRoot
1
2
3
4
5
6
7
function renderComponentRoot(instance) {
const { type: Component, vnode, proxy, withProxy, props, propsOptions: [propsOptions], slots, attrs, emit, render, renderCache, data, setupState, ctx, inheritAttrs } = instance;
// 执行render函数
// render函数内部会通过_createVNode或者_createElementVNode等函数进一步生成子vnode
let result = normalizeVNode(render.call(proxyToUse, proxyToUse, renderCache, props, setupState, data, ctx))
return result
}

renderComponentRoot核心就是调用组件的渲染函数render方法生成组件模板的虚拟DOM,然后扔给patch方法更新就好了。

updateComponent

看完了mountComponent方法,再来看看updateComponent方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
const updateComponent = (n1, n2, optimized) => {
const instance = (n2.component = n1.component);
// 根据新旧子组件 vnode 判断是否需要更新子组件
if (shouldUpdateComponent(n1, n2, optimized)) {
// 新的子组件 vnode 赋值给 instance.next
instance.next = n2;
// 子组件也可能因为数据变化被添加到更新队列里了,移除它们防止对一个子组件重复更新
invalidateJob(instance.update)
// 执行子组件的副作用渲染函数
instance.update();
}else {
// 不需要更新
n2.el = n1.el;
instance.vnode = n2;
}
}

先调用shouldUpdateComponent方法判断组件是否需要更新,主要是通过检测和对比组件 vnode 中的 props、chidren、dirs、transiton 等属性,来决定子组件是否需要更新。

如果需要更新,那么会执行instance.update方法,这个方法就是前面setupRenderEffect方法里保存的effect.run方法,所以最终执行的也是componentUpdateFn方法。

updateComponentPreRender
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
const updateComponentPreRender = (
instance: ComponentInternalInstance,
nextVNode: VNode,
optimized: boolean,
) => {
// 新组件 vnode 的 component 属性指向组件实例
nextVNode.component = instance
// 旧组件 vnode 的 props 属性
const prevProps = instance.vnode.props
// 组件实例的 vnode 属性指向新的组件 vnode
instance.vnode = nextVNode
// 清空 next 属性,为了下一次重新渲染准备
instance.next = null
// 更新 props
updateProps(instance, nextVNode.props, prevProps, optimized)
// 更新 插槽
updateSlots(instance, nextVNode.children, optimized)

pauseTracking()
// props update may have triggered pre-flush watchers.
// flush them before the render update.
flushPreFlushCbs(instance)
resetTracking()
}

processElement

1
2
3
4
5
6
7
8
9
10
11
const processElement = (n1, n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized) => {
isSVG = isSVG || n2.type === 'svg'
if (n1 == null) {
//挂载元素节点
mountElement(n2, container, anchor, parentComponent, parentSuspense, isSVG, optimized)
}
else {
//更新元素节点
patchElement(n1, n2, parentComponent, parentSuspense, isSVG, optimized)
}
}

该函数的逻辑很简单,如果 n1 为 null,走挂载元素节点的逻辑,否则走更新元素节点逻辑。

我们接着来看挂载元素的 mountElement 函数的实现:

mountElement
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
const mountElement = (vnode, container, anchor, parentComponent, parentSuspense, isSVG, optimized) => {
let el
const { type, props, shapeFlag } = vnode
// 创建 DOM 元素节点
el = vnode.el = hostCreateElement(vnode.type, isSVG, props && props.is)
if (props) {
// 处理 props,比如 class、style、event 等属性
for (const key in props) {
if (!isReservedProp(key)) {
hostPatchProp(el, key, null, props[key], isSVG)
}
}
}
if (shapeFlag & 8 /* TEXT_CHILDREN */) {
// 处理子节点是纯文本的情况
hostSetElementText(el, vnode.children)
}
else if (shapeFlag & 16 /* ARRAY_CHILDREN */) {
// 处理子节点是数组的情况
mountChildren(vnode.children, el, null, parentComponent, parentSuspense, isSVG && type !== 'foreignObject', optimized || !!vnode.dynamicChildren)
}
// 把创建的 DOM 元素节点挂载到 container 上
hostInsert(el, container, anchor)
}

可以看到,挂载元素函数主要做四件事:创建 DOM 元素节点、处理 props、处理 children、挂载 DOM 元素到 container 上。

首先是创建 DOM 元素节点,通过 hostCreateElement 方法创建,这是一个平台相关的方法,我们来看一下它在 Web 环境下的定义:

1
2
3
4
function createElement(tag, isSVG, is) {
isSVG ? document.createElementNS(svgNS, tag)
: document.createElement(tag, is ? { is } : undefined)
}

它调用了底层的 DOM API document.createElement 创建元素,如果是其他平台比如 Weex,hostCreateElement 方法就不再是操作 DOM ,而是平台相关的 API 了,这些平台相关的方法是在创建渲染器阶段作为参数传入的。

创建完 DOM 节点后,接下来要做的是判断如果有 props 的话,给这个 DOM 节点添加相关的 class、style、event 等属性,并做相关的处理,这些逻辑都是在 hostPatchProp 函数内部做的,这里就不展开讲了。

接下来是对子节点的处理,我们知道 DOM 是一棵树,vnode 同样也是一棵树,并且它和 DOM 结构是一一映射的。

如果子节点是纯文本,则执行 hostSetElementText 方法,它在 Web 环境下通过设置 DOM 元素的 textContent 属性设置文本:

1
2
3
function setElementText(el, text) {
el.textContent = text
}

如果子节点是数组,则执行 mountChildren 方法:

mountChildren
1
2
3
4
5
6
7
8
9
10
const mountChildren = (children, container, anchor, parentComponent, parentSuspense, isSVG, optimized, start = 0) => {
for (let i = start; i < children.length; i++) {
// 预处理 child
const child = (children[i] = optimized
? cloneIfMounted(children[i])
: normalizeVNode(children[i]))
// 递归 patch 挂载 child
patch(null, child, container, anchor, parentComponent, parentSuspense, isSVG, optimized)
}
}

子节点的挂载逻辑同样很简单,遍历 children 获取到每一个 child,然后递归执行 patch 方法挂载每一个 child 。

可以看到,mountChildren 函数的第二个参数是 container,而我们调用 mountChildren 方法传入的第二个参数是在 mountElement 时创建的 DOM 节点,这就很好地建立了父子关系。

另外,通过递归 patch 这种深度优先遍历树的方式,我们就可以构造完整的 DOM 树,完成组件的渲染。

处理完所有子节点后,最后通过 hostInsert 方法把创建的 DOM 元素节点挂载到 container 上,它在 Web 环境下这样定义:

1
2
3
4
5
6
7
8
function insert(child, parent, anchor) {
if (anchor) {
parent.insertBefore(child, anchor)
}
else {
parent.appendChild(child)
}
}

这里会做一个 if 判断,如果有参考元素 anchor,就执行 parent.insertBefore ,否则执行 parent.appendChild 来把 child 添加到 parent 下,完成节点的挂载。

因为 insert 的执行是在处理子节点后,所以挂载的顺序是先子节点,后父节点,最终挂载到最外层的容器上。

总结

image-1

到这里,从我们创建实例到页面渲染,再到更新的全流程就讲完了,总结一下,大致就是:

  1. 每个Vue组件都需要产出一份虚拟DOM,也就是组件的render函数的返回值,render函数你可以直接手写,也可以通过template传递模板字符串,由Vue内部来编译成渲染函数,平常我们开发时写的Vue单文件,最终也会编译成普通的Vue组件选项对象;

  2. render函数会作为副作用函数执行,也就是如果在模板中使用到了响应式数据(所谓响应式数据就是能拦截到它的各种读取、修改操作),那么响应式数据和属性会与render函数关联起来,那么当响应式数据被修改以后,就能找到依赖它的render函数,那么就可以通知依赖的组件进行更新;

event loop 中的 Update the rendering(更新渲染)

这是 event loop 中很重要部分,在这篇文章处理流程(processing model)中第 3 步会进行 Update the rendering(更新渲染),规范允许浏览器自己选择是否更新视图。也就是说可能不是每轮事件循环都去更新视图,只在有必要的时候才更新视图。

渲染的基本流程:

alt text

  1. 处理 HTML 标记并构建 DOM 树。
  2. 处理 CSS 标记并构建 CSSOM 树, 将 DOM 与 CSSOM 合并成一个渲染树。
  3. 根据渲染树来布局,以计算每个节点的几何信息。
  4. 将各个节点绘制到屏幕上。

Note: 可以看到渲染树的一个重要组成部分是 CSSOM 树,绘制会等待 css 样式全部加载完成才进行,所以 css 样式加载的快慢是首屏呈现快慢的关键点。

下面讨论一下渲染的时机。规范定义在一次循环中,Update the rendering 会在 Microtasks: Perform a microtask checkpoint 后运行。

渲染时机

以下的例子中,用 chrome 的 Developer tools 的 Timeline 查看各部分运行的时间点。当我们点击这个 div 的时候,截取了部分时间线。

黄色部分是脚本运行,紫色部分是更新 render 树、计算布局,绿色部分是绘制。

绿色和紫色部分可以认为是 Update the rendering。

例子 1

1
2
3
4
5
6
7
8
9
10
<div id="con">this is con</div>
<script>
var t = 0
var con = document.getElementById('con')
con.onclick = function () {
setTimeout(function setTimeout1() {
con.textContent = t
}, 0)
}
</script>

alt text

在这一轮事件循环中,setTimeout1 是作为 task 运行的,可以看到 paint 确实是在 task 运行完后才进行的。

例子 2

现在换成一个 microtask 任务,看看有什么变化

1
2
3
4
5
6
7
8
9
<div id="con">this is con</div>
<script>
var con = document.getElementById('con')
con.onclick = function () {
Promise.resolve().then(function Promise1() {
con.textContext = 0
})
}
</script>

alt text

和上一个例子很像,不同的是这一轮事件循环的 task 是 click 的回调函数,Promise1 则是 microtask,paint 同样是在他们之后完成。

标准就是那么定义的,答案似乎显而易见,我们把例子变得稍微复杂一些。

例子 3

1
2
3
4
5
6
7
8
9
10
11
12
<div id="con">this is con</div>
<script>
var con = document.getElementById('con')
con.onclick = function click1() {
setTimeout(function setTimeout1() {
con.textContent = 0
}, 0)
setTimeout(function setTimeout2() {
con.textContent = 1
}, 0)
}
</script>

alt text
alt text

经过多次测试,执行和渲染顺序可能会出现上图的两种情况,根据 timeline 可以看出,图 1 中 setTimeout 分别执行并且浏览器绘制了两次,图二中 setTimeout1 和 setTimeout2 中间并没有绘制, 而是最后绘制了一次,也基本符合规范,但是需要验证这两次 setTimeout 是否在两次 task 中。

例子 4

在两个 setTimeout 中增加 microtask。再次确认 setTimeout 在两个 task 中

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<div id="con">this is con</div>
<script>
var con = document.getElementById('con')
con.onclick = function () {
setTimeout(function setTimeout1() {
con.textContent = 0
Promise.resolve().then(function Promise1() {
console.log('Promise1')
})
}, 0)
setTimeout(function setTimeout2() {
con.textContent = 1
Promise.resolve().then(function Promise2() {
console.log('Promise2')
})
}, 0)
}
</script>

alt text
alt text

从 run microtasks 中可以看出来,setTimeout1、setTimeout2 是运行在两次 event loop 中

例子 5

将时间间隔加大一些。

1
2
3
4
5
6
7
8
9
10
11
12
<div id="con">this is con</div>
<script>
var con = document.getElementById('con')
con.onclick = function click1() {
setTimeout(function setTimeout1() {
con.textContent = 0
}, 0)
setTimeout(function setTimeout2() {
con.textContent = 1
}, 17)
}
</script>

当间隔增大,大部分时候可以肉眼从看到先变成 0,再变成 1 的过程。但是有时也是会合并起来,只 paint 一次

alt text

通过 timeline,可以看到 setTimeout1 后接着 paint,后执行了 setTimeout2 后也有 paint

例子 6

我们在同一时间执行多个 setTimeout 来模拟执行间隔很短的 task。

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
<div id="con">this is con</div>
<script>
var con = document.getElementById('con')
con.onclick = function () {
setTimeout(function () {
con.textContent = 0
}, 0)
setTimeout(function () {
con.textContent = 1
}, 0)
setTimeout(function () {
con.textContent = 2
}, 0)
setTimeout(function () {
con.textContent = 3
}, 0)
setTimeout(function () {
con.textContent = 4
}, 0)
setTimeout(function () {
con.textContent = 5
}, 0)
setTimeout(function () {
con.textContent = 6
}, 0)
}
</script>

alt text
图一中总共 paint 了一次

alt text

图二中一共 paint 了两次,所以多次 task 的间隔很短,仍会进行绘制。

例子 7

有说法是一轮 event loop 执行的 microtask 有数量限制,多余的 microtask 会放到下一轮执行。下面例子将 microtask 的数量增加到 25000。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<div id="con">this is con</div>
<script>
var con = document.getElementById('con')
con.onclick = function () {
setTimeout(function setTimeout1() {
con.textContent = 'task1'
for (var i = 0; i < 250000; i++) {
Promise.resolve().then(function () {
con.textContent = i
})
}
}, 0)
setTimeout(function setTimeout2() {
con.textContent = 'task2'
}, 0)
}
</script>

alt text

可以看到脚本的运行耗费大量的时间,并且阻塞了渲染。

我们看 setTimeout2 的运行情况

alt text

可以看到 setTimeout2 这轮 event loop 没有 run microtasks,microtasks 在 setTimeout1 被全部执行完了。

25000 个 microtasks 不能说明 event loop 对 microtasks 数量没有限制,有可能这个限制数很高,远超 25000,但日常使用基本不会使用那么多了。

对 microtasks 增加数量限制,一个很大的作用是防止脚本运行时间过长,阻塞渲染。

例子 8

使用 requestAnimationFrame。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<div id="con">this is con</div>
<script>
var con = document.getElementById('con')
var i = 0
var raf = function () {
requestAnimationFrame(function () {
con.textContent = i
Promise.resolve().then(function () {
i++
if (i < 3) raf()
})
})
}
con.onclick = function () {
raf()
}
</script>

看下总体的 timeline
alt text

单看某一个 requestAnimationFrame
alt text

  • 可以看出 requestAnimationFrame 是在更新渲染阶段的执行的,其执行顺序早于 paint,并且不会被合并执行,非常适合做动画
  • 在 requestAnimationFrame 回调内有新的 microtasks 进入时,也会执行完所有的 microtasks 才会进入到渲染阶段

例子 9

验证 postMessage 是否是 task

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<script>
setTimeout(function setTimeout1() {
console.log('setTimeout1')
}, 0)
var channel = new MessageChannel()
channel.port1.onmessage = function onmessage1() {
console.log('postMessage')
Promise.resolve().then(function promise1() {
console.log('promise1')
})
}
channel.port2.postMessage(0)
setTimeout(function setTimeout2() {
console.log('setTimeout2')
}, 0)
console.log('sync')
</script>

执行顺序:

1
2
3
4
5
sync
setTimeout1
postMessage
promise1
setTimeout2

alt text

第一个黄块是 setTimeout1,第二个是 onmessage1,第三个是 promise1,第四个是 setTimeout2。显而易见,postMessage 属于 task。

总结

结合规范和以上验证案例可以得出一些结论:

  • event loop 的大致循环过程,可以用下边的图表示:

alt text

  • 在一轮 event loop 中多次修改同一 dom,只有最后一次会进行绘制。
  • 例 3-例 6 这几个结果是非常不可控的,如果这两个 Task 之间正好遇到了浏览器认定的渲染机会,那么它会重绘,否则就不会。
  • 渲染更新(Update the rendering)会在 event loop 中的 tasks 和 microtasks 完成后进行,但并不是每轮 event loop 都会更新渲染,这里有一个 rendering opportunity 的概念,判断是否需要渲染,这取决于是否修改了 dom 和浏览器觉得是否有必要在此时立即将新状态呈现给用户,也要要根据屏幕刷新率、页面性能、页面是否在后台运行来共同决定。通常来说这个渲染间隔是固定的。(所以多个 task 很可能在一次渲染之间执行)
  • 事件循环不一定每轮都伴随着重渲染,但是如果有微任务,一定会伴随着微任务执行。
  • 如果希望在每轮 event loop 都即时呈现变动,可以使用 requestAnimationFrame。

前言

在之前的分享中,我们团队已经成功运用了 Gitlab CI,并且已经结构化提交 git commit 记录,我们希望更进一步,自动管理发布版本,自动生成更新日志,自动发布 NPM 包,因此我们引入了 semantic-release 进一步自动化管理我们的发布流程。

semantic-release 概述

有关semantic-release的详细介绍可以阅读官方文档,这里只做一些概述性的总结。和 standard-version 相比,semantic-release 更适合在 CI 环境中运行,它自带支持各种 git server 的认证支持,如 Github,Gitlab,Bitbucket 等等,此外,还支持插件,以便完成其他后续的流程步骤,比如自动生成 git tag 和 release note 之后再 push 回中央仓库,自动发布 npm 包等等。

semantic-release 会根据规范化的 commit 信息生成发布日志,默认使用 angular 规则,其他规则可以配置插件完成。

semantic-release 大致的工作流如下:

  • 提交到特定的分支触发 release 流程
  • 验证 commit 信息,生成 release note,打 git tag
  • 其他后续流程,如生成 CHANGELOG.md,npm publish 等等(通过插件完成)

由 CI 自动执行之后的效果就像这样,在 Git tag 页面可以看到 tag 信息,同时包含更新记录:
86f880f284d3072ec2179cbc46c32396

如果启用了@semantic-release/git 插件,还会将生成的 CHANGELOG.md 反向 push 回中央仓库:
6cfc655f89c58bc6a2faa5e3aee6eea1

commit history 的实际效果如下
c15015e5517056c98eca17829d479a12

实践

在项目工程中添加 release.config.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
parserOpts = {
mergePattern: /^Merge pull request #(\d+) from (.*)$/,
mergeCorrespondence: ['id', 'source'],
}
// Copied from https://github.com/conventional-changelog/conventional-changelog/blob/master/packages/conventional-changelog-angular/writer-opts.js#L27
// and modified to support adding all commit types to the release notes
customTransform = (commit, context) => {
const issues = []
commit.notes.forEach((note) => {
note.title = `BREAKING CHANGES`
})
if (commit.type === `feat`) {
commit.type = `✨ Features`
} else if (commit.type === `fix`) {
commit.type = `🐞 Bug Fixes`
} else if (commit.type === `perf`) {
commit.type = `🎈 Performance Improvements`
} else if (commit.type === `revert`) {
commit.type = `Reverts`
} else if (commit.type === `docs`) {
commit.type = `📃 Documentation`
} else if (commit.type === `style`) {
commit.type = `🌈 Styles`
} else if (commit.type === `refactor`) {
commit.type = `🦄 Code Refactoring`
} else if (commit.type === `test`) {
commit.type = `🧪 Tests`
} else if (commit.type === `build`) {
commit.type = `🔧 Build System`
} else if (commit.type === `ci`) {
commit.type = `🐎 Continuous Integration`
} else {
return
}
if (commit.scope === `*`) {
commit.scope = ``
}
if (typeof commit.hash === `string`) {
commit.shortHash = commit.hash.substring(0, 7)
}
if (typeof commit.subject === `string`) {
commit.subject = commit.subject.substring(2)
let url = context.repository
? `${context.host}/${context.owner}/${context.repository}`
: context.repoUrl
if (url) {
url = `${url}/issues/` // Issue URLs.
commit.subject = commit.subject.replace(/#([0-9]+)/g, (_, issue) => {
issues.push(issue)
return `[#${issue}](${url}${issue})`
})
}
if (context.host) {
// User URLs.
commit.subject = commit.subject.replace(
/\B@([a-z0-9](?:-?[a-z0-9/]){0,38})/g,
(_, username) => {
if (username.includes('/')) {
return `@${username}`
}
return `[@${username}](${context.host}/${username})`
}
)
}
commit.subject = `${commit.subject} (by @${commit.committer.name})`
} // remove references that already appear in the subject
commit.references = commit.references.filter((reference) => {
if (issues.indexOf(reference.issue) === -1) {
return true
}
return false
})
return commit
}
module.exports = {
branches: 'master',
parserOpts,
writerOpts: { transform: customTransform },
plugins: [
[
'@semantic-release/commit-analyzer',
{
preset: 'angular',
releaseRules: [
{ type: 'docs', scope: 'README', release: 'patch' },
{ type: 'refactor', release: 'patch' },
{ type: 'style', release: 'patch' },
{ type: 'test', release: 'patch' },
{ type: 'build', release: 'patch' },
{ type: 'ci', release: 'patch' },
],
},
],
'@semantic-release/release-notes-generator',
['@semantic-release/changelog', { changelogFile: 'CHANGELOG.md' }],
'@semantic-release/npm',
[
'@semantic-release/git',
{
assets: ['package.json', 'CHANGELOG.md'],
},
],
[
'@semantic-release/gitlab',
{
gitlabUrl: 'http://git.example.com',
assets: [],
},
],
],
}

完成 .gitlab-ci.yml 配置如下(仅部分关键的配置片段):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
stages:
    - lint
    - release
commitlint:
    stage: lint
    rules:
        - if: '$CI_COMMIT_TITLE =~ /^chore\(release\)/'
          when: never
        - if: '$CI_COMMIT_TITLE =~ /^Merge branch/'
          when: never
        - if: $CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH # 一般当分支master有 push 或 merge 时才会执行该工作
    script:
        - sh bin/commitlint-gitlab-ci.sh
release:
    stage: release
    rules:
        - if: $CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH
    script:
        - semantic-release

commitlint 脚本请参考commitlint-gitlab-ci.sh

小结

至此,我们完成了通过 CI 自动管理版本号和发布日志的需求,大大节省了人力,同时,还留下了发布痕迹,方便追溯历史版本。
另外,需要注意的是上述的配置并不会修改源码部分的版本号配置内容(如 build.gradle 或 package.json 等),如果需要自动管理这些地方的版本,与 git tag 版本保持一致,可以引入@semantic-release/exec 插件,自己写脚本,通过脚本自动化修改这些地方的版本号。
还需要注意的是 semantic-release 默认产生的 commit 记录为了避免不必要的 CI 流程,会在 commit 记录加上[skip ci](见上面的截图)来跳过 CI,如果你的流水线需要由 git tag 触发,可以配置@semantic-release/git 插件,自定义 commit 记录,去掉[skip ci]。

异步的思考

提起异步,相信每个人都知道。异步背后的“靠山”就是 event loops。这里的异步准确的说应该叫浏览器的 event loops 或者说是 javaScript 运行环境的 event loops,因为ECMAScript中没有 event loops,event loops 是在HTML Standard定义的。

(1)单线程的 JavaScript

我们知道,JavaScript 是一种单线程语言,它主要用来与用户互动,以及操作 DOM。
JavaScript 有同步和异步的概念,这就解决了代码阻塞的问题:

- 同步:如果在一个函数返回的时候,调用者就能够得到预期结果,那么这个函数就是同步的;
- 异步:如果在函数返回的时候,调用者还不能够得到预期结果,而是需要在将来通过一定的手段得到,那么这个函数就是异步的。
那单线程有什么好处呢?
在 JS 运行的时候可能会阻止 UI 渲染,这说明了两个线程是互斥的。这是因为 JS 可以修改 DOM,如果在 JS 执行的时候 UI 线程还在工作,就可能导致不能安全的渲染 UI。
得益于 JS 是单线程运行的,可以达到节省内存,节约上下文切换时间的好处。

(2)多线程的浏览器

JS 是单线程的,在同一个时间只能做一件事情,那为什么浏览器可以同时执行异步任务呢?

这是因为浏览器是多线程的,当 JS 需要执行异步任务时,浏览器会另外启动一个线程去执行该任务。也就是说,JavaScript 是单线程的指的是执行 JavaScript 代码的线程只有一个,是浏览器提供的 JavaScript 引擎线程(主线程)。除此之外,浏览器中还有定时器线程、 HTTP 请求线程等线程,这些线程主要不是来执行 JS 代码的。

比如主线程中需要发送数据请求,就会把这个任务交给异步 HTTP 请求线程去执行,等请求数据返回之后,再将 callback 里需要执行的 JS 回调交给 JS 引擎线程去执行。也就是说,浏览器才是真正执行发送请求这个任务的角色,而 JS 只是负责执行最后的回调处理。所以这里的异步不是 JS 自身实现的,而是浏览器为其提供的能力。

alt text

可以看到,Chrome 不仅拥有多个进程,还有多个线程。以渲染进程为例,就包含 GUI 渲染线程、JS 引擎线程、事件触发线程、定时器触发线程、异步 HTTP 请求线程。这些线程为 JS 在浏览器中完成异步任务提供了基础。

定义

我们来看看 event loop 在 HTML Standard 中的定义章节:

第一句话:

为了协调事件,用户交互,脚本,渲染,网络等,用户代理必须使用本节所述的 event loop。

事件用户交互脚本渲染网络这些都是我们所熟悉的东西,他们都是由 event loop 协调的。触发一个 click 事件,进行一次 ajax 请求,背后都有 event loop 在运作。

task queues

一个 event loop 有一个或者多个 task 队列。

当用户代理安排一个任务,必须将该任务增加到相应的 event loop 的一个 tsak 队列中。

每一个 task 都来源于指定的任务源,比如可以为鼠标、键盘事件提供一个 task 队列,其他事件又是一个单独的队列。可以为鼠标、键盘事件分配更多的时间,保证交互的流畅。

哪些是 task 任务源呢?

规范在Generic task sources中有提及:

DOM 操作任务源:
此任务源被用来相应 dom 操作,例如一个元素以非阻塞的方式插入文档。

用户交互任务源:
此任务源用于对用户交互作出反应,例如键盘或鼠标输入。响应用户操作的事件(例如 click)必须使用 task 队列。

网络任务源:
网络任务源被用来响应网络活动。

history traversal 任务源:
当调用 history.back()等类似的 api 时,将任务插进 task 队列。

task 任务源非常宽泛,比如 ajax 的 onload,click 事件,基本上我们经常绑定的各种事件都是 task 任务源,还有数据库操作(IndexedDB ),需要注意的是 setTimeout、setInterval、setImmediate 也是 task 任务源。总结来说 task 任务源:

  • setTimeout
  • setInterval
  • setImmediate
  • I/O
  • UI rendering

microtask

微任务队列不是任务队列。

每个事件循环都有一个微任务队列,这是一个微任务队列,最初是空的。微任务是一种口语化的方式,指的是通过微任务算法队列创建的任务。

如果在初期执行时,spin the event loop,microtasks 有可能被移动到常规的 task 队列,在这种情况下,microtasks 任务源会被 task 任务源所用。通常情况,task 任务源和 microtasks 是不相关的。

microtask 队列和 task 队列有些相似,都是先进先出的队列,由指定的任务源去提供任务,不同的是一个 event loop 里只有一个 microtask 队列。

HTML Standard 没有具体指明哪些是 microtask 任务源,通常认为是 microtask 任务源有:

  • process.nextTick
  • promises
  • Object.observe
  • MutationObserver

处理流程(processing model)

  1. 在 tasks 队列中选择最老的一个 task,用户代理可以选择任何 task 队列,如果没有可选的任务,则跳到下边的 microtasks 步骤。(从任务队列中取出一个宏任务并执行)

    1.1 将上边选择的 task 设置为正在运行的 task

    1.2 Run: 运行被选择的 task。

    1.3 将 event loop 的 currently running task 变为 null。

    1.4 从 task 队列里移除前边运行的 task。

  2. Microtasks: 执行 microtasks 任务检查点。(检查微任务队列,执行并清空微任务队列,如果在微任务的执行中又加入了新的微任务,也会在这一步一起执行。)

  3. 进入更新渲染阶段,判断是否需要渲染,这里有一个 rendering opportunity 的概念,也就是说不一定每一轮 event loop 都会对应一次浏览 器渲染,要根据屏幕刷新率、页面性能、页面是否在后台运行来共同决定,通常来说这个渲染间隔是固定的。(所以多个 task 很可能在一次渲染之间执行)

    • 浏览器会尽可能的保持帧率稳定,例如页面性能无法维持 60fps(每 16.66ms 渲染一次)的话,那么浏览器就会选择 30fps 的更新速率,而不是偶尔丢帧。
    • 如果浏览器上下文不可见,那么页面会降低到 4fps 左右甚至更低。
    • 如果满足以下条件,也会跳过渲染:
      • 浏览器判断更新渲染不会带来视觉上的改变。
      • map of animation frame callbacks 为空,也就是帧动画回调为空,可以通过 requestAnimationFrame 来请求帧动画。
  4. 如果上述的判断决定本轮不需要渲染,那么下面的几步也不会继续运行:

This step enables the user agent to prevent the steps below from running for other reasons, for example, to ensure certain tasks are executed immediately after each other, with only microtask checkpoints interleaved (and without, e.g., animation frame callbacks interleaved). Concretely, a user agent might wish to coalesce timer callbacks together, with no intermediate rendering updates.
有时候浏览器希望两次「定时器任务」是合并的,他们之间只会穿插着 microTask 的执行,而不会穿插屏幕渲染相关的流程

  1. 对于需要渲染的文档,如果窗口的大小发生了变化,执行监听的 resize 方法。

  2. 对于需要渲染的文档,如果页面发生了滚动,执行 scroll 方法。

  3. 对于需要渲染的文档,执行帧动画回调,也就是 requestAnimationFrame 的回调。(后文会详解)

  4. 对于需要渲染的文档, 执行 IntersectionObserver 的回调。

  5. 对于需要渲染的文档,重新渲染绘制用户界面。

  6. 判断 task 队列和 microTask 队列是否都为空,如果是的话,则进行 Idle 空闲周期的算法,判断是否要执行 requestIdleCallback 的回调函数。(后文会详解)

对于 resize 和 scroll 来说,并不是到了这一步才去执行滚动和缩放,那岂不是要延迟很多?浏览器当然会立刻帮你滚动视图,根据 CSSOM 规范所讲,浏览器会保存一个 pending scroll event targets,等到事件循环中的 scroll 这一步,去派发一个事件到对应的目标上,驱动它去执行监听的回调函数而已。resize 也是同理。

microtasks 检查点(microtask checkpoint)

上文 event loop 处理流程第 2 步,执行了一个 microtask checkpoint,看看规范如何描述 microtask checkpoint:

当用户代理去执行一个 microtask checkpoint,如果 microtask checkpoint 的 flag(标识)为 false,用户代理必须运行下面的步骤:

  1. 将 microtask checkpoint 的 flag 设为 true。
  2. Microtask queue handling: 如果 event loop 的 microtask 队列为空,直接跳到第八步(Done)。
  3. 在 microtask 队列中选择最老的一个任务。
  4. 将上一步选择的任务设为 event loop 的 currently running task。
  5. 运行选择的任务。
  6. 将 event loop 的 currently running task 变为 null。
  7. 将前面运行的 microtask 从 microtask 队列中删除,然后返回到第二步(Microtask queue handling)。
  8. Done: 每一个 environment settings object 它们的 responsible event loop 就是当前的 event loop,会给 environment settings object 发一个 rejected promises 的通知。
  9. 清理 IndexedDB 的事务。
  10. 将 microtask checkpoint 的 flag 设为 flase。

microtask checkpoint 所做的就是执行 microtask 队列里的任务。什么时候会调用 microtask checkpoint 呢?

  • 当上下文执行栈为空时,执行一个 microtask checkpoint。
  • 在 event loop 的第六步(Microtasks: Perform a microtask checkpoint)执行 checkpoint,也就是在运行 task 之后,更新渲染之前。

多任务队列

上文也提到了,每一个 task 都来源于指定的任务源, 所以 task 队列并不是我们想象中的那样只有一个,根据规范里的描述:

An event loop has one or more task queues. For example, a user agent could have one task queue for mouse and key events (to which the user interaction task source is associated), and another to which all other task sources are associated. Then, using the freedom granted in the initial step of the event loop processing model, it could give keyboard and mouse events preference over other tasks three-quarters of the time, keeping the interface responsive but not starving other task queues. Note that in this setup, the processing model still enforces that the user agent would never process events from any one task source out of order.

事件循环中可能会有一个或多个任务队列,这些队列分别为了处理:

  1. 鼠标和键盘事件
  2. 其他的一些 Task

浏览器会在保持任务顺序的前提下,可能分配四分之三的优先权给鼠标和键盘事件,保证用户的输入得到最高优先级的响应,而剩下的优先级交给其他 Task,并且保证不会“饿死”它们。

这个规范也导致 vue 的这个Issue。简单描述一下就是采用了 task 实现的 nextTick,在用户持续滚动的情况下 nextTick 任务被延后了很久才去执行,导致动画跟不上滚动了。

迫于无奈,尤大还是改回了 microTask 去实现 nextTick,目前来说 promise.then 微任务已经很稳定并且 vue3 就是这样实现的,并且 Chrome 也已经实现了 queueMicroTask 这个官方 API。我们想要调用微任务队列的话,也可以节省掉实例化 Promise 在开销了。

从这个 Issue 的例子中我们可以看出,稍微去深入了解一下规范还是比较有好处的,以免在遇到这种比较复杂的 Bug 的时候一脸懵逼。

执行栈(JavaScript execution context stack)

task 和 microtask 都是推入栈中执行的,要完整了解 event loops 还需要认识 JavaScript execution context stack,它的规范位于https://tc39.github.io/ecma262/#execution-context-stack。

javaScript 是单线程,也就是说只有一个主线程,主线程有一个栈,每一个函数执行的时候,都会生成新的 execution context(执行上下文),执行上下文会包含一些当前函数的参数、局部变量之类的信息,它会被推入栈中, running execution context(正在执行的上下文)始终处于栈的顶部。当函数执行完后,它的执行上下文会从栈弹出。

alt text

举个简单的例子:

1
2
3
4
5
6
7
8
9
10
function bar() {
console.log('bar')
}

function foo() {
console.log('foo')
bar()
}

foo()

执行过程中栈的变化:
alt text

完整异步过程

主线程类似一个加工厂,它只有一条流水线,待执行的任务就是流水线上的原料,只有前一个加工完,后一个才能进行。event loops 就是把原料放上流水线的工人。只要已经放在流水线上的,它们会被依次处理,称为同步任务。一些待处理的原料,工人会按照它们的种类排序,在适当的时机放上流水线,这些称为异步任务。

alt text

举个简单的例子,假设一个 script 标签的代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
Promise.resolve().then(function promise1() {
console.log('promise1')
})
setTimeout(function setTimeout1() {
console.log('setTimeout1')
Promise.resolve().then(function promise2() {
console.log('promise2')
})
}, 0)

setTimeout(function setTimeout2() {
console.log('setTimeout2')
}, 0)

运行过程:

script 里的代码被列为一个 task,放入 task 队列。

  • 循环 1

  • 【task 队列:script ;microtask 队列:】

    • 1.从 task 队列中取出 script 任务,推入栈中执行。
    • 2.promise1 列为 microtask,setTimeout1 列为 task,setTimeout2 列为 task。
  • task 队列:setTimeout1 setTimeout2;microtask 队列:promise1】

    • 3.script 任务执行完毕,执行 microtask checkpoint,取出 microtask 队列的 promise1 执行。
  • 循环 2

  • 【task 队列:setTimeout1 setTimeout2;microtask 队列:】

    • 4.从 task 队列中取出 setTimeout1,推入栈中执行,将 promise2 列为 microtask。
  • 【task 队列:setTimeout2;microtask 队列:promise2】

    • 5.执行 microtask checkpoint,取出 microtask 队列的 promise2 执行。
  • 循环 3

  • 【task 队列:setTimeout2;microtask 队列:】

    • 6.从 task 队列中取出 setTimeout2,推入栈中执行。
    • 7.setTimeout2 任务执行完毕,执行 microtask checkpoint。
  • 【task 队列:;microtask 队列:】

结语

以上就是对 event loop 规范 和一些运行的流程做的总结。但是包括 requestAnimationFrame,requestIdleCallback 和 浏览器中 js 的执行机制等细节的总结将会单独写文章进行归纳总结。

介绍

GitLab CI/CD 是一个内置在 GitLab 中的工具,用于通过持续方法进行软件开发:

Continuous Integration(持续集成)

假设一个应用程序,其代码存储在 GitLab 的 Git 仓库中。开发人员每天都要多次推送代码更改。对于每次向仓库的推送,你都可以创建一组脚本来自动构建和测试你的应用程序,从而减少了向应用程序引入错误的机会。这种做法称为持续集成,对于提交给应用程序(甚至是开发分支)的每项更改,它都会自动连续进行构建和测试,以确保所引入的更改通过你为应用程序建立的所有测试,准则和代码合规性标准。

Continuous Delivery(持续交付)

持续交付是超越持续集成的更进一步的操作。应用程序不仅会在推送到代码库的每次代码更改时进行构建和测试,而且,尽管部署是手动触发的,但作为一个附加步骤,它也可以连续部署。此方法可确保自动检查代码,但需要人工干预才能从策略上手动触发以必输此次变更。

Continuous Deployment(持续部署)

与持续交付类似,但不同之处在于,你无需将其手动部署,而是将其设置为自动部署。完全不需要人工干预即可部署你的应用程序。

工作流程

为了使用 GitLab CI/CD,你需要一个托管在 GitLab 上的应用程序代码库,并且在根目录中的.gitlab-ci.yml 文件中指定构建、测试和部署的脚本。

在这个文件中,你可以定义要运行的脚本,定义包含的依赖项,选择要按顺序运行的命令和要并行运行的命令,定义要在何处部署应用程序,以及指定是否 要自动运行脚本或手动触发脚本。

为了可视化处理过程,假设添加到配置文件中的所有脚本与在计算机的终端上运行的命令相同。

一旦你已经添加了.gitlab-ci.yml 到仓库中,GitLab 将检测到该文件,并使用名为 GitLab Runner 的工具运行你的脚本。该工具的操作与终端类似。

这些脚本被分组到 jobs,它们共同组成一个 pipeline。一个最简单的.gitlab-ci.yml 文件可能是这样的:

1
2
3
4
5
6
before_script:
- apt-get install rubygems ruby-dev -y

run-test:
script:
- ruby --version 6

before_script 属性将在运行任何内容之前为你的应用安装依赖,一个名为 run-test 的 job(作业)将打印当前系统的 Ruby 版本。二者共同构成了在每次推送到仓库的任何分支时都会被触发的 pipeline(管道)。

GitLab CI/CD 不仅可以执行你设置的 job,还可以显示执行期间发生的情况,正如你在终端看到的那样:
74b8f022354a48a2a663a996ae2c1dd8

通过 GitLab UI 所有的步骤都是可视化的

aae548d462b9c6b5e8bab016b96c5512

实践

创建一个.gitlab-ci.yml 文件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
stages:
- lint
- release
commitlint:
stage: lint
rules:
- if: '$CI_COMMIT_TITLE =~ /^chore\(release\)/'
when: never
- if: '$CI_COMMIT_TITLE =~ /^Merge branch/'
when: never
- if: $CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH # 一般当分支master有 push 或 merge 时才会执行该工作
script:
- sh bin/commitlint-gitlab-ci.sh
release:
stage: release
rules:
- if: $CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH
script:
- semantic-release

配置一个 Runner

在 GitLab 中,Runner 运行你定义在.gitlab-ci.yml 中的作业(job)
一个 Runner 可以是一个虚拟机、物理机、docker 容器,或者一个容器集群 GitLab 与 Runner 之间通过 API 进行通信,因此只需要 Runner 所在的机器有网络并且可以访问 GitLab 服务器即可
你可以去 Settings ➔ CI/CD 看是否已经有 Runner 关联到你的项目,设置 Runner 简单又直接
7b99f010133b6b0689a1dc90c2893831

查看 pipeline 和 jobs 状态

e57d2652f09e520231cae9358ac69e10

小结

团队计划使用 semantic-release 自动管理发布版本,结合 Gitlab CI/CD 是一个很不错的选择,具体的实践过程和心得会在后面分享。

概况图

71d6a82a01519413ca2a1be4b957c544

无痕模式

无痕模式可以保证 Chrome 在一个相对干净的环境下运行,避免 chrome 上安装的插件影响性能分析结果。
文件—>打开新的无痕式窗口,或使用快捷键 ctrl + shift + N 打开无痕模式下的 chrome 新标签页

性能记录

点击面板里的 ○,可以记录运行时的性能记录,如下图:
0b82ba7820c6d3ec5b21fe563191a08d

再次点击 Record 或者点击 Stop 停止记录。

加载时性能记录录制

若要分析页面记载时性能需要录制加载时性能。

  1. 打开待分析性能的页面。
  2. 打开 Devtools 中 Performance 窗口。
  3. 点击左上角重新加载按钮。DevTools 会自动记录页面加载是各项性能指标,加载完成几秒后自动停止记录。
    Devtools 记录会自动增加
    bcc3faa215dab1b20aefba15f42fa29d

页面加载时记录
8faf902b1637f885df323e7791713c33

清除记录

清除 Performance 窗口中的记录数据。

抓取运行时屏幕快照

点选 Screenshots 选项开启为每一帧记录屏幕快照功能。

查看内存度量值

点选 Memory 选项打开内存度量功能。
DevTools 在 Summary 面板上侧显示一个新的 Memory 图表。在 NET 图表下边也显示一个 HEAP 图表。HEAP 图表提供的信息同 Memory 面板中 JS Heap 提供的信息相同。

开启加速渲染工具

点选 Enable advanced paint instrumentation 选项(会带来大量的性能开销)

控制录制过程中 CPU 工作频率

将 CPU 设置为需要的运算速度模式。
886c6b4d80132d8e286af2d6ed89cdb9

CPU 工作频率的控制结果跟实际使用的机器能力有关。例如,4x slowdown 选项会使你本地 CPU 运算速率比正常情况下降低 4 倍。不同设备由于设计架构不同,Devtools 不能精确模拟移动端设备的 CPU 运算模式。

保存记录

da01643afd3f94a7927b96b2ddd7a044
会生成 JSON 文件

加载记录

单击鼠标右键选择 Load Profile 加载记录

分析性能记录

运行时或者加载时性能录制结束后,在 Performance 窗口中会显示相关数据,从而对于记录过程中的情况进行分析。

选择记录中的一部分

在 Overview 窗口中,可以选中记录的某一部分。 Overview 窗口指的是包含 FPS, CPU 和 NET 图表部分。
3c66d685cee9f55228ca867b5a07a88e

查看主线程活动

利用 Main 区域查看页面主线程加载时的主要活动。
26247567fd6f9281d4a16d8fcbcba32f

点击某一函数在 Summary 窗口中查看更多详细信息。如图中所示 DevTools 选中_getRequestData 事件。
DevTools 采用随机的颜色标识脚本信息,如上面图中浅绿色标识函数调用,深黄色标识脚本活动。
若想隐藏火焰图中的 JavaScript 中调用的详细信息,请查看前面介绍的禁用 JavaScript 样例功能。若禁用 JavaScript 样例功能,你只可以看到初始调用事件。比如,图中标识的 Timer fired 和 Function Call 。

在表格中查看活动

录制结束后,利用 Main 窗口中信息不是分析数据的唯一方式。DevTools 另外提供了三种表格式分析活动方式,每种方式都是从不同的角度出发:
若想分析那些活动占用时间更多时,可以利用 Bottom-Up 窗口。
若想分析导致更多活动的根活动时,可以采用 Call Tree。
若想按顺序分析记录中发生的活动时,可以利用 Event Log 窗口。

根活动

根活动指的是浏览器触发的一系列流程。例如,当你点击页面内容,浏览器触发一个 Event 作为根活动,该 Event 可能回调一个事件处理事件。
在 Main 面板中的火焰图中,根活动展示在上部,在 Call Tree 和 Event Log 面板中,根活动展示在顶层。

Call Tree 标签页

Call Tree 标签页中展示记录中被选中部分的活动信息。
6f4e0500bac02ec6143b950c159c63ae

图中 Activity 列中显示的 Timer fired、 Paint、Recalculate Style 和 Layout 代表根活动。层级嵌套表示代表回调栈。如图中 Function Call 调用 u,再调用 getImageUrl,继续调用 getLinkUrl 等等。
Self Time 表示对应活动消耗的时间,Total Time 表示对应活动以及子活动共同消耗的时间。
点击 Self Time,Total Time 或者 Activity 表头区域,可按对应列排序。
利用 Filter 输入框区域,输入活动名过滤事件。
Grouping 分组菜单默认为 No Grouping,利用该功能可以根据不同的分类将活动进行分组。
点击右侧 Show Heaviest Stack,在右侧展示当前选中活动中占用时间最多的子活动信息。

6562aca6f7e21ca7a5746f1cff491455

Bottom-Up 标签页

利用 Bottom-Up 标签查看占用最多时间的活动。
Self Time 表示对应活动消耗的时间。
Total Time 表示对应活动以及子活动共同消耗的时间。

Event Log 标签页

Event Log 标签页按顺序展示记录中发生的活动。
Start Time 列表示该项活动的开始时间,该时间相对于记录开始时间计算。例如图中选中项开始时间为 1566.2 ms,代表该活动在记录开始之后 1566.2 ms 后开始。
Self Time 表示对应活动消耗的时间。
Total Time 表示对应活动以及子活动共同消耗的时间。
点击 Start Time 、Self Time、Total Time 表头区域,可按对应列排序。
利用 Filter 输入框区域,输入活动名过滤事件。
利用 Duration 下拉菜单过滤>=1ms 或者>=15ms 的活动。该菜单默认选中 All 选项,展示所有活动。
利用 Loading、Experience、Scripting、Rendering、Painting 选项进行分类过滤。
58c29a0d627c26c7d9fc1afd80a348ef

分析每秒传输帧数(FPS)

  • 查看 FPS 图表了解整个记录中 FPS 的概况。
  • Frames 模块查看每一帧时间消耗。
  • 利用 FPS meter 工具(MoreTools—>Rendering)在页面运行时实时查看 FPS 信息。

FPS 图表

FPS 图表显示了整个记录过程中帧率的概况。图表中绿色折线越高代表帧率越好。
FPS 折线图上测出现的红色横线为一条性能警示线,表示帧率低于该值会严重影响用户体验。

Frames 模块

Frames 模块清晰表明每个帧消耗时间。
鼠标在某一帧上悬停可以查看更多详细信息。

查看交互信息

利用 Interactions 模块查看并分析记录过程中用户的交互操作。

查看 GPU 活动

在 GPU 模块查看 GPU 活动信息
69c660e3d248b30ffc310f2813525e08

查看栅格活动

在 Raster 模块查看栅格活动信息

前言

由于公司标准化产品逐渐成熟,项目越来越多,需要一款脚手架通过业务标准化模板自动生成项目,可以让开发人员专注于业务开发,降低心智成本,提升团队效率。

参考了常用的脚手架,create-react-app、vue-cli、egg-init 的实现,搭建出了一套符合团队实际情况的脚手架工具。

介绍

脚手架就是在启动的时候询问一些简单的问题,并且通过用户回答的结果去渲染对应的模板文件。
基本工作流程如下:

  1. 通过命令行交互询问用户问题
  2. 拉取远端标准化模板
  3. 根据用户回答的结果生成文件
  4. 自动下载依赖

热门脚手架工具库

116d0e8220abc84e44593738478cbee7

搭建

1. 判断当前环境是否符合要求

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#!/usr/bin/env node
'use strict'
const currentNodeVersion = process.versions.node
const semver = currentNodeVersion.split('.')
const major = semver[0]
if (major < 14) {
console.error(
'You are running Node ' +
currentNodeVersion +
'.\n' +
'Create React App requires Node 14 or higher. \n' +
'Please update your version of Node.'
)
process.exit(1)
}

2. 创建脚手架启动命令(使用 commander)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
const program = new commander.Command(packageJson.name)
.version(packageJson.version)
.arguments('<project-directory>')
.usage(`${chalk.green('<project-directory>')} [options]`)
.action((name) => {
projectName = name
})
.parse(process.argv)
if (typeof projectName === 'undefined') {
console.error('Please specify the project directory:')
console.log(
`  ${chalk.cyan(program.name())} ${chalk.green('<project-directory>')}`
)
console.log()
console.log('For example:')
console.log(`  ${chalk.cyan(program.name())} ${chalk.green('my-app')}`)
process.exit(1)
}

3. 判断当前脚手架版本是否是最新

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
// 获取最新的version
const checkForLatestVersion = (url) => {
return new Promise((resolve, reject) => {
http
.get(url, (res) => {
if (res.statusCode === 200) {
let body = ''
res.on('data', (data) => (body += data))
res.on('end', () => {
resolve(JSON.parse(body).version)
})
} else {
reject()
}
})
.on('error', () => {
reject()
})
})
}

checkForLatestVersion(`${host}create-my-app/latest`)
.catch(() => {
try {
return execSync('npm view create-my-app version').toString().trim()
} catch (e) {
return null
}
})
.then((latest) => {
if (latest && semver.lt(packageJson.version, latest)) {
console.log()
console.error(
chalk.yellow(
`You are running \`create-my-app\` ${packageJson.version}, which is behind the latest release (${latest}).\n\n` +
'We recommend always using the latest version of create-my-app if possible.'
)
)
console.log()
console.log(
'The latest instructions for creating a new app can be found here:\n' +
`${host}-/web/detail/create-my-app`
)
console.log()
} else {
createApp()
}
})

4. 询问用户问题获取创建所需信息(使用 inquirer)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
const getAnswers = async () => {
  return await inquirer.prompt([
    {
      type: 'input',
      name: 'projectZHName',
      message: '请输入项目中文名',
      default: '',
    },
...,
    {
      type: 'input',
      name: 'version',
      message: '请输入版本',
      default: '1.0.0',
    },
    {
      type: 'input',
      name: 'author',
      message: '请输入创建人(拼音全拼)',
      default: '',
    },
  ])
}

5. 下载远程模板(使用 hyperquest 或 download-git-repo)

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
const latest = await checkForLatestVersion(
`${host}my-frontend-template/latest`
).catch(() => {
try {
return execSync('npm view my-frontend-template version').toString().trim()
} catch (e) {
return null
}
})
const { tmpdir, cleanup } = await getPackageInfo(
`${host}my-frontend-template/-/my-frontend-template-${latest}.tgz`
)

const extractStream = (stream, dest) => {
return new Promise((resolve, reject) => {
stream.pipe(
unpack(dest, (err) => {
if (err) {
reject(err)
} else {
resolve(dest)
}
})
)
})
}
const getTemporaryDirectory = () => {
return new Promise((resolve, reject) => {
// Unsafe cleanup lets us recursively delete the directory if it contains
// contents; by default it only allows removal if it's empty
tmp.dir({ unsafeCleanup: true }, (err, tmpdir, callback) => {
if (err) {
reject(err)
} else {
resolve({
tmpdir: tmpdir,
cleanup: () => {
try {
callback()
} catch (ignored) {
// Callback might throw and fail, since it's a temp directory the
// OS will clean it up eventually...
}
},
})
}
})
})
}

const getPackageInfo = (installPackage) => {
if (installPackage.match(/^.+\.(tgz|tar\.gz)$/)) {
return getTemporaryDirectory().then((obj) => {
let stream
if (/^http/.test(installPackage)) {
stream = hyperquest(installPackage)
} else {
stream = fsExtra.createReadStream(installPackage)
}
return extractStream(stream, obj.tmpdir).then(() => obj)
})
}
return Promise.resolve({ name: installPackage })
}

6. 根据需要读取修改文件和复制文件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
const { tmpdir, cleanup } = await getPackageInfo(
    `${host}my-frontend-template/-/my-frontend-template-${latest}.tgz`
)

const root = path.resolve(projectName)
fsExtra.mkdirSync(root)
process.chdir(root)

const templates = [
   ...,
    'src/common',
...
  ]
  for (const item of templates) {
    fsExtra.copy(templatePath(tmpdir, item), item)
  }

7. 自动下载依赖

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
const child = spawn('pnpm', ['install'], { stdio: 'inherit' })
child.on('close', (code) => {
if (code !== 0) {
return
}
console.log()
console.log(
`${chalk.cyan(projectName)} is created ${chalk.green('successfully')}`
)
console.log()
console.log('Get Started with the following commands:')
console.log()
console.log(
`${chalk.gray('$')} ${chalk.green('cd')} ${chalk.green(projectName)}`
)
console.log(`${chalk.gray('$')} ${chalk.green('pnpm run serve')}`)
})

8. 发布和使用

1
2
3
4
5
6
7
8
// package.json
{
...
"bin": {
"create-my-app": "./index.js"
},
...
}

将脚手架发布到公司私库内,在使用中使用一行命令即可完成生成工作。

1
npx create-my-app <project-directory>

小结

文章中贴出来了关键部分的脱敏代码,整体主要参考了create-react-app的实现,对其代码逻辑之严谨深受启发,其几乎对每一个环节可能出现的问题都做了第二种甚至第三种容错处理,在自己实现过程中,学习到了很多,对日后的开发大有帮助。

诞生背景

在打包工具出现之前,在浏览器中运行 Javascript 中有两种写法:

  • 第一种方式,引用一些脚本(script 标签)来存放每个功能;此解决方案很难扩展,因为加载太多脚本会导致网络瓶颈。
  • 第二种方式,使用一个包含所有项目代码的大型 .js 文件,但是这会导致作用域、文件大小、可读性和可维护性方面的问题。

历史的解决方案:

  • iife
  • commonjs(最大的问题是浏览器不支持 commonjs,因为 commonjs 是运行时 动态加载的,是同步的,浏览器同步的话,太慢了)
  • ESM
    • 未来的官方标准和主流。但是浏览器的版本需要比较高,比如 chorme 都需要 63 版本以上
    • esm 是静态的,可以在编译的时候就分析出对应的依赖关系,不用像 commonjs 一样,运行时加载

背景总结:

  1. commonjs 很好,推出 npm 管理 JavaScript 模块包,但浏览器不支持
  2. esm 更好,浏览器也支持,但只有很新的浏览器才支持。 你可以源代码内写 esm 模块,webpack 可以帮忙打包,让不兼容 esm 的浏览器,也能兼容

打包原理

在了解了背景之后,理解打包原理就很简单了。

webpack

所有内容打包到一个 chunk 包内(单 chunk 包)

无额外配置,webpack 一般会把所有 js 打成一个包。实现步骤

  • 读文件,扫描代码,按模块加载顺序,排列模块,分为模块 1,模块 2,…,模块 n 。放到一个作用域内,用 modules 保存,modules 是一个数组,所有模块按加载顺序,索引排序
  • webpack 自己实现对应的 api(比如自己实现 require),让浏览器支持源代码内的模块化的写法(比如:module.export, require, esm 稍微有些不同 见下方)打包外部依赖也是一样的
结果
  • 纯 commonjs
    • 所有的 js 依赖,打包到一个文件内,然后自己实现一套 require 和 module.exports,让浏览器可以执行源代码
    • 源代码的 require 会被换成 webpack_require
    • 源代码的 module.exports 不变,会由 webpack 作为函数的参数传给源代码
  • 纯 esm
    • webpack 会做 tree shaking,最终的产物,会和 rollup 的产物比较接近,不会有过多的 webpack 注入的兼容代码
    • 实现思路类似 rollup,通过 esm 的静态特性,可以在编译的时候,就分析出对应的依赖关系
  • esm + commonjs 混用
    • webpack 很强大,他是支持混用的!!
    • 你可以 module.exports 导出, import xx from xx 导入
    • 也可以 exports { } 导出,require 引入
    • 实现的思路和上面的模拟 module.exports 和提供webpack_require替代 require 的思路类似,webpack 会去模拟 esm 的 exports 对象 让浏览器支持

多个 chunk 包

多个打包入口
  • 多个入口分离多个包,然后生成多个 script 标签(按入口的顺序
  • 分离出来的多个包,都包含同样多的模拟代码(webpack 注入的代码)
分离公共依赖
  • 先加载 venders 包(第三方公共依赖),此加载不是解析代码,只是把第三方依赖的模块,以 webpack 能解析的格式,存到全局对象 window[“webpackJsonp”]内,方便后续的代码能访问到
  • 只需要把 window[“webpackJsonp”]内的 venders 内的模块,放到 main 代码作用域内的 modules 里面,后面就和单 chunk 解析是一样的了
import() 动态加载(懒加载)
  • 先执行 main 模块的内容,从上到下执行,关注 import(‘xx’).then()行。打包后,import()会被替换成 webpack 的 api(webpack_require.e(/_ import() _/ 1).then(webpack_require.bind(null, 1)).then())

  • 替换后的 api 做了几件事

    • 生成 script 标签,并 appendChild 到 ducument.head 内
    • return 一个 Promise 对象,状态是 pending(pending 状态不会往后执行.then(webpack_require.bind(null, 1)).then(),但不会阻塞主程序,因为是异步的,不懂的可以了解一下 promise)
    • (异步)等了一段时间后,需求懒加载的模块通过 script 标签,被下载到浏览器后会直接解析执行,触发 window[“webpackJsonp”].push(此方法被改写了,和生成同步多 chunk 有点不一样,会触发 webpackJsonpCallback 函数
    • webpackJsonpCallback 函数的作用
      • 懒加载的模块 内容 会被加入到 main 的 mudules 的模块列表内去(等效 push 的作用)
      • 会把 Promise 的状态从 pending 改成 fulfilled,因为要懒加载的模块,通过 script 标签,已经解析完成了,所以.then()可以往后了
    • 后面就是正常解析包,和单 chunk 解析多模块是一样的了

rollup

浏览器环境使用的话:

  1. 无需考虑浏览器兼容问题的话:
    • 开发者写 esm 代码 -> rollup 通过入口,递归识别 esm 模块 -> 最终打包成一个或多个 bundle.js -> 浏览器直接可以支持引入
  2. 需考虑浏览器兼容问题的话
    • 可能会比较复杂,需要用额外的 polyfill 库,或结合 webpack 使用

打包成 npm 包的话:

  • 开发者写 esm 代码 -> rollup 通过入口,递归识别 esm 模块 -> (可以支持配置输出多种格式的模块,如 esm、cjs、umd、amd)最终打包成一个或多个 bundle.js
    • (开发者要写 cjs 也可以,需要插件@rollup/plugin-commonjs) 初步看来
  • 很明显,rollup 比较适合打包 js 库(react、vue2 等的源代码库都是 rollup 打包的)或 高版本无需往下兼容的浏览器应用程序
  • 这样打包出来的库,可以充分使用上 esm 的 tree shaking,使源库体积最小

单 chunk 包

无额外配置,一般会把所有 js 打成一个包。打包外部依赖(第三方)也是一样的

多 chunk 包

  • 配置多个入口,此法比较简单,可自行测试
  • 代码分离 (动态 import,懒加载, import(xxx).then(module => {}) )

总结

webpack

webpack 诞生在 esm 标准出来前,commonjs 出来后

  • 当时的浏览器只能通过 script 标签加载模块

    • script 标签加载代码是没有作用域的,只能在代码内 用 iife 的方式 实现作用域效果
      • 这就是 webpack 打包出来的代码 大结构都是 iife 的原因
      • 并且每个模块都要装到 function 里面,才能保证互相之间作用域不干扰。
      • 这就是为什么 webpack 打包的代码为什么乍看会感觉乱,找不到自己写的代码的真正原因
  • 关于 webpack 的代码注入问题,是因为浏览器不支持 cjs,所以 webpack 要去自己实现 require 和 module.exports 方法(才有很多注入)

    • 这么多年了,甚至到现在 2022 年,浏览器为什么不支持 cjs
      • cjs 是同步的,运行时的,node 环境用 cjs,node 本身运行在服务器,无需等待网络握手,所以同步处理是很快的
      • 浏览器是 客户端,访问的是服务端资源,中间需要等待网络握手,可能会很慢,所以不能 同步的 卡在那里等服务器返回的,体验太差
  • 后续出来 esm 后,webpack 为了兼容以前发在 npm 上的老包(并且当时心还不够决绝,导致这种“丑结构的包”越来越多,以后就更不可能改这种“丑结构了”),所以保留这个 iife 的结构和代码注入,导致现在看 webpack 打包的产物,乍看结构比较乱且有很多的代码注入,自己写的代码都找不到

rollup

rollup 诞生在 esm 标准出来后

  • 出发点就是希望开发者去写 esm 模块,这样适合做代码静态分析,可以做 tree shaking 减少代码体积,也是浏览器除了 script 标签外,真正让 JavaScript 拥有模块化能力。是 js 语言的未来
  • rollup 完全依赖高版本浏览器原生去支持 esm 模块,所以无额外代码注入,打包后的代码结构也是清晰的(不用像 webpack 那样 iife)
    • 目前浏览器支持模块化只有 3 种方法:
      • script 标签(缺点没有作用域的概念
      • script 标签 + iife + window + 函数作用域(可以解决作用域问题。webpack 的打包的产物就这样
      • esm (什么都好,唯一缺点 需要高版本浏览器)

最终使用推荐

  1. 打包开源库:rollup 会是更好的选择
  2. 打包应用程序:看是否需要兼容老浏览器
    如果不考虑兼容老浏览器,建议用 vite 开发应用程序,vite打包实际使用的就是rollup,开发体验很棒,打的生产包比用webpack小很多,有不错的性能提升
    如果需要考虑兼容,则选择webpack

前言

TypeScript带来的类型系统以及强大的IDE支持,让前端开发也变得严谨而流畅。但TypeScript不是原生的Javascript代码,需要进行编译才能转换为Javascript代码。

tsconfig.json是编译TypeScript的配置文件,对书写TypeScript代码十分重要。接下来会介绍一些常用的编译选项和所有tsconfig.json选项的解释。

常用属性

1.experimentalDecorators

是否启用实验性的ES装饰器。 默认值:false, 官方解释

TypeScript 和 ES6 中引入了 Class 的概念,同时在 stage 2 proposal (opens new window)提出了 Java 等服务器端语言早就有的装饰器模式。通过引入装饰器模式,能极大简化书写代码,把一些通用逻辑封装到装饰器中。

2.strictPropertyInitialization

是否类的非undefined属性已经在构造函数里初始化。 默认值:false

1
2
3
export default class Home {
id: string // 如果开启strictPropertyInitialization,则这里会报错,因为没有赋值默认值
}

3.noImplicitAny

有隐含的 any类型时是否报错。 默认值:false

ts 是有默认推导的,同时还有 any 类型,所以不是每个变量或参数定义需要明确告知类型是什么。如果开启该值,当有隐含 any 类型时,会报错。

1
2
3
// 当开启noImplicitAny时,需要隐含当any需要明确指出
arr.find((item) => item.name === name) // error
arr.find((item: any) => item.name === name) // ok

4.target

指定编译的ECMAScript目标版本。默认值: “ES3”官方解释
枚举值:”ES3”, “ES5”, “ES6”/ “ES2015”, “ES2016”, “ES2017”,”ES2018”,”ES2019”,”ES2020”,”ES2021”,”ES2022”,”ESNext”

target: “ESNext” 是指 tc39 最新的ECMAScript proposals

当编译 ts 代码时,可以把 ts 转为 ES5 或更早的 js 代码。所以需要选择一个编译的目标版本。

5.module

指定生成哪个模块系统代码 官方解释

枚举值:”None”, “CommonJS”, “AMD”, “System”, “UMD”, “ES6”, “ES2015”,”ES2020”, “ES2022”, “ESNext”, “node16”, “nodenext”,

默认值根据–target 选项不同而不同,当 target 设置为 ES6 时,默认 module 为“ES6”,否则为“commonjs”

6.lib

编译过程中需要引入的库文件的列表

string[]类型,可选的值有很多,常用的有 ES5,ES6,ESNext,DOM,DOM.Iterable、WebWorker、ScriptHost 等。

该值默认值是根据–target 选项不同而不同。
当 target 为 ES5 时,默认值为[‘DOM ‘, ‘ES5’, ‘ScriptHost’];
当 target 为 ES6 时,默认值为[‘DOM’, ‘ES6’, ‘DOM.Iterable’, ‘ScriptHost’]

7.moduleResolution

决定如何处理模块 官方解释

说直白点,也就是遇到 import { AAA } from ‘./aaa’该如何去找对应文件模块解析。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// 在源文件/root/src/A.ts中import { b } from "./moduleB"
// 两种解析方式查找文件方式不同

// classic模块解析方式
1. /root/src/moduleB.ts
2. /root/src/moduleB.d.ts

// node模块解析方式
1. /root/src/moduleB.ts
2. /root/src/moduleB.tsx
3. /root/src/moduleB.d.ts
4. /root/src/moduleB/package.json (if it specifies a "types" property)
5. /root/src/moduleB/index.ts
6. /root/src/moduleB/index.tsx
7. /root/src/moduleB/index.d.ts

8.paths

模块名或路径映射的列表

这是一个非常有用的选项,比如我们经常使用‘@/util/help’来代替’./src/util/help’,省的每次在不同层级文件 import 模块时,都纠结于是’./‘还是’../‘。该选项告诉编译器遇到匹配的值时,去映射的路径下加载模块。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
{
"baseUrl": ".", // 注意:baseUrl不可少
"paths": {
// 映射列表
"@/*": [
"src/*"
],
"moduleA": [
"src/libs/moduleA"
]
}
}

// in ts code
import Setting from '@/components/Setting.vue' // 模块实际位置: src/components/Setting.vue
import TestModule from 'moduleA/index.js' // 模块实际位置: src/libs/moduleA/index.js

9.strictNullChecks

是否启用严格的 null检查模式 默认值: false 建议开启该选项

未处理的 null 和 undefined 经常会导致 BUG 的产生,所以 TypeScript 包含了 strictNullChecks 选项来帮助我们减少对这种情况的担忧。当启用了 strictNullChecks,null 和 undefined 获得了它们自己各自的类型 null 和 undefined。开启该模式有助于发现并处理可能为 undefined 的赋值。

1
2
3
4
5
6
7
8
9
// 未开启strictNullChecks,number类型包含了null和undefined类型
let foo: number = 123
foo = null // Okay
foo = undefined // Okay

// 开启strictNullChecks
let foo: string[] | undefined = arr.find((key) => key === 'test')
// foo.push('1') // error - 'foo' is possibly 'undefined'
foo && foo.push('1') // okay

注意:启用 –strict 相当于启用 –noImplicitAny, –noImplicitThis, –alwaysStrict, –strictNullChecks, –strictFunctionTypes 和–strictPropertyInitialization

10.noUnusedLocals

有未使用的变量时,是否抛出错误 默认值: false

当发现变量定义但没有使用时,编译不报错。eslint 的 rule 中也有该条,建议正式项目将该选项开启,设置为 true,使得代码干净整洁

11.noUnusedParameters

有未使用的参数时,是否抛出错误 默认值: false

建议开启,理由同上。

12.allowJs

是否允许编译javascript文件 默认值: false

如果设置为 true,js 后缀的文件也会被 typescript 进行编译。

13.typeRoots 和 types

默认所有可见的”@types”包会在编译过程中被包含进来。如果指定了 typeRoots,只有 typeRoots 下面的包才会被包含进来。如果指定了 types,只有被列出来的 npm 包才会被包含进来。详细解释

可以指定”types”: []来禁用自动引入@types 包#14. files、include 和 exclude

14.files、include 和 exclude

编译文件包含哪些文件以及排除哪些文件

未设置 include 时,编译器默认包含当前目录和子目录下所有的 TypeScript 文件(.ts, .d.ts 和 .tsx)。如果 allowJs 被设置成 true,JS 文件(.js 和.jsx)也被包含进来。exclude 排除那些不需要编译的文件或文件夹。

1
2
3
4
5
6
7
8
9
10
{
"compilerOptions": {},
"include": [
"src/**/*"
],
"exclude": [
"node_modules",
"**/*.spec.ts"
]
}

全解析

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
{
"compilerOptions": {
/* 基本选项 */
"target": "es5", // 指定 ECMAScript 目标版本: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017', or 'ESNEXT'("ESNext"表示最新的ES语法,包括还处在stage X阶段)
"module": "commonjs", // 指定使用模块: 'commonjs', 'amd', 'system', 'umd' or 'es2015'
"lib": [], // 指定要包含在编译中的库文件
"allowJs": true, // 允许编译 javascript 文件
"checkJs": true, // 报告 javascript 文件中的错误
"jsx": "preserve", // 指定 jsx 代码的生成: 'preserve', 'react-native', or 'react'
"declaration": true, // 生成相应的 '.d.ts' 文件
"sourceMap": true, // 生成相应的 '.map' 文件
"outFile": "./", // 将输出文件合并为一个文件
"outDir": "./", // 指定输出目录
"rootDir": "./", // 用来控制输出目录结构 --outDir.
"removeComments": true, // 删除编译后的所有的注释
"noEmit": true, // 不生成输出文件
"importHelpers": true, // 从 tslib 导入辅助工具函数
"isolatedModules": true, // 将每个文件做为单独的模块 (与 'ts.transpileModule' 类似).

/* 严格的类型检查选项 */
"strict": true, // 启用所有严格类型检查选项
"noImplicitAny": true, // 在表达式和声明上有隐含的 any类型时报错
"strictNullChecks": true, // 启用严格的 null 检查
"noImplicitThis": true, // 当 this 表达式值为 any 类型的时候,生成一个错误
"alwaysStrict": true, // 以严格模式检查每个模块,并在每个文件里加入 'use strict'

/* 额外的检查 */
"noUnusedLocals": true, // 有未使用的变量时,抛出错误
"noUnusedParameters": true, // 有未使用的参数时,抛出错误
"noImplicitReturns": true, // 并不是所有函数里的代码都有返回值时,抛出错误
"noFallthroughCasesInSwitch": true, // 报告 switch 语句的 fallthrough 错误。(即,不允许 switch 的 case 语句贯穿)

/* 模块解析选项 */
"moduleResolution": "node", // 选择模块解析策略: 'node' (Node.js) or 'classic' (TypeScript pre-1.6)。默认是classic
"baseUrl": "./", // 用于解析非相对模块名称的基目录
"paths": {}, // 模块名到基于 baseUrl 的路径映射的列表
"rootDirs": [], // 根文件夹列表,其组合内容表示项目运行时的结构内容
"typeRoots": [], // 包含类型声明的文件列表
"types": [], // 需要包含的类型声明文件名列表
"allowSyntheticDefaultImports": true, // 允许从没有设置默认导出的模块中默认导入。

/* Source Map Options */
"sourceRoot": "./", // 指定调试器应该找到 TypeScript 文件而不是源文件的位置
"mapRoot": "./", // 指定调试器应该找到映射文件而不是生成文件的位置
"inlineSourceMap": true, // 生成单个 soucemaps 文件,而不是将 sourcemaps 生成不同的文件
"inlineSources": true, // 将代码与 sourcemaps 生成到一个文件中,要求同时设置了 --inlineSourceMap 或 --sourceMap 属性

/* 其他选项 */
"experimentalDecorators": true, // 启用装饰器
"emitDecoratorMetadata": true, // 为装饰器提供元数据的支持
"strictFunctionTypes": false // 禁用函数参数双向协变检查。
},
/* 指定编译文件或排除指定编译文件 */
"include": [
"src/**/*"
],
"exclude": [
"node_modules",
"**/*.spec.ts"
],
"files": [
"core.ts",
"sys.ts"
],
// 从另一个配置文件里继承配置
"extends": "./config/base",
// 让IDE在保存文件的时候根据tsconfig.json重新生成文件
"compileOnSave": true // 支持这个特性需要Visual Studio 2015, TypeScript1.8.4以上并且安装atom-typescript插件
}

起因

目前正在推动团队做持续部署,包括自动提交 npm 包到公司私库和自动生成 CHANGELOG,重中之重是根据 commit 来自动判断版本号,所以规范 git commit message 势在必行。
本文就介绍下团队是如何做的 commit message 的规范和格式化。

Commit Message 格式

1
<type>(<scope>): <subject>

type(必须)

  • feat: 新功能(feature)
  • fix: 修复 bug
  • perf: 优化相关,比如提升性能、体验
  • refactor: 重构(即不是新增功能,也不是修改 bug 的代码变动)
  • docs: 文档(documentation)
  • style: 格式(不影响代码运行的变动, 不是 css 样式)
  • test: 增加测试

scope(可选)

  • scope 用于说明 commit 影响的范围

subject(必须)

  • subject 是 commit 目的的简短描述,不超过 50 个字符。
1
2
3
e.g.
fix: 修复xxx问题
feat: 新增xxx功能

Commintlint 校验你的 Message

commitlint 可以帮助我们 lint commit messages, 如果我们提交的不符合预设的规范, 直接拒绝提交。

目前团队使用的是
@commitlint/config-conventional
,因为团队后面要使用
@semantic-release/commit-analyzer进行一系列自动化操作,

,二者底层均依赖conventional-changelog-conventionalcommits,属于完美适配。

结合 Husky

本地校验 commit message 的最佳方式是结合 git hook, 所以需要配合 Husky。

1
2
3
4
5
npm install husky --save-dev

npx husky install

npm set-script prepare "husky install"
执行完上面命令将带来以下几个变化
  • 在.git 同级目录生成.husky 文件夹,文件夹下有一个可以编辑的示例 pre-commit 钩子
  • 在 package.json 中的 scripts 中添加了”prepare”: “husky install”
  • 更改 git 配置项 core.hooksPath 为.husky
package.json 配置:
1
2
3
4
5
"husky": {
        "hooks": {
            "commit-msg": "commitlint -E HUSKY_GIT_PARAMS"
        }
    },

shell 脚本

上文的操作中会有一个小 bug,就是在本地绕过 git hook 校验的情况,当通过 gitlab 做持续部署时,会导致自动计算版本号等操作出错,所以在持续部署脚本启动前,通过 commit lint shell 脚本来避免这种情况发生。

最后

commit message 的规范性很重要, 但是是否需要像本文这样强制限制, 每个团队和个人都有自己的想法, 但是个人认为: 好的习惯, 受益终身.