热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Vue源码学习(二)——从宏观看Vue

上一篇文章我们写到从入口文件一步步找到Vue的构造函数,现在我们要去看看Vue实例化经历的过程Vue的构造函数我们知道Vue的构造函数在srccoreinstanceindex.j

上一篇文章我们写到从入口文件一步步找到Vue的构造函数,现在我们要去看看Vue实例化经历的过程

Vue的构造函数

我们知道Vue的构造函数在src/core/instance/index.js中,不明白的可以去看上一篇文章 Vue源码学习笔记一。那我们关注一下Vue的构造函数的内容:

// src/core/instance/index.js
import { initMixin } from './init'
// Vue的构造函数
function Vue (options) {
//... 验证环境
this._init(options)
}
// 在Vue原型上绑定实例方法
initMixin(Vue) // init
stateMixin(Vue) // $set $delete $watch
eventsMixin(Vue) // $on $once $off $emit
lifecycleMixin(Vue) // _update $forceUpdate $destroy
renderMixin(Vue) // $nextTick _render

添加Vue属性和方法

这边我们可以看到Vue的构造函数中执行了init方法,从下方得知init是在src\core\instance\init.js中导出的initMixin函数中定义的

initMixin

  1. vmthis ,同时为实例添加一个唯一的uid vm._isVue = true 监听对象变化时用于过滤vm,因为Vue的实例是不需要监听变化的。

// src/core/instance/init.js
Vue.prototype._init = function (options?: Object) {
const vm: CompOnent= this
// 当前实例添加了一个唯一的uid
vm._uid = uid++
// ...
// 监听对象变化时用于过滤vm
vm._isVue = true
//...
}

  1. 参数处理,根据我们的小栗子,我们的options处理直接进入了else,然后对参数进行合并,这里是对vue extend的参数需要进行合并处理,我们这里resolveConstructorOptions 返回的即是constructor.options本身
  2. 生命周期相关变量初始化 initLifecycle(vm)

// src\core\instance\lifecycle.js
// 为组件挂载相应属性,并初始化
vm.$parent = parent
vm.$root = parent ? parent.$root : vm
vm.$children = []
vm.$refs = {}
vm._watcher = null
vm._inactive = null
vm._directInactive = false
vm._isMounted = false
vm._isDestroyed = false
vm._isBeingDestroyed = false

4.vm 事件监听初始化 initEvents()

// src/core/instance/events.js
export function initEvents (vm: Component) {
// 创建事件对象,用于存储事件
vm._events = Object.create(null)
// 系统事件标识位
vm._hasHookEvent = false
// init parent attached events npm
// 将父组件模板中注册的事件放到当前组件实例的listeners
const listeners = vm.$options._parentListeners
if (listeners) {
updateComponentListeners(vm, listeners)
}
}

initEvents(vm)
initRender(vm)
callHook(vm, 'beforeCreate')
initInjections(vm)
// vm状态初始化,prop/data/computed/method/watch都在这里初始化完成,vue实例create的关键
initState(vm)
initProvide(vm)
callHook(vm, 'created')

stateMixin

Vue实例方法–数据,该文件对应的是Vue的数据的处理,首先对$data进行挂载,然后设置数据
$set、删除数据
$delete、观测数据
$watch方法挂载

// stateMixin(Vue) src/core/instance/state.js
export function stateMixin (Vue: Class) {
// data
const dataDef = {}
dataDef.get = function () { return this._data }
// prop
const propsDef = {}
propsDef.get = function () { return this._props }
// ...
// 定义$data & prop属性
Object.defineProperty(Vue.prototype, '$data', dataDef)
Object.defineProperty(Vue.prototype, '$props', propsDef)
// 原型链添加函数set 和 delete
Vue.prototype.$set = set
Vue.prototype.$delete = del
// 原型链添加函数$watch
Vue.prototype.$watch = function (){
// ...
}
}

eventsMixin

Vue实例方法–事件,该文件主要挂载Vue实例方法的事件,监听事件
on once、移除事件
off、触发事件
emit的挂载

// eventsMixin(Vue) src/core/instance/events.js
export function eventsMixin (Vue: Class) {
Vue.prototype.$on = function (event: string, fn: Function): Component {
// ...
}
Vue.prototype.$Once= function (event: string, fn: Function): Component {
// ...
}
Vue.prototype.$off = function (event?: string, fn?: Function): Component {
// ...
}
Vue.prototype.$emit = function (event: string): Component {
// ...
}
}

lifecycleMixin

Vue实例方法–生命周期,,该文件主要挂载Vue实例方法中的生命周期方法,重新渲染
$forceUpdate()、销毁实例
$destroy()

// lifecycleMixin(Vue) src/core/instance/lifecycle.js
Vue.prototype._mount = function(){}
Vue.prototype._update = function (vnode: VNode, hydrating?: boolean) {}
Vue.prototype._updateFromParent = function(){}
Vue.prototype.$forceUpdate = function () {}
Vue.prototype.$destroy = function () {}

renderMixin

文件主要挂载Vue实例方法中的dom更新回调
$nextTick及一些其他的render函数,后续我们再的深挖一下

// renderMixin(Vue) src/core/instance/render.js
Vue.prototype.$nextTick = function (fn: Function) {}
Vue.prototype._render = function (): VNode {}
Vue.prototype._s = _toString
Vue.prototype._v = createTextVNode
Vue.prototype._n = toNumber
Vue.prototype._e = createEmptyVNode
Vue.prototype._q = looseEqual
Vue.prototype._i = looseIndexOf
Vue.prototype._m = function(){}
Vue.prototype._o = function(){}
Vue.prototype._f = function resolveFilter (id) {}
Vue.prototype._l = function(){}
Vue.prototype._t = function(){}
Vue.prototype._b = function(){}
Vue.prototype._k = function(){}

全局API

上面部分,我们对Vue的构造函数,在
src/core/instance/index.js文件中的作用进行了大体的了解,当然这并没有结束,依据我们
Vue源码学习笔记一中提到的,我们追溯到上一级
src/core/index.js

// src/core/index.js
import Vue from './instance/index'
import { initGlobalAPI } from './global-api/index'
import { isServerRendering } from 'core/util/env'
import { FunctionalRenderContext } from 'core/vdom/create-functional-component'
// 初始化全局变量
initGlobalAPI(Vue)
// 为vue原型定义属性 isServer 判断是否为服务端渲染
Object.defineProperty(Vue.prototype, '$isServer', {
get: isServerRendering
})
// 为vue原型定义属性 ssrContext
Object.defineProperty(Vue.prototype, '$ssrContext', {
get () {
/* istanbul ignore next */
return this.$vnode && this.$vnode.ssrContext
}
})
Object.defineProperty(Vue, 'FunctionalRenderContext', {
value: FunctionalRenderContext
})
Vue.version = '__VERSION__'
export default Vue

initGlobalAPI(Vue)

在Vue 构造函数上挂载静态属性和方法即
全局API

// src/core/global-api/index.js
export function initGlobalAPI(Vue: GlobalAPI) {
const cOnfigDef= {}
configDef.get = () => config
// ...
Object.defineProperty(Vue, 'config', configDef)
Vue.util = { // Vue.util
warn,
extend,
mergeOptions,
defineReactive
}
Vue.set = set
Vue.delete = del
Vue.nextTick = nextTick
Vue.optiOns= Object.create(null)
ASSET_TYPES.forEach(type => {
Vue.options[type + 's'] = Object.create(null)
})
Vue.options._base = Vue
extend(Vue.options.components, builtInComponents)
initUse(Vue) // Vue.use
initMixin(Vue) // Vue.mixin
initExtend(Vue) // Vue.extend
initAssetRegisters(Vue) // Vue.component Vue.directive Vue.filter
}

内置组件&命令

在追溯到上一级,在文件
src/platforms/web/runtime/index.js,该文件注册了一些
Vue内置的组件:包裹动态组件
KeepAlive、元素过渡效果
Transition、多个元素过渡
TransitionGroup

// src/platforms/web/runtime/index.js 执行后
// 安装平台特定的utils
Vue.config.isUnknownElement = isUnknownElement
Vue.config.isReservedTag = isReservedTag
Vue.config.getTagNamespace = getTagNamespace
Vue.config.mustUseProp = mustUseProp
// 安装平台特定的 指令 和 组件
Vue.optiOns= {
components: {
KeepAlive,
Transition,
TransitionGroup
},
directives: {
model,
show
},
filters: {},
_base: Vue
}
Vue.prototype.__patch__
Vue.prototype.$mount

compiler编译器添加

再上一级为
src/platforms/web/entry-runtime-with-compiler.js,该文件对原来的
Vue.prototype.$mount进行覆盖定义,并且在Vue上挂载了
compile。给Vue的 $mount 方法添加 compiler 编译器,支持 template。

// src/platforms/web/entry-runtime-with-compiler.js
const mount = Vue.prototype.$mount
// ...
Vue.prototype.$mount = function (){
//... 覆盖 Vue.prototype.$mount
}
// ...
//在 Vue 上挂载 compile
//compileToFunctions 函数的作用,就是将模板 template 编译为render函数。
Vue.compile = compileToFunctions

总结

至此的话我们从宏观上过了一下从我们一层层找到vue到一层层往外看到对Vue的添加属性方法等,我们有了一个整体的概念

  1. src/core/instance/index.js vue的构造函数,添加Vue属性和方法
  2. src/core/index.js 全局API的挂载
  3. src/platforms/web/runtime/index.js 主要是添加web平台特有的配置、组件和指令
  4. web/entry-runtime-with-compiler.js 给Vue的 $mount 方法添加 compiler 编译器,支持 template
  5. scripts/config.js 编译入口文件

推荐阅读
  • 本文介绍了如何使用vue-awesome-swiper组件,包括在main.js中引入和使用swiper和swiperSlide组件,以及设置options和ref属性。同时还介绍了如何在模板中使用swiper和swiperSlide组件,并展示了如何通过循环渲染swipes数组中的数据,并使用picUrl属性显示图片。最后还介绍了如何添加分页器。 ... [详细]
  • Oracle seg,V$TEMPSEG_USAGE与Oracle排序的关系及使用方法
    本文介绍了Oracle seg,V$TEMPSEG_USAGE与Oracle排序之间的关系,V$TEMPSEG_USAGE是V_$SORT_USAGE的同义词,通过查询dba_objects和dba_synonyms视图可以了解到它们的详细信息。同时,还探讨了V$TEMPSEG_USAGE的使用方法。 ... [详细]
  • 单页面应用 VS 多页面应用的区别和适用场景
    本文主要介绍了单页面应用(SPA)和多页面应用(MPA)的区别和适用场景。单页面应用只有一个主页面,所有内容都包含在主页面中,页面切换快但需要做相关的调优;多页面应用有多个独立的页面,每个页面都要加载相关资源,页面切换慢但适用于对SEO要求较高的应用。文章还提到了两者在资源加载、过渡动画、路由模式和数据传递方面的差异。 ... [详细]
  • vue使用
    关键词: ... [详细]
  • 本文介绍了闭包的定义和运转机制,重点解释了闭包如何能够接触外部函数的作用域中的变量。通过词法作用域的查找规则,闭包可以访问外部函数的作用域。同时还提到了闭包的作用和影响。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 本文记录了在vue cli 3.x中移除console的一些采坑经验,通过使用uglifyjs-webpack-plugin插件,在vue.config.js中进行相关配置,包括设置minimizer、UglifyJsPlugin和compress等参数,最终成功移除了console。同时,还包括了一些可能出现的报错情况和解决方法。 ... [详细]
  • 网址:https:vue.docschina.orgv2guideforms.html表单input绑定基础用法可以通过使用v-model指令,在 ... [详细]
  • 本文讨论了微软的STL容器类是否线程安全。根据MSDN的回答,STL容器类包括vector、deque、list、queue、stack、priority_queue、valarray、map、hash_map、multimap、hash_multimap、set、hash_set、multiset、hash_multiset、basic_string和bitset。对于单个对象来说,多个线程同时读取是安全的。但如果一个线程正在写入一个对象,那么所有的读写操作都需要进行同步。 ... [详细]
  • 十大经典排序算法动图演示+Python实现
    本文介绍了十大经典排序算法的原理、演示和Python实现。排序算法分为内部排序和外部排序,常见的内部排序算法有插入排序、希尔排序、选择排序、冒泡排序、归并排序、快速排序、堆排序、基数排序等。文章还解释了时间复杂度和稳定性的概念,并提供了相关的名词解释。 ... [详细]
  • 用Vue实现的Demo商品管理效果图及实现代码
    本文介绍了一个使用Vue实现的Demo商品管理的效果图及实现代码。 ... [详细]
  • 在package.json中有如下两个对象:husky:{hooks:{pre-commit:lint-staged}},lint-staged:{src** ... [详细]
  • 本文介绍了Java工具类库Hutool,该工具包封装了对文件、流、加密解密、转码、正则、线程、XML等JDK方法的封装,并提供了各种Util工具类。同时,还介绍了Hutool的组件,包括动态代理、布隆过滤、缓存、定时任务等功能。该工具包可以简化Java代码,提高开发效率。 ... [详细]
  • Netty源代码分析服务器端启动ServerBootstrap初始化
    本文主要分析了Netty源代码中服务器端启动的过程,包括ServerBootstrap的初始化和相关参数的设置。通过分析NioEventLoopGroup、NioServerSocketChannel、ChannelOption.SO_BACKLOG等关键组件和选项的作用,深入理解Netty服务器端的启动过程。同时,还介绍了LoggingHandler的作用和使用方法,帮助读者更好地理解Netty源代码。 ... [详细]
  • 在C#中,使用关键字abstract来定义抽象类和抽象方法。抽象类是一种不能被实例化的类,它只提供部分实现,但可以被其他类继承并创建实例。抽象类可以用于类、方法、属性、索引器和事件。在一个类声明中使用abstract表示该类倾向于作为其他类的基类成员被标识为抽象,或者被包含在一个抽象类中,必须由其派生类实现。本文介绍了C#中抽象类和抽象方法的基础知识,并提供了一个示例代码。 ... [详细]
author-avatar
jasonmysh
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有