最后
全网独播-价值千万金融项目前端架构实战
从两道网易面试题-分析JavaScript底层机制
RESTful架构在Nodejs下的最佳实践
开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】
一线互联网企业如何初始化项目-做一个自己的vue-cli
思维无价,看我用Nodejs实现MVC
代码优雅的秘诀-用观察者模式深度解耦模块
前端高级实战,如何封装属于自己的JS库
VUE组件库级组件封装-高复用弹窗组件
const util = {
sum(a,b){
return a + b;
}
}
我可以使用 export 将这个对象导出:
const util = {
sum(a,b){
return a + b;
}
}
export {util};
当然,也可以简写为:
export const util = {
sum(a,b){
return a + b;
}
}
export
不仅可以导出对象,一切 JS 变量都可以导出。比如:基本类型变量、函数、数组、对象。
当要导出多个值时,还可以简写。比如我有一个文件:user.js:
var name = “jack”
var age = 21
export {name,age}
省略名称
上面的导出代码中,都明确指定了导出的变量名,这样其它人在导入使用时就必须准确写出 变量名,否则就会出错。
因此 js 提供了default
关键字,可以对导出的变量名进行省略
例如:
// 无需声明对象的名字
export default {
sum(a,b){
return a + b;
}
}
这样,当使用者导入时,可以任意起名字
- import
使用export
命令定义了模块的对外接口以后,其他 JS 文件就可以通过import
命令加载这个模块。
例如我要使用上面导出的 util:
// 导入 util
import util from ‘hello.js’
// 调用 util 中的属性
util.sum(1,2)
要批量导入前面导出的 name 和 age:
import {name, age} from ‘user.js’
console.log(name + " , 今年"+ age +“岁了”)
但是上面的代码暂时无法测试,因为浏览器目前还不支持 ES6 的导入和导出功能。除非借助于工具,把 ES6 的语法进行编译降级到 ES5,比如Babel-cli
工具
我们暂时不做测试,大家了解即可。
====================================================================
前端开发,少不了 node.js;Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行环境。
我们关注与 node.js 的 npm 功能就行;
NPM 是随同 NodeJS 一起安装的包管理工具,JavaScript-NPM,Java-Maven;
-
官网下载安装 node.js,并使用 node -v 检查版本
-
配置 npm 使用淘宝镜像
npm config set registry http://registry.npm.taobao.org/
- 大家如果 npm install 安装依赖出现 chromedriver 之类问题,先在项目里运行下面命令
npm install chromedriver --chromedriver_cdnurl=http://cdn.npm.taobao.org/dist/chromedriver
然后再运行 npm install
================================================================
-
M:即 Model,模型,包括数据和一些基本操作
-
V:即 View,视图,页面渲染结果
-
VM:即 View-Model,模型与视图间的双向操作(无需开发人员干涉)
在 MVVM 之前,开发人员从后端获取需要的数据模型,然后要通过 DOM 操作 Model 渲染到 View 中。而后当用户操作视图,我们还需要通过 DOM 获取 View 中的数据,然后同步到Model 中。
在 MVVM 之前,开发人员从后端获取需要的数据模型,然后要通过 DOM 操作 Model 渲染到 View 中。而后当用户操作视图,我们还需要通过 DOM 获取 View 中的数据,然后同步到Model 中。
而 MVVM 中的 VM 要做的事情就是把 DOM 操作完全封装起来,开发人员不用再关心 Model
和 View 之间是如何互相影响的:
-
只要我们 Model 发生了改变,View 上自然就会表现出来。
-
当用户修改了 View,Model 中的数据也会跟着改变。
把开发人员从繁琐的 DOM 操作中解放出来,把关注点放在如何操作 Model 上。
Vue (读音 /vjuː/,类似于 view) 是一套用于构建用户界面的渐进式框架。与其它大型框架不同的是,Vue 被设计为可以自底向上逐层应用。Vue 的核心库只关注视图层,不仅易于上手,还便于与第三方库或既有项目整合。另一方面,当与现代化的工具链以及各种支持类库 结合使用时,Vue 也完全能够为复杂的单页应用提供驱动。
参考:https://cn.vuejs.org/v2/guide/
Git 地址:https://github.com/vuejs
尤雨溪,Vue.js 创作者,Vue Technology 创始人,致力于 Vue 的研究开发。
3.1. 安装
官网文档提供了 3 中安装方式:
-
直接 script 引入本地 vue 文件。需要通过官网下载 vue 文件。
-
通过 script 引入 CDN 代理。需要联网,生产环境可以使用这种方式
-
通过 npm 安装。这种方式也是官网推荐的方式,需要 nodejs 环境。
本课程就采用第三种方式
3.2. 创建示例项目
-
新建文件夹 hello-vue,并使用 vscode 打开
-
使用 vscode 控制台,
npm install -y
项目会生成 package-lock.json 文件,类似于 maven 项目的 pom.xml 文件。
- 使用
npm install vue
给项目安装 vue;项目下会多 node_modules 目录,并且在下面有一个 vue 目录。
3.3. HelloWorld
在 hello.html 中,我们编写一段简单的代码。
h2 中要输出一句话:xx 非常帅
。前面的xx
是要渲染的数据。
3.4. Vue声明式渲染
页面代码
{{name}},非常帅!!!
-
首先通过 new Vue()来创建 Vue 实例
-
然后构造函数接收一个对象,对象中有一些属性:
-
el
:是 element 的缩写,通过 id 选中要渲染的页面元素,本例中是一个 div -
data:数据,数据是一个对象,里面有很多属性,都可以渲染到视图中
name
:这里我们指定了一个 name 属性
- 页面中的
h2
元素中,我们通过{{name}}的方式,来渲染刚刚定义的 name 属性。
打开页面查看效果:
更神奇的在于,当你修改 name 属性时,页面会跟着变化:
3.5. 双向绑定
我们对刚才的案例进行简单修改:
{{name}},非常帅!!!有{{num}}个人为他点赞。
双向绑定:
效果:我们修改表单项,num 会发生变化。我们修改 num,表单项也会发生变化。为了实时观察到这个变化,我们将 num 输出到页面。
我们不需要关注他们为什么会建立起来关联,以及页面如何变化,我们只需要做好数据和视图的关联即可(MVVM)
3.6. 事件处理
给页面添加一个按钮:
关注
{{name}},非常帅!!!有{{num}}个人为他点赞。
-
这里用
v-on
指令绑定点击事件,而不是普通的onclick
,然后直接操作 num -
普通 click 是无法直接操作 num 的。
-
未来我们会见到更多 v-xxx,这些都是 vue 定义的不同功能的指令。
简单使用总结:
-
使用 Vue 实例管理 DOM
-
DOM 与数据/事件等进行相关绑定
-
我们只需要关注数据,事件等处理,无需关心视图如何进行修改
4.1. 创建 Vue 实例
每个 Vue 应用都是通过用 Vue 函数创建一个新的 Vue 实例开始的:
let app = new Vue({
···
});
在构造函数中传入一个对象,并且在对象中声明各种 Vue 需要的数据和方法,包括:
-
el
-
data
-
methods
等等
接下来我们一 一介绍。
4.2. 模板或元素
每个 Vue 实例都需要关联一段 HTML 模板,Vue 会基于此模板进行视图渲染。
我们可以通过 el 属性来指定。
例如一段 HTML模板:
···
然后创建 Vue 实例,关联这个 div
let vm = new Vue(
{
el: “#app”
这样,Vue 就可以基于 id 为app
的 div 元素作为模板进行渲染了。在这个 div 范围以外的部分是无法使用 vue 特性的。
4.3. 数据
当 Vue 实例被创建时,它会尝试获取在 data 中定义的所有属性,用于视图的渲染,并且监视 data 中的属性变化,当 data 发生改变,所有相关的视图都将重新渲染,这就是“响应式“系统。
HTML:
JS:
let vm = new Vue(
{
el: “#app”,
data: {
}
})
-
name 的变化会影响到
input
的值 -
input 中输入的值,也会导致 vm 中的 name 发生改变
4.4. 方法
Vue 实例中除了可以定义 data 属性,也可以定义方法,并且在 Vue 实例的作用范围内使用。
HTML:
{{num}}
加
JS:
let vm = new Vue(
{
el: “#app”,
data: {
num: 0
},
methods: {
add: function () {
// this 代表的当前 vue 实例
this.num++;
}
}
})
4.5. 安装 vue-devtools 方便调试
-
将软件包中的 vue-devtools 解压。
-
打开 chrome 设置->扩展程序
- 开启开发者模式,并加载插件
- 打开浏览器控制台,选择 vue
4.6. 安装 vscode 的 vue 插件
安装这个插件就可以有语法提示
什么是指令?
-
指令 (Directives) 是带有
v-
前缀的特殊特性。 -
指令特性的预期值是:单个 JavaScript 表达式。
-
指令的职责是,当表达式的值改变时,将其产生的连带影响,响应式地作用于 DOM。
例如我们在入门案例中的 v-on,代表绑定事件。
5.1. 插值表达式
- 花括号
格式:{{表达式}}
说明:
-
该表达式支持 JS 语法,可以调用 js 内置函数(必须有返回值)
-
表达式必须有返回结果。例如 1 + 1,没有结果的表达式不允许使用,如:let a = 1 + 1;
-
可以直接获取 Vue 实例中定义的数据或函数
- 插值闪烁
使用{{}}方式在网速较慢时会出现问题。在数据未加载完成时,页面会显示出原始的{{}}
, 加载完毕后才显示正确数据,我们称为插值闪烁。
我们将网速调慢一些,然后刷新页面,试试看刚才的案例:
- v-text 和v-html
可以使用 v-text 和 v-html 指令来替代{{}} 说明:
-
v-text:将数据输出到元素内部,如果输出的数据有 HTML 代码,会作为普通文本输出
-
v-html:将数据输出到元素内部,如果输出的数据有 HTML 代码,会被渲染
示例:
v-text:
v-html:
效果:
并且不会出现插值闪烁,当没有数据时,会显示空白或者默认数据。
5.2. v-bind
html 属性不能使用双大括号形式绑定,我们使用 v-bind 指令给 HTML 标签属性绑定值; 而且在将v-bind
用于class
和style
时,Vue.js 做了专门的增强。
- 绑定class
- 绑定style
v-bind:style
的对象语法十分直观,看着非常像 CSS,但其实是一个 JavaScript 对象。style 属性名可以用驼峰式 (camelCase) 或短横线分隔 (kebab-case,这种方式记得用单引号括起来) 来命名。
例如:font-size–>fontSize
结果:<div style=“color: red; font-size: 30px;”>
- 绑定其他任意属性
v-bind:user=“userName”>
效果: <div id=“app” user=“zhangsan” style=“color: red; font-size: 30px;”>
v-bind
缩写
:user=“userName”>
5.3. v-model
刚才的 v-text、v-html、v-bind 可以看做是单向绑定,数据影响了视图渲染,但是反过来就不行。接下来学习的 v-model 是双向绑定,视图(View)和模型(Model)之间会互相影响。
既然是双向绑定,定是在视图中可以修改数据,这样就限定了视图的元素类型。目前
v-model 的可使用元素有:
-
input
-
select
-
textarea
-
checkbox
-
radio
-
components(Vue 中的自定义组件)
基本上除了最后一项,其它都是表单的输入项。
示例:
Java
PHP
Swift
你选择了:{{language.join(‘,’)}}
-
多个
CheckBox
对应一个 model 时,model 的类型是一个数组,单个 checkbox 值默认是boolean 类型 -
radio 对应的值是 input 的 value 值
-
text
和textarea
默认对应的 model 是字符串 -
select
单选对应字符串,多选对应也是数组
效果:
5.4. v-on
- 基本用法
v-on 指令用于给页面元素绑定事件。
语法: v-on:事件名="js 片段或函数名"
示例:
点赞
取消
有{{num}}个赞
另外,事件绑定可以简写,例如v-on:click='add'
可以简写为@click='add'
- 事件修饰符
在事件处理程序中调用 event.preventDefault()
或 event.stopPropagation()
是非常常见的需求。尽管我们可以在方法中轻松实现这点,但更好的方式是:方法只有纯粹的数据逻辑, 而不是去处理 DOM 事件细节。
为了解决这个问题,Vue.js 为 v-on
提供了事件修饰符。修饰符是由点开头的指令后缀来表示的。
-
.stop
:阻止事件冒泡到父元素 -
.prevent
:阻止默认事件发生
-
.capture
:使用事件捕获模式 -
.self
:只有元素自身触发事件才执行。(冒泡或捕获的都不执行) -
.once
:只执行一次
取消
有{{num}}个赞
效果:右键“点赞”,不会触发默认的浏览器右击事件;右键“取消”,会触发默认的浏览器右击事件)
- 按键修饰符
在监听键盘事件时,我们经常需要检查常见的键值。Vue 允许为v-on
在监听键盘事件时添加按键修饰符:
记住所有的 keyCode
比较困难,所以 Vue 为最常用的按键提供了别名:
<input @keyup.enter=“submit”>
全部的按键别名:
-
.enter
-
.tab
-
.delete
(捕获“删除”和“退格”键)` -
.esc
-
``.space`
-
.up
-
.down
-
.left
-
.right
- 组合按钮
可以用如下修饰符来实现仅在按下相应按键时才触发鼠标或键盘事件的监听器。
-
.ctrl
-
.alt
-
``.shift`
<input @keyup.alt.67=“clear”>
5.5. v-for
遍历数据渲染页面是非常常用的需求,Vue 中通过 v-for 指令来实现。
- 遍历数组
语法:v-for="item in items"
-
items:要遍历的数组,需要在 vue 的 data 中定义好。
-
item:迭代得到的当前正在遍历的元素
示例:
{{user.name}} - {{user.gender}} - {{user.age}}
效果:
- 遍历角标
在遍历的过程中,如果我们需要知道数组角标,可以指定第二个参数: 语法:v-for=“(item,index) in items”
-
items:要迭代的数组
-
item:迭代得到的数组元素别名
-
index:迭代到的当前元素索引,从 0 开始。
示例:
{{index + 1}}. {{user.name}} - {{user.gender}} - {{user.age}}
效果:
- 遍历对象
v-for 除了可以迭代数组,也可以迭代对象。语法基本类似语法:
- v-for=“value in object”
1 个参数时,得到的是对象的属性值
- v-for=“(value,key) in object”
2 个参数时,第一个是属性值,第二个是属性名
- v-for=“(value,key,index) in object”
3 个参数时,第三个是索引,从 0 开始
示例:
{{index + 1}}. {{key}} - {{value}}
效果:
- Key
用来标识每一个元素的唯一特征,这样 Vue 可以使用“就地复用”策略有效的提高渲染的效率。
示例:
最佳实践:
如果 items 是数组,可以使用 index 作为每个元素的唯一标识
如果 items 是对象数组,可以使用 item.id 作为每个元素的唯一标识
5.6. v-if和v-show
- 基本用法
v-if,顾名思义,条件判断。当得到结果为 true 时,所在的元素才会被渲染。
v-show,当得到结果为 true 时,所在的元素才会被显示。
语法:
v-if="布尔表达式", v-show="布尔表达式",
示例:
点我呀
看到我啦?!
看到我啦?!show
- 与v-for结合
当 v-if 和 v-for 出现在一起时,v-for 优先级更高。也就是说,会先遍历,再判断条件。修改 v-for 中的案例,添加 v-if:
{{index + 1}}. {{user.name}} - {{user.gender}} - {{user.age}}
效果:只显示女性
5.7. v-else 和 v-else-if
else 元素必须紧跟在带
v-if
或者v-else-if
的元素的后面,否则它将不会被识别。示例:
点我呀
{{random}}
看到我啦?!v-if >= 0.75
看到我啦?!v-else-if > 0.5
看到我啦?!v-else-if > 0.25
看到我啦?!v-else
6.1. 计算属性(computed)
某些结果是基于之前数据实时计算出来的,我们可以利用计算属性来完成
示例:
- 西游记:价格{{xyjPrice}},数量:
- 水浒传:价格{{shzPrice}},数量:
- 总价:{{totalPrice}}
效果:只要依赖的属性发生变化,就会重新计算这个属性
6.2. 侦听(watch)
watch 可以让我们监控一个值的变化。从而做出相应的反应。
示例:
- 西游记:价格{{xyjPrice}},数量:
- 水浒传:价格{{shzPrice}},数量:
- 总价:{{totalPrice}}
{{msg}}
效果:
6.3. 过滤器(filters)
过滤器不改变真正的
data
,而只是改变渲染的结果,并返回过滤后的版本。在很多不同的情况下,过滤器都是有用的,比如尽可能保持 API 响应的干净,并在前端处理数据的格式。示例:展示用户列表性别显示男女
{{user.id}} {{user.name}} {{user.gender===1? "男":"女"}}
- 局部过滤器
注册在当前 vue 实例中,只有当前实例能用
let app = new Vue({
el: “#app”,
data: {
userList: [
{ id: 1, name: ‘jacky’, gender: 1 },
{ id: 2, name: ‘peter’, gender: 0 }
]
},
// filters 定义局部过滤器,只可以在当前 vue 实例中使用
filters: {
genderFilter(gender) {
return gender === 1 ? ‘男~’ : ‘女~’
}
}
});
{{user.gender | genderFilter}}- 全局过滤器
// 在创建 Vue 实例之前全局定义过滤器:
Vue.filter(‘capitalize’, function (value) {
return value.charAt(0).toUpperCase() + value.slice(1)
})
任何 vue 实例都可以使用:{{user.name | capitalize}}
过滤器常用来处理文本格式化的操作。过滤器可以用在两个地方:双花括号插值和 v-bind表达式
在大型应用开发的时候,页面可以划分成很多部分。往往不同的页面,也会有相同的部分。 例如可能会有相同的头部导航。
但是如果每个页面都独自开发,这无疑增加了我们开发的成本。所以我们会把页面的不同部 分拆分成独立的组件,然后在不同页面就可以共享这些组件,避免重复开发。
在 vue 里,所有的 vue 实例都是组件
7.1. 全局组件
我们通过 Vue 的 component 方法来定义一个全局组件。
-
组件其实也是一个 Vue 实例,因此它在定义时也会接收:data、methods、生命周期函数等
-
不同的是组件不会与页面的元素绑定,否则就无法复用了,因此没有 el 属性。
-
但是组件渲染需要 html 模板,所以增加了 template 属性,值就是 HTML 模板
-
全局组件定义完毕,任何 vue 实例都可以直接在 HTML 中通过组件名称来使用组件了
-
data 必须是一个函数,不再是一个对象。
7.2. 组件的使用
定义好的组件,可以任意复用多次:.
组件的 data 属性必须是函数!
一个组件的 data 选项必须是一个函数,因此每个实例可以维护一份被返回对象的独立的拷贝;
否则:data-必须是一个函数
7.3. 局部组件
一旦全局注册,就意味着即便以后你不再使用这个组件,它依然会随着 Vue 的加载而加载。因此,对于一些并不频繁使用的组件,我们会采用局部注册。
我们先在外部定义一个对象,结构与创建组件时传递的第二个参数一致:
const counter = {
template: ‘你点了我 {{ count }} 次,我记住了.’,
data() {
return {
count: 0
}
}
};
然后在 Vue 中使用它:
let app = new Vue({
el: “#app”,
components: {
counter: counter // 将定义的对象注册为组件
}
-
components 就是当前 vue 对象子组件集合。
-
其 key 就是子组件名称
-
其值就是组件对象名
-
效果与刚才的全局注册是类似的,不同的是,这个 counter 组件只能在当前的 Vue 实例中使用
简写:
let app = new Vue({
el: “#app”,
components: {
counter // 将定义的对象注册为组件
}
})
8.1. 生命周期
每个 Vue 实例在被创建时都要经过一系列的初始化过程 :创建实例,装载模板,渲染模板等等。Vue 为生命周期中的每个状态都设置了钩子函数(监听函数)。每当 Vue 实例处于不同的生命周期时,对应的函数就会被触发调用。
生命周期:你不需要立马弄明白所有的东西。
8.2. 钩子函数
-
beforeCreated:我们在用 Vue 时都要进行实例化,因此,该函数就是在 Vue 实例化时调用,也可以将他理解为初始化函数比较方便一点,在 Vue1.0 时,这个函数的名字就是init。
-
created:在创建实例之后进行调用。
-
beforeMount:页面加载完成,没有渲染。如:此时页面还是{{name}}
-
mounted:我们可以将他理解为原生 js 中的 window.οnlοad=function({.,.}),或许大家也在用 jquery,所以也可以理解为 jquery 中的$(document).ready(function(){….}),他的功能就是: 在 dom 文档渲染完毕之后将要执行的函数, 该函数在 Vue1.0 版本中名字为compiled。 此时页面中的{{name}}已被渲染成张三
-
beforeDestroy:该函数将在销毁实例前进行调用 。
-
destroyed:改函数将在销毁实例时进行调用。
-
beforeUpdate:组件更新之前。
-
updated:组件更新之后。
示例:
{{num}}
赞!
{{name}},非常帅!!!有{{num}}个人点赞。