vueJs 源码解析 (三) 具体代码

在之前的文章中提到了 vuejs 源码中的 架构部分,以及 谈论到了 vue 源码三要素 vm、compiler、watcher 这三要素,那么今天我们就从这三要素逐步了解清楚

好了,话不多说, let's do it

在这之前,我们需要 对上文中讲到的 vuejs 的源码是 flow 写法的问题进行一个简化。 毕竟还有有工具是可以解决的。

可以用babel-plugin-transform-flow-strip-types去转化下即可。

1、 npm install --save-dev babel-plugin-transform-flow-strip-types 2、 .babelrc 文件中 {
"plugins": ["transform-flow-strip-types"]
}

具体转换方法见 github地址

一、 instance 实例化入口 核心代码
/src/core/instance/index.js

import { initMixin } from './init' // 实例化 混合
import { stateMixin } from './state' // 各类数据应用 混合
import { renderMixin } from './render' // render 函数等 混合
import { eventsMixin } from './events' // 例如 父子组件的 emit on 事件
import { lifecycleMixin } from './lifecycle' // 这个暂时比较模糊,后面的文章更新
import { warn } from '../util/index' // warn 报错工具 在控制台经常会看到的 vue 的warn function Vue (options) {
if (process.env.NODE_ENV !== 'production' &&
!(this instanceof Vue) // 这里就是判断当前 this 的 prototype 是否是 Vue
) {
warn('Vue is a constructor and should be called with the `new` keyword')
}
this._init(options)
} initMixin(Vue)
stateMixin(Vue)
eventsMixin(Vue)
lifecycleMixin(Vue)
renderMixin(Vue) export default Vue
一、 instance 实例化入口 核心代码 之 init.js

核心代码区块一:


if (options && options._isComponent) {
// optimize internal component instantiation
// since dynamic options merging is pretty slow, and none of the
// internal component options needs special treatment.
initInternalComponent(vm, options);
} else {
vm.$options = mergeOptions(resolveConstructorOptions(vm.constructor),
options || {}, vm);
}

解析:

判断传入的值当中是否有组件,如果有则先实例化组件。

核心代码区块二:


vm._self = vm;
initLifecycle(vm);
initEvents(vm);
initRender(vm);
callHook(vm, 'beforeCreate');
initInjections(vm); // resolve injections before data/props
initState(vm);
initProvide(vm); // resolve provide after data/props
callHook(vm, 'created');

解析:

看字面上的意思是分别一次 实例化了 生命周期、事件、渲染函数 、回调钩子 ‘beforeCreate’、

依赖注入、状态、 提供 、回调钩子 ‘created’、

对,看到这里我们还是很奇怪这些东西是干嘛的?

那么下面我们继续依次往下看:

一、实例化生命周期 initLifecycle


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;

看上去是增加了一系列的 属性。但是还是不知道这个有什么用。不着急,我们继续往下看。

二、实例化生命周期 initEvents


export function initEvents(vm) {
vm._events = Object.create(null);
vm._hasHookEvent = false;
// init parent attached events
const listeners = vm.$options._parentListeners;
if (listeners) {
updateComponentListeners(vm, listeners);
}
}

再走到这步的时候就会发现, vm.$options 这个对象 频频出现在我们的视野中,如果每次都不能理解的话。就会遇到 理解障碍。 那么 我们就需要 做一个 最简单的测试用例。 (实际的把 vue 跑起来)

三、new Vue({ options }) Vue 最简单的测试用例

我们在created 钩子下打印 this 对象 部分结果如下

console.log(this)
// console.log(this)
_events: {}
_hasHookEvent :false
_inactive :null
_isBeingDestroyed :false
_isDestroyed :false
_isMounted: true
_isVue: true

这里我们就能够看到比较清晰的一些属性了 在 init.js 中的第一步 initLifecycle.js 中定义的

// console.log(this)
_events: {} // 事件对象集合
_hasHookEvent :false // 是否有钩子事件
_inactive :null // 未知
_isBeingDestroyed :false // 是否要销毁
_isDestroyed :false // 是否已经销毁
_isMounted: true // 是否已挂载
_isVue: true // 是否是 Vue 对象

这里我们就再回到 initLifecycle.js 源码 中去看

  const options = vm.$options;
let parent = options.parent;
if (parent && !options.abstract) {
while (parent.$options.abstract && parent.$parent) {
parent = parent.$parent;
}
parent.$children.push(vm);
} vm.$parent = parent;
vm.$root = parent ? parent.$root : vm; vm.$children = []; // 这里定义的 children 数字对象未知是什么意思
vm.$refs = {}; // 这里定义的 refs 对象依然未知

走到这一步 大概对 vm.$options 这个对象 有个初步的了解,但是还有部分依然是未知。好了,我们继续往下走。

四、 我回到 第二步 实例化生命周期 initEvents


export function initEvents(vm) {
vm._events = Object.create(null);
vm._hasHookEvent = false;
// init parent attached events
const listeners = vm.$options._parentListeners;
if (listeners) {
updateComponentListeners(vm, listeners);
}
}

给 vm 对象 新增了一个 _events 对象 , 并且会去判断 vm.$options 是否有父级的事件监听。


propsData : undefined
_componentTag : "App"
_parentElm : null
_parentListeners : undefined
_parentVnode : VNode {tag: "vue-component-1-Apps-test", data: {…}, children: undefined, text: undefined, elm: div.test11, …}
_refElm : null
_renderChildren : undefined

_parentListeners 值为空。 这里我们做一个大胆的假设: 是否有组件的引用的时候这个值就会发生改变。 那么下面我们简单的测试下。新增一个基础组件.

// 验证失败:这个值依然还是为 空

// 但是弄清楚了一个问题:

this.$root   				// 为根对象
this.$root.$parent // 根对象的 父属性一定是为空
this.$root.$children // 根对象的 子代属性一定是 一个数组。 // 如果你引用的 组件
this.$root.$children[0] // 为第一个组件
this.$root.$children[1] // 为第二个组件
... // 以此类推

好,我们接着往下看

五、 initRender(vm)

第五步: 比较核心的 渲染功能。

今天先到这里... 先下个班,回家继续整理

vueJs 源码解析 (三) 具体代码的更多相关文章

  1. ReactiveCocoa源码解析(三) Signal代码的基本实现

    上篇博客我们详细的聊了ReactiveSwift源码中的Bag容器,详情请参见<ReactiveSwift源码解析之Bag容器>.本篇博客我们就来聊一下信号量,也就是Signal的的几种状 ...

  2. ReactiveSwift源码解析(三) Signal代码的基本实现

    上篇博客我们详细的聊了ReactiveSwift源码中的Bag容器,详情请参见<ReactiveSwift源码解析之Bag容器>.本篇博客我们就来聊一下信号量,也就是Signal的的几种状 ...

  3. VueJs 源码解析 (四) initRender.Js

    vueJs 源码解析 (四) initRender.Js 在之前的文章中提到了 vuejs 源码中的 架构部分,以及 谈论到了 vue 源码三要素 vm.compiler.watcher 这三要素,那 ...

  4. Celery 源码解析三: Task 对象的实现

    Task 的实现在 Celery 中你会发现有两处,一处位于 celery/app/task.py,这是第一个:第二个位于 celery/task/base.py 中,这是第二个.他们之间是有关系的, ...

  5. Mybatis源码解析(三) —— Mapper代理类的生成

    Mybatis源码解析(三) -- Mapper代理类的生成   在本系列第一篇文章已经讲述过在Mybatis-Spring项目中,是通过 MapperFactoryBean 的 getObject( ...

  6. React的React.createRef()/forwardRef()源码解析(三)

    1.refs三种使用用法 1.字符串 1.1 dom节点上使用 获取真实的dom节点 //使用步骤: 1. <input ref="stringRef" /> 2. t ...

  7. Spring源码解析 - springMVC核心代码

    一.首先来讲解下springMVC的底层工作流程 1.首先我们重点放在前端控制器(DispatcherServlet) 其类图: 因为从流程图看,用户的请求最先到达就是DispatcherServle ...

  8. Vue源码解析-调试环境-代码目录和运行构建

    目录 前言 1 代码结构 1.1 octotree插件 1.2 vue工程项目目录 1.3 主要代码目录src compiler core platforms server sfc shared 2 ...

  9. Spring源码解析三:IOC容器的依赖注入

    一般情况下,依赖注入的过程是发生在用户第一次向容器索要Bean是触发的,而触发依赖注入的地方就是BeanFactory的getBean方法. 这里以DefaultListableBeanFactory ...

随机推荐

  1. 怎么修改无法启动的docker容器的配置?

    原因: 由于错误的配置导致原来可以启动的docker容器不能启动了.相信很多人的做法是删除容器重建一个,这样也是可以的,但是你的配置和插件就得重新安装,非常麻烦.最小的代价当然是修改原来的配置让他能启 ...

  2. Hive数据仓库笔记(一)

    Hive建表: CREATE TABLE records (year STRING,temperature INT, quality INT) ROW FORMAT DELIMITED FIELDS ...

  3. Java虚拟机系列-Java类加载机制

    原文:http://www.ityouknow.com/jvm/2017/08/19/class-loading-principle.html 一. 类加载: 1. 将.class文件的二进制数据加载 ...

  4. Gatling - 用 session 实现关联 传递 token 值

    项目中的某个接口必须先登录后调用,但是 header 中的Authorization 需要在登录返回的token中添加一个字串,所以需要先获得 token 并修改后传递给该接口的请求. 虽然这是常见的 ...

  5. centos7上安装ELK

    author:headsen chen data :2017-12-04  18:00:57 notice:created by chen himself and not allowed to cop ...

  6. java语言环境jdk的安装和环境变量的配置

    一.jdk的安装 我安装的为64位的1.7.0_17版本的jdk,双击运行,全部默认下一步就行 此处可整改安装目录,然后点击下一步进行安装. 二.环境变量的配置 1.右键点击我的电脑à点击属性,出现如 ...

  7. ASP.NET Core Web 支付功能接入 支付宝-电脑网页支付篇

    这篇文章将介绍ASP.NET Core中使用 开源项目 Payment,实现接入支付宝-电脑网页支付接口及同步跳转及异步通知功能. 开发环境:Win 10 x64.VS2017 15.6.4..NET ...

  8. Django+xadmin打造在线教育平台(三)

    五.完成注册.找回密码和激活验证码功能 5.1.用户注册 register.html拷贝到templates目录 (1)users/views.py class RegisterView(View): ...

  9. 笔记:Spring Cloud Eureka 服务治理

    Spring Cloud Eureka 是 Spring Cloud Netflix 微服务套件的一部分,基于 Netflix Eureka 做了二次封装,主要负责完成微服务架构中的服务治理功能,服务 ...

  10. bashell基础

    身为一个iOS程序员,虽然iOS相关技术十分重要,但是bash也是不可不了解的,因为技能的成长,除了深度,还需要广度.下面就来介绍下bash. Shell是C语言编写的,所以他是解释性语言,运行在Li ...