第 1 章、权衡的艺术

命令式 or 声明式

命令式:关注过程

声明式:关注结果

声明式直接声明想要的结果,框架帮用户封装好命令式的代码,所以在封装的过程中要做一些其他的事情来(生成要做的事情/找出差异)生成命令式的代码,优势是可维护性强,但是性能:命令式 ≥ 声明式

框架设计要做的就是,保持可维护性,同时让性能损失更少。

虚拟 DOM

性能:innerHTML < 虚拟DOM < 原生JavaScript

心智负担/可维护性:虚拟DOM < innerHTML < 原生JavaScript

运行时和编译时

三种选择:纯运行时、运行时+编译时、纯编译时

手写vdom太麻烦,所以通过模板 or JSX 完成书写,但是这些还需要经历一次编译

Vue:运行时+编译时

Svelte: 纯编译时

第 2 章、设计框架要注意什么?

提升开发体验

用户未按要求使用时的错误提示,console 自定义 formatter 直观显示数据

控制框架代码的体积

开发环境提供良好提示,不增加生产环境体积。(通过 __DEV__ 变量判断)

Tree-Shaking

使用 ESM

如果一个函数有副作用,将不会被 Tree-Shaking 删除,通过 /*#__PURE__*/ 在打包工作( rollup/webpack )中声明没有副作用。

输出产物

IIFE:rollup format: 'iife'

ESM: format: 'esm'

CommonJS: format: 'cjs'

特性开关

用户关闭的特性,利用 Tree-Shaking 不打包到最终资源里。比如在 Vue3 中,对于 options API

处理错误

执行用户提供的函数时,做统一的错误处理(try...catch),并提供给用户错误接口来处理。(可以错误上报等。)

TypeScript类型支持

第 3 章、Vue3 的设计思路

声明式的描述 UI

模板描述,或者虚拟DOM描述

渲染器

渲染器:把虚拟DOM变成真实DOM并渲染到浏览器页面。一个简单的渲染器实现:

点击查看代码
const vnode = {
tag: 'div',
props: {
onClick: () => alert('hello'),
},
children: 'click me',
}; function renderer(vnode, container) {
const el = document.createElement(vnode.tag);
for (const key in vnode.props) {
if (/^on/.test(key)) {
// on 开头是事件
el.addEventListener(
key.substr(2).toLowerCase(), // 事件名 onClick -> click
vnode.props[key] // 事件处理函数
);
}
}
// 处理 children
if (typeof vnode.children === 'string') {
el.appendChild(document.createTextNode(vnode.children));
} else if (Array.isArray(vnode.children)) {
vnode.children.forEach((child) => {
renderer(child, el);
});
} container.appendChild(el);
} renderer(vnode, document.body);

组件的本质

组件就是一组 DOM 元素的封装。上面的渲染器兼容组件(包括函数写法和对象写法):

点击查看代码
const MyComponent = function() {
return {
tag: 'div',
props: {
onClick: () => alert('hello'),
},
children: 'click me - MyComponent',
}
} const MyComponent1 = {
render() {
return {
tag: 'div',
props: {
onClick: () => alert('hello'),
},
children: 'click me - MyComponent1',
}
}
} const vnode = {
tag: MyComponent
}; function renderer(vnode, container) {
if (typeof vnode.tag === 'string') {
// 标签元素
mountElement(vnode, container);
} else if (typeof vnode.tag === 'function') {
// 函数描述组件
mountFunctionComponent(vnode, container);
} else if (typeof vnode.tag === 'object') {
// 对象描述组件
mountObjectComponent(vnode, container)
}
} function mountElement(vnode, container) {
const el = document.createElement(vnode.tag);
for (const key in vnode.props) {
if (/^on/.test(key)) {
// on 开头是事件
el.addEventListener(
key.substr(2).toLowerCase(), // 事件名 onClick -> click
vnode.props[key] // 事件处理函数
);
}
}
// 处理 children
if (typeof vnode.children === 'string') {
el.appendChild(document.createTextNode(vnode.children));
} else if (Array.isArray(vnode.children)) {
vnode.children.forEach((child) => {
renderer(child, el);
});
} container.appendChild(el);
} function mountFunctionComponent(vnode, container) {
// 获取 vnode
const subtree = vnode.tag();
renderer(subtree, container);
} function mountObjectComponent(vnode, container) {
// 获取 vnode
const subtree = vnode.tag.render();
renderer(subtree, container);
} renderer(vnode, document.body);

模板的工作原理

编译器:模板编译为渲染函数

<div @click="handler">
click me
</div>

编译为

render() {
return h('div', { onClick: handler }, 'click me')
}

Vue 是各模块组成的有机体

模板 --[编译器]--> 渲染函数 --[渲染器]--> 真实DOM

如果在编译器中增加优化,比如静态节点的判断,就可以在渲染器减少一些工作。

《Vue.js 设计与实现》读书笔记(1-3章)的更多相关文章

  1. Javascript高级程序设计读书笔记(第二章)

    第二章  在HTML中使用Javascript 2.1<script>元素 延迟脚本(defer = "defer")表明脚本在执行时不会影响页面的构造,脚本会被延迟到 ...

  2. javascript高级程序设计读书笔记-事件(一)

    读书笔记,写的很乱   事件处理程序   事件处理程序分为三种: 1.html事件2. DOM0级,3,DOM2级别  没有DOM1 同样的事件 DOM0会顶掉html事件   因为他们都是属性  而 ...

  3. Javascript高级程序设计读书笔记(第六章)

    第6章  面向对象的程序设计 6.2 创建对象 创建某个类的实例,必须使用new操作符调用构造函数会经历以下四个步骤: 创建一个新对象: 将构造函数的作用域赋给新对象: 执行构造函数中的代码: 返回新 ...

  4. JavaScript高级程序设计 读书笔记 第一章

    JavaScript是一种专门为与网页交互而设计的脚本语言 JavaScript实现 ECMAscript---核心 DOM---文档对象模型 BOM---浏览器对象模型

  5. Javascript高级程序设计读书笔记(第10章 DOM)

    第10章 DOM 10.1  节点层次 每个节点都有一个nodeType属性,用于表明节点的类型.任何节点类型必是下面中的一个: Node.Element_NODE(1); NODE.ATTRIBUT ...

  6. Javascript高级程序设计读书笔记(第三章)

    第3章 基本概念 3.4 数据类型 5种简单数据类型:undefined.boolean.number.null.string. typeof操作符,能返回下列某个字符串:“undefined”.“b ...

  7. JavaScript高级程序设计 读书笔记 第二章

    <script>元素 直接在页面中嵌入JavaSript代码或包含外部JavaSript文件. 在代码中任何地方不能出现</script>,可通过转义字符'\'解决. 在XHT ...

  8. javascript高级程序设计读书笔记

    第2章  在html中使用javascript 一般都会把js引用文件放在</body>前面,而不是放在<head>里, 目的是最后读取js文件以提高网页载入速度. 引用js文 ...

  9. JavaScript高级程序设计-读书笔记(7)

    第22章 高级技巧 1.高级函数 (1)安全的类型检测 在任何值上调用Object原生的toString()方法,都会返回一个[object NativeConstructorName]格式的字符串. ...

  10. JavaScript高级程序设计 读书笔记

    第一章 JavaScript 简介 第二章 Html中使用JavaScript 第三章 基本概念 第四章 变量,作用域,内存 第五章 引用类型 第六章 面向对象 第七章 函数表达式 第八章 BOM 第 ...

随机推荐

  1. 逆向动态加载Dex(内存加载class)

    逆向一个app, 其核心算法是通过反射调用的, 反编译软件中无法找到该类, 并且也无法hook. Java.perform(function(){ Java.enumerateClassLoaders ...

  2. 【教程】解决npm 报错 npm WARN config global `--global`, `--local` are deprecated. Use `--location=global` instead.

    问题描述 只要在控制台执行npm,不论有没有参数,都会有此警告: npm WARN config global `--global`, `--local` are deprecated. Use `- ...

  3. java一些位运算的方法

    记录一些java位运算的方法 Integer.bitCount:用于记录一个int输转化成二进制之后里面包含了多少个1. 使用例题为leetcode2859 切记下面规则要先转二进制 &相同位 ...

  4. 【Python】Django学习1

    按黑马程序员的美多商场作方向: https://www.bilibili.com/video/BV1nf4y1k7G3 一.应用创建.注册处理.配置 Pycharm 创建Django项目: 自应用注册 ...

  5. 【Canal】01 入门 & Kafka模式

    什么是Canal (卡耐尔) ? Canal 是用 Java 开发的基于数据库增量日志解析,提供增量数据订阅&消费的中间件 原理基于MySQL的binlog从库监听       一.MySQL ...

  6. 【Win10】找不到Telnet命令

    百度才发现原来我的Telnet程序就没启动... CMD终端就找不到命令... 但是有一点要注意!不能使用CMD之外的终端访问[GitBash访问无效] 打开控制面板,点卸载程序: 然后点击启动和关闭 ...

  7. TensorBoard标量图中的平滑曲线是如何做的平滑?—— tensorflow TensorBoard标量图中“平滑”参数背后的数学原理是什么?—— 指数移动平均(EMA)

    TensorFlow的tensorboard的平滑曲线的实现代码: 使用"指数移动平均"技术实现. 地址: https://github.com/tensorflow/tensor ...

  8. 【转载】 图解协程调度模型-GMP模型

    原文地址: https://www.cnblogs.com/codexiaoyi/p/14975236.html =========================================== ...

  9. [USACO22FEB] Paint by Rectangles P 题解

    前言 没用线段树的小常数.小短码. 题目链接:洛谷. 题意简述 给出 \(n\) 个平行于坐标轴的矩形,各边所在直线互不重合,钦定最外面为白色,对这个平面图黑白染色,分别求黑色块数和白色块数. 题目分 ...

  10. .NET 开源权限认证项目 MiniAuth上线

    前言 在Web应用项目中权限认证是个绕不开的话题,传统方法复杂又耗时.MiniAuth推出专为.NET开发者设计的简单.实用的权限认证项目. MiniAuth,作为ASP.NET Core的插件,让我 ...