vue就是一个当下最为主流的前端框架,用于构建用户界面的渐进式javascript框架
渐进式:先完成最基本的功能 开发应用 但是如果想开发复杂功能 那么就可以引入各种vue插件来完成(vue是一个自底向上逐层开发的一个框架)
尤雨溪
1、组件化。实现了封装和重用,且组件间可以相互嵌套; 2、轻量级。相对于其他框架,Vue学习成本低,简单易上手; 3、虚拟 DOM。虚拟 dom 中存在 diff算法,是 cpu 密集型运算,占用内存较少,可以提高运行效率,并压缩运行时体积; 4、Vue是一个MVVM框架,数据绑定。数据单项绑定(v-bind),单向指data中的值绑定到视图中,但视图中修改不会影响到data数据;双向绑定(v-model),数据发生变化会驱动视图的更新,视图的更新也会驱动数据的变化; 5、单页面应用(SPA)。用户体验好,内容改变时不需要重新加载整个页面,不会进行多个 html 页面间的切换;服务器压力小等。
MVVM是Model-View-ViewModel的简写
M --- model 模型==数据==变量 (只要今后听见了M 或者是模型 就直接想象成 data数据·) data数据·
V --- view 视图==页面==用户可以看见的界面 模板
VM --- viewModel 视图模型==用来关联数据与视图之前的桥梁 vue实例
Data Bindings 数据绑定 (按照上图描述 就是把model数据 ----》数据绑定 ----》绑定到页面上)
DOM Listeners dom监听 (按照上图描述 就是view页面的数据改变-----》dom监听------》反应到数据上)
安装淘宝镜像( npm服务器在国外可能会导致网速很慢导致下载失败,可以使用淘宝镜像 当然不是必须要安装 ---- 如果安装了今后所有使用npm 在下载内容的时候 就要把npm 变成 cnpm 剩下都一样)
npm install cnpm -g --registry=https://registry.npm.taobao.org
检查是否安装成功
cnpm -v
安装成功显示如图下:
1.下载vue库文件 npm init -y (初始化) npm install --save vue@2 简写 : npm i -S vue@2(下载)
2.新建html页面先把vue引用进来
3.编写如下代码
Document {{text}}
{{num}}
{{obj.name}}
{{arr[4]}}
多个vue实例生效的就是对应的 视图
Document {{text}}
{{text}}
在vue中{{}}被称之为模板语法 双花括号赋值法 vue数据插值 。。。。 作用:就是可以在双大括号中写入表达式 并且展示在页面中
语法:在你想展示数据的任何位置 {{表达式}} (表达式 通过计算可以返回结果的公式)
Document {{num}}
{{num*2}}
{{bool?"你好":"你坏"}}
{{text.toUpperCase().slice(1,5)}}
Vue 的$mount()为手动挂载,在项目中可用于延时挂载(比如vue是一个渐进式的 在我们使用一些负载功能的时候 我们需要依赖很多后面会学到的vue插件 来完成 但是使用这些插件 我们必须先要把这些插件先挂载之后 在加载视图的关联 所以在这个时候我们会用到延迟健在 ),之后要手动挂载上。new Vue时,el和$mount并没有本质上的不同。
Document 两种挂载方式
{{text}}
Document 两种挂载方式
{{text}}
通过写在html开标签中的 使用属性="属性值" 的这些东西 可以扩展标签的功能
就是在vue中给html标签添加的带有v-前缀的特殊属性(在vue中 给html标签添加个一些特殊性功能属性)
作用 : 就是给表单元素进行数据的双向绑定
双向绑定
视图改变模型也会改变
模型变视图也会随之改变
语法: <标签 v-model=""值/>
Document v-model
{{inputval}}
双向绑定的原理
在vue中基于数据劫持--数据代理与发布者订阅者模式完成的
数据劫持:数据拦截 就是对data中的数据在初始化的时候监听起来(Object.defineProperty 来进行监听/代理) 当数据改变setter之后 vm就会知道 在视图改变getter 他就会通知模型你要修改了 模型改变了也会通知视图改变
发布者订阅者模式:就是一个一对多的关系 发布者就是数据提供者 订阅者就是页面展示的 一个发布者可以对应无数个订阅者 但是发布者改变了 所有订阅者也会改变
作用:控制dom元素的显示或者隐藏 v-show的显示和隐藏是通过css的dispaly方式来控制的
语法 v-show="布尔值" true显示 false隐藏
Document {{bool?"勾选了":"没勾选"}}我是站位的
作用:就是给vue的dom绑定事件的
语法:
传统写法(写法稍微麻烦点): v-on:你的事件不加on="函数" v-on:click=”函数“
简写写法: @你的事件不加on="函数" @click="函数"
那今后工作用哪个 简写的函数 传统的 (想用那个用哪个)
注意:函数需要写在 与el data同级位置使用methods来进行包裹
Document
注意 函数中怎么使用data数据 使用this.变量名(this指向的就是vue实例)
methods:{ fun(){ // console.log("你好") console.log(this.text) console.log(this._data.text); }, }
概念: 用来遍历数据 生成页面内容
语法: v-for="(遍历的值,遍历的下标) in 你要遍历的数据"
Document
- {{v}}------{{i}}
遍历复杂数据
Document
- {{v}}------{{i}}
{{v.name}} {{v.age}}
key属性
可以在是用v-for的时候搭配使用 他的作用就是给便利出来的dom 起个唯一的名字 相当于我们的身份证号 通过添加了这个key属性可以增加我们在便利展示的时候生成dom元素的效率 (key里面是唯一的不建议使用我们便利出来的下标 因为如果一个页面有两次便利的话key的值就有可能相同)
概念: 就是html的属性插入变量
语法:
传统写法: v-bind:html的属性="值"
简写写法: :html的属性="值"
Document
v-if
作用:就是对dom元素进行移除和添加
语法: 写在开标签中 true添加 false移除
Document v-if
{{bool}}我是v-show的标签
我是v-if的标签
v-if与v-show的区别
1.v-show是使用css的方式对dom元素进行显示和隐藏的 在频繁切换的时候效率更高 在初始化的时候对性能的损耗比较高
2.v-if是直接把这个dom元素移除或者是添加 在频繁切换的时候效率比较低 在初始化的时候对性能的损耗比较低
v-else
语法: 必须配合v-if使用 不能单独使用
Document v-if
{{bool}}请您登录
欢迎您尊敬的vip
v-else-if
Document v-else-if
我是吃饭的dom
我是睡觉的dom
我是在吃饭的dom我是在睡觉的dom
我什么都没有干
v-if与v-for能同时使用吗
当 v-if
与 v-for
不推荐同时使用,v-for
具有比 v-if
更高的优先级,这意味着 v-if
将分别重复运行于每个 v-for
循环中,影响速度
如果必须要使用的话 变成一个嵌套关系
解决方式
1.使用computed处理数据在便利
2.使用v-show替代v-if
作用:就是把字符串标签插入到页面中
Document v-html
{{text}}
作用: 一次性插值 数据插入到页面中就不会被改变了
Document v-once
{{text}}
{{text}}
{{text}}
{{text}}
{{text}}
{{text}}
{{text}}
{{text}}
{{text}}
作用:向dom中插入文本内容 和{{}}作用一样
Document v-text
{{num}}
屏幕闪烁
当用户的设备和网络比较慢的时候可能就会在页面中吧{{}}全部展现出来 当网络恢复正常之后 有突然间显示ok
使用{{}}模板语法的话就会出现这个问题
解决方式
1.使用v-text (全部在页面中使用v-text指令很麻烦)
2.使用v-cloak指令 等待vue实例渲染完成之后在进行(这个指令是防止页面加载时出现 vuejs 的变量名而设计)
Document v-text
{{num}}
谁触发这个事件事件对象就指向谁 (事件对象中包含触发这个事件的元素所有信息)类似于拖拽
Document
通过vue提供的修饰符可以来处理dom事件的一些细节性的内容
v-on:事件.修饰符="函数()"
优化我们对于键盘事件的相关处理
.up .down .left .right .ctrl .space .enter .tab .delete .shift .esc
Document
.stop修饰符 阻止事件传播
.prevent修饰符 阻止事件默认行为
.captrue修饰符 设置捕获
.self 修饰符 只会触发自己范围内的事件 不包含子元素
.once修饰符 只触发当前事件一次
Document
用户使用v-model之后,用户每次修改输入内容,都会将数据同时绑定,为了避免这种情况的发生,使用lazy修饰符来进行限定。只有当用户的input中失去焦点或者用户点击回车按钮时,才会将的数据进行修改。
Document {{inputval}}
当用户在input中输入数字时,浏览器会默认将输入的数字转化为string类型,使用number修饰符来将输入的数字转为number类型
Document 没有使用number
使用了number
使用trim修饰符来去掉字符串首部或者尾部的所有空格
Document 基本展示:{{text}}
大写:{{text.toUpperCase()}}
大写截取:{{text.toUpperCase().slice(1,4)}}
上面的代码没有问题但是后期可能会导致代码的可读性非常差
所以在我们遇到了一条数据 在不同位置 展示出不同形态的时候 我们可以使用计算属性
在弄懂什么是计算属性之前 思考一个问题 什么是属性
什么是计算属性?
就是把属性通过加工通过计算返回一个新的结果那么这个就是计算属性
那么计算属性写在哪里 并且怎么写呢?
写在与el data methods 同级位置 使用 computed
其中有两个方法 get与set
get方法:必须要写,该函数不接受参数,当初次读取计算属性或者计算属性所依赖的数据发生变化时被调用,getter函数有一个返回值,该返回值就是计算属性的值
computed:{ 计算出存储结果的名字:{ //必须要写,该函数不接受参数 //什么时候被调用?:当初次读取计算属性或者计算属性所依赖的数据发生变化时被调用,getter函数有一个返回值,该返回值就是计算属性的值 get(){ return 你的计算逻辑 }, //可选项 接受一个可选参数(计算属性被修改之后的值) //什么时候被调用?: 当计算属性被修改时被调用 set(value){ } } }
读取
Document 读取data中的数据--{{text}}
读取计算属性的数据--{{newtext}}
修改
set方法:可选项 接受一个可选参数(计算属性被修改之后的值)当计算属性的值被修改时被调用
Document 读取data中的数据--{{text}}
读取计算属性的数据--{{newtext}}
通常我们的计算属性都是把计算出来的结果展示到页面上 set这种修改使用的非常少 所以在计算属性中也给我们提供了一种读取的简写语法
computed:{ 你处理好的变量(){ return 你的处理逻辑 } }
Document 读取data中的数据--{{text}}
读取计算属性的数据--{{newtext}}
计算属性是依赖data里面的属性的 当计算属性依赖的data数据改变了 那么计算属性也会收到通知 做出相关的计算返回新的结果
计算属性处理的数据 如果被多次调用的时候 计算属性只执行一次(因为计算属性在第一次处理好数据之后 就会把数据放到缓存中 之后的每次读取都是从缓存中读取的 所以 多次调用他只执行一次)总结一下就是计算属性是依赖缓存的
方法 他是一个憨憨 只要被调用就会执行 那么相对于计算属性依赖缓存而言 他更消耗内存
watch是vue实例的一个属性 他的作用就是用来监听data中的数据 当数据变了watch就会触发 从而调用函数处理一些逻辑
语法:写在与el data methods 同级位置
watch:{ 你要监听的data数据(newval,oldval){ } }
Document watch
{{text}}
不会触发
watch 的一个特点是,最初绑定的时候是不会执行的,要等到 监听的数据 改变时才执行监听。那我们想要一开始就让他最初绑定的时候就执行改怎么办呢?
data监听的数据改变触发的回调函数
//text变量改变的时候。handler方法就会触发 watch:{ text:{ handler(newval,oldval){ console.log("aaaa"); }, }, }
watch默认绑定,页面首次加载时,是不会执行的。只有值发生改变才会执行。
设置immediate为true后,监听会在被监听值初始化的时候就开始,也就页面上的数据还未变化的时候。
watch:{ text:{ handler(){ console.log("aaaa"); }, immediate:true //true就表示会立即执行 }, }
属性 deep,默认值是 false,代表是否深度监听
观察下面的代码
当我们在在输入框中输入数据视图改变obj.name的值时,我们发现是无效的。
watch在监听对象的时候受现代 JavaScript 的限制. Vue 不能检测到对象属性的添加或删除。
如果我们需要监听obj里的属性name的值呢?
{{obj.name}}
这时候deep属性就派上用场了。
deep的意思就是深入观察,监听器会一层层的往下遍历,给对象的所有属性都加上这个监听器,但是这样性能开销就会非常大了,任何修改obj里面任何一个属性都会触发这个监听器里的 handler。
{{obj.name}}
使用了deep之后 他会把对象中的所有属性都监听到不管你用不用 反正都会给你监听 那么这样性能开销就会非常大了
deep优化
优化,我们可以是使用字符串形式监听
{{obj.name}}
如果进行被问到了watch任何知识点?
首先watch的作用就是监听data数据 当data的数据改变了 watch就会触发一个回调函数完成对应的逻辑操作。
其次watch有两种写法 第一种简写 使用监听的数据对应一个函数 里面有两个形参newval和oldval
第二种就是传统写法 监听的数据里面 有一个handler方法 这个方法就是被监听的数据被触发之后 调用的回调函数 当然里面也有两个形参newval和oldval 其次里面还有一个属性immediate属性 他的作用是watch'首次加载不会触发 如果想让watch首次加载/初始化监听 那么就可以设置这个属性为true 开启初始化监听 还有就是 vue不能检测到对象的属性操作 所以我们如果要监听对象的属性 那么必须使用deep开启深度监听 但是开启deep之后还有问题 就是deep会把监听的对象所有属性都监听上 不管用不用都监听 浪费资源 所以我们也可以使用字符串的方式来监听对象的属性 从而优化性能
计算属性 : 计算属性是依赖data的数据 当data的数据改变之后 计算属性会重新计算返回一个新的结果
watch: watch是监听data的数据 当data的数据改变之后 watch会调用一个回调函数完成一些特定的逻辑
计算属性是同步的 watch是异步的
计算属性(computed) | 属性检测(watch) |
---|---|
首次运行 | 首次不运行 |
默认深度依赖 | 默认浅度观测 |
适合做筛选,不可异步 | 适合做执行异步或开销较大的操作 |
在不改变原始数据的情况下格式化展示内容
我们今后的数据都是从后台请求过来的--但是有的时候后台给我们的数据网王和我们想展示的结果不太一样
过滤器的地方有很多,比如单位转换、数字打点 59,999、文本格式化 斯大林格日勒保卫战 大于5个字 截取并且在后面加...、时间格式化 2022825 2022-8-25
在vue2x中 vue已经取消了内置过滤器 在vue1x中有内置和自定义过滤器
在面试的时候有很多挖坑事提问 很可能人家说 你给我简述下vue的内置过滤器?
你的想法是老师那货没有教我 咋办呀?
你的回答是 您刚才提到的是vue的内置过滤器 在我所知 vue在1x中是有内置过滤器的 但是我学习vue的时候1x已经淘汰了 我是从2x开始学习的 所以这个内置过滤器没有太多的了解 但是如果贵公司需要 我也可以第一时间掌握 因为就是一个小小的过滤没有什么复杂的
在所有vue实例组件中都可以使用
定义
使用filter()来定义全局过滤器
写在vue实例创建之前
语法:
Vue.filter("过滤器的名字",(你要过滤的数据)=>{return 逻辑})
Document {{text|xiaoming}}
{{textb|xiaoming}}
使用
在想使用的地方 使用| 来完成
{{变量|过滤器的名字}}
仅仅只能在指定的实例组件中使用
定义
写在el data等同级位置
filters:{ 过滤器的名字(你要过滤的数据){ return 你的逻辑 } }
Document {{text|xiaoming}}
{{textb|xiaoming}}
使用
在想使用的地方 使用| 来完成
{{变量|过滤器的名字}}
全局过滤器和局部过滤器的优先级
注意:当全局过滤器和局部过滤器重名时,会采用局部过滤器
在vue中可以自动执行的函数叫做钩子函数
在现有内置指令不够用的时候 我们可以自己定义指令来进行使用
bind 指令绑定到元素之上的时候执行 但是只执行一次
unbind 指令被移除的时候执行 只执行一次
update 所有组件节点更新的时候执行调用
componentUpdate 指令所在节点以及所有的子节点都更新完成的时候调用
inserted 绑定指令的元素在页面展示的时候调用
在所有位置都能生效
Vue.directive("自定义指令名字",{ 自定义指令钩子函数(el你绑定自定义指令的dom){ 你的逻辑 } })
指令 就是在vue中带有v-前缀的html特殊属性
在你想使用的位置 v-你的自定义指令名
只能在局部范围内生效
语法:写在与data methods watch computed 同级的位置
directives:{ 自定义指令的名字:{ 自定义指令的钩子函数(el代表的就是指定放在那个dom上形参就是谁){ 你的逻辑 } }, 自定义指令的名字2:{ }, }
使用 v-自定指令的名字
脚手架就是项目的开发环境 在脚手架中已经把我们需要开发的一切环境已经配置好了 我们只需要直接写业务代码即可
1全局安装脚手架 npm install -g @vue/cli
2查看版本 vue --version
上面的两部 除非你重新装系统或者 重新装node了 否则不需要重复
3。创建项目
(3-1)把你的cmd切换到你要创建项目的文件夹中
(3-2)vue create 你的项目名不要中文空格特殊符号
4 cd到项目名下
5启动项目 npm run serve
6 根据提示打开浏览器访问
1.删除src文件夹下的components文件夹下的helloword.vue文件
2.在components文件夹中创建属于你自己的文件xxxx.vue(名字必须使用大驼峰命名法 首字母大写其后单词也要大写)
3.在vscode下载 vetur插件 安装完成 在你新建的文件中 使用 参考视频 1.cd项目下 npm install 下载依赖 2.启动项目 找到项目下的package.json文件 的scripts节点去查看启动命令 注意 所有单词都是npm run 你的配置 唯独 start不一样 因为start可以不加run 1.找到vue。config.js文件写入如下内容 组件的本质就是自定义标签 组件其实就是把我们的页面差分成一个个的小模块 分开编写 增加开发效率 降低维度难度 复用性更高 在vue中组件使用.vue文件来进行表示 .vue文件叫做单文件组件 在创建组件的时候 我们是在components文件夹中进行创建的 一个.vue文件中 有三个部分 template -------》写html script -------》写js逻辑 style -------》写css 有的时候 一个组件在很多个地方都要被重复使用 那么默认情况下 我们使用局部组件的引用调用使用 每次在这样写很麻烦 全局组件 只需要配置一次main.js 那么就可以在当前项目的任意位置直接使用(全局组件慎用 因为全组件可能会造成组件命名污染) 局部组件 只能在特定区域使用的组件 谁引用的 谁才能用 1.引用 2.调用 3.使用 使用scoped属性可以让 当前样式仅对当前组件生效 之前学过 data methods watch 还有一堆堆指令 除了data以外 剩下的都一样 数据以函数返回值形式定义,这样每复用一次组件,就会返回一份新的 组件和组件之间相互嵌套 父组件的数据 子组件不能直接使用 子组件的数据 父组件也不能直接使用 组件与组件之间是一个完整地 独立地个体 他们之间的数据 默认是不能相互使用的 props是vue实例的一个属性 他的作用是用来让组件接受外部传递进来的数据 语法: 写在data methods watch 的同级 props:[接收参数1,接受参数2,........n] 基本props使用 1.在子组件中 使用props来定义接收参数 2父组件 开始给子组件传递参数 props验证语法 在上面的例子中 大家会发现 我们给子组件传递任意数据类型都可以 但是如果我们想限制父组件给子组件传递的数据类型时候 那么就要使用props验证 props验证可以对传递给子组件的数据 进行约束(数据类型 默认值 等) props验证 仅仅只在浏览器的控制台中打印出警告 但是不会对页面的展示造成影响(props验证只是给我们开发人员一个数据类型不匹配的 提示) 语法: props:{ 接受参数:{ type:数据类型,//验证类型 default:"默认值" }, 接受参数2:{ type:数据类型 }, } 子组件 $emit自定义事件 逆向传值默认是不被允许的 我们需要使用一些歪门邪道 需要使用自定义事件来完成 $emit() 注意:很多同学后期在被问到$emit是什么的时候 总会回到他是逆向传值 这个回答是错的 因为$emit是自定义事件 而逆向传值只是他能完成的一个小功能 1.在子组件中必须必须通过事件来触发自定义事件的抛出 2.父组件接收 ref 把ref绑定到子组件身上 那么就可以得到当前这个子组件的所有信息 包含他的data数据 从而完成了逆向传值 因为vue中是单向数据流(组件与组件之间的数据传递只能是单向一层一层的进行传值)那么在多层级传值的时候如果我们一层一层的传值 非常麻烦 vuex就是在vue中的统一状态(数据)管理工具 在创建的时候选择vuex即可 state属性的作用 就是在vuex中存放数据的 我们今后在vuex的所有数据都写在state中 使用state的数据 因为vuex是统一状态管理 所以在项目下的任何数据 都可以直接使用state的数据 语法: this.$store.state.xxx 方式1 直接读取 方式2 使用计算属性间接读取 随着项目的体积逐渐增大 那么变量与今后的其他操纵就会在vuex的文件中 增多 导致这个文件中的内容原来越冗余 后期也几乎无法维护 为了避免上述情况 所以我们可以使用vuex的模块把内容进行拆分 1.在store文件夹下创建文件夹用来存放我们拆分的模块 2.创建模块文件 写入如下内容 3.在store下的index.js 中关联并使用我们创建的模块 4,如果vuex被拆成了模块的话 那么我们要使用数据 必须使用 this.$store.state.模块名.xxxx mutations的作用就是在vuex中修改state的 如果想修改state的数据必须使用mutations来修改 mutations是一个属性 这个属性中包含的是一个个修改的函数 mutaitons想使用 那么我们需要在组件中使用commit()来进行调用 组件内使用commit调用 创建mutations中的内容 mutations的payload 在我们使用commit()的时候 第一个参数是你要调用的修改动作名 第二个参数是你要给mutations传递的数据 在mutations中可以读取这个第二个参数 扩展---vuex数据修改刷新丢失 监听页面刷新 如果刷新 那么就把vuex的数据存储到本地存储中 然后当页面在此加载得的时候 把本地存储存的原始vuex的数据那出来 替换当前的state数据 actions是vuex的一个属性 他的作用就是在vuex中进行异步操作的触发(很多同学后期总爱说actions是异步请求 但是要注意他不是异步请求 他是进行异步操纵的触发 异步请求只是它触发的众多异步操纵的其中一种) 语法:要触发actions使用dispatch (dispath触发actions进行异步触发 把请求来的数据 通过commit交给mutations修改state 在页面读取展示) 1.在组件内使用dispatch()触发vuex的actions进行异步请求的发送 2。需要在对应的actions创建你要触发的异步触发器 3.把请求来的数据通过context.commit()触发修改 4创建对应mutations修改state vue的计算属性是computed 对data的数据进行依赖 处理之后返回新的计算之后的结果 vuex的getters也是计算属性 只是他和上面的computed最大的区别就是 他处理的数据可以在任何组件直接使用 而vue的computed 处理的数据只能在当前组件使用 使用: this.$store.getters.xxx 1.区别: vuex存储在内存; localstorage(本地存储)则以文件的方式存储在本地,永久保存; sessionstorage( 会话存储 ) ,临时保存。 localStorage和sessionStorage只能存储字符串类型, 对于复杂的对象可以使用ECMAScript提供的JSON对象的stringify和parse来处理。 2.应用场景:vuex用于组件之间的传值,localstorage则主要用于不同页面之间的传值。 3.永久性:当刷新页面时vuex存储的值会丢失,localstorage不会。 注:很多同学觉得用localstorage可以代替vuex, 对于不变的数据确实可以,但是当两个组件共用一个数据源(对象或数组)时,如果其中一个组件改变了该数据源,希望另一个组件响应该变化时,localstorage无法做到响应式,vuex可以绑定数据响应式。 Vuex数据状态持久化的使用场景 1、购物车 比如你把商品加入购物车后,没有保存到后台的情况下,前端来存,就可以通过这种方式vuex+localStorage(sessionStorage)。 2、会话状态 授权登录后,token就可以用Vuex+localStorage(sessionStorage)来存储。 3、一些不会经常改变的数据 比如城市列表等(当前也要留下可以更新的入口,比如版本号) 小提示:localStorage.setItem(key, String), set的值必须是字符串,如果你的数据是对象都需要先行转换(JSON.stringify(xxx)),取出时localStorage.getItem(key),取出后的字符串可以通过JSON.parse(xxx)转回对象。 组件的本质是自定义标签 标签分为双标签和单标签(就是需要在页面中展示标签所蕴含内容的时候用双标签 单标签就是在页面中表现一个行为的时候 ) 组件单标签不考虑了 但是双标签我在其中插入内容是否可以 xxxxxxxxx 如果在组件的开关标签中插入内容 默认是不显示的(组件是一个完整的独立的个体 如果没有特殊设置 那么没有办法向组件中插入内容) 用来混合父组件与子组件自己的模板 slot其实就是让组件接受一个外部插入进来的dom元素 并且进行显示 通过slot就可以扩展组件的复用性 组件的本质 自定义标签 标签可以是双标签也可以是单标签 那么我们能不能向组件的开关标签内插入dom内容? 默认情况下 给组件的开关标签中写入dom页面不显示原因是因为 组件是一个完整地独立地个体 外部的内容默认插入不进来 在组件中想接受外部插入的dom位置 直接写slot标签 即可接受外部的dom 什么时候使用? props 接收的是组件外部传递进来的数据变量 slot接收的是 组件外部传递进来的dom 语法 在定义slot的时候 使用name属性起个名字 在使用的时候使用slot属性 指定那个槽口 vue实例从创建到销毁的过程中被自动执行的函数 写在与 data methods watch computed directives 同级的位置 就是给程序提供一个自动执行逻辑的场所 实例创建 实例创建之前-----beforeCreate 实例创建之后-----created 模板渲染 模板渲染之前-----beforeMount 模板渲染之后-----mounted 数据更新 数据更新之前-----beforeUpdate 数据更新之后-----updated 实例销毁 实例销毁之前-----beforeDestory 实例销毁之后-----destoryed vue实例从创建到销毁的过程中被自动执行的函数 实例创建之前-----beforeCreate 实例创建之后-----created 模板渲染之前-----beforeMount 模板渲染之后-----mounted 4大阶段 8个钩子 实例创建前后 模板渲染前后 mounted 实例创建 实例创建之前-----beforeCreate 数据的观测与事件的初始化 属性的创建 还没有进行 实例创建之后-----created 在此时vue实例已经创建完毕 所以 数据的观测 属性 方法等内容都已经创建完毕(el属性还没有挂载) 模板渲染 模板渲染之前-----beforeMount 在页面挂载前调用的 所以在此阶段 页面还没有进行渲染与模板的编译 程序在此时会把数据绑定到页面上 但是页面并没有显示 模板渲染之后-----mounted 页面已经渲染出来了 html的内容会在dom中进行加载展示 数据更新 数据更新之前-----beforeUpdate 在此时数据会不停的在dom中进行修改 数据更新之后-----updated 把修改之后的dom内容已经在页面成功的展示了 实例销毁 实例销毁之前-----beforeDestory 此时vue实例还能用 实例销毁之后-----destoryed 什么都没有了 vue实例等内容都没了 前端指的是数据展示 后端指的是数据处理 1.传统的ajax 就是值使用XMLHttpRequest方法实现的数据请求 他隶属于原生的js 核心就是XMLHttpRequest对象 如果多个请求有先后顺序的话 那么容易造成回调地狱问题 jqueryajax 就是对原生XMLHttpRequest封装 2.axios 是基于promise封装的 本质上还是XMLHttpRequest的封装 只不过他是基于最新的语法进行封装的 3.fetch 就是原生js最新标准 和XMLHttpRequest没有半点关系 1.原生ajax 2.jqueryAjax 对上面的XHR对象进行了封装 方便使用 (1)下载jquery npm install --save jquery (2)引用jquery (3)使用jquery 3.axios 也是对XHR对象进行封装 当时它是使用符合当下的promise来进行的封装 (1)下载 cnpm install --save axios (2)引用 (3)使用 4.fetch fetch和上面三个都不一样 因为他没有使用XHRajax对象 而是es最新的请求标准 但是既然是最新的 那么兼容性有很大问题 axios 是目前最优秀的 HTTP 请求库之一,虽然 axios 已经封装的非常好了,我们可以直接拿过来用。但是在实际的项目中,我们可能还需要对 axios 在封装一下,以便我们更好的管理项目和各个接口。 axios.request 该方法是axios项目的核心处理方法,实现用户自定义配置、应用拦截器、发送请求核心功能 axios其他api 就是发送相关请求 比如get请求 delete请求等 get方式--params发送参数 post方式--data发送参数 但是大家会发现后台接收不到我们发送的数据 原因是因为: 在发送post的时候Content-type(表示请求和响应中的媒体类型信息。它用来告诉服务端如何处理请求的数据,以及告诉客户端(一般是浏览器)如何解析响应的数据)常见有三种形式: Content-Type: application/json 就是给后台的数据是一个json对象 Content-Type: application/x-www-form-urlencoded 表单数据编码为键值对,&分隔 如:name=java&age = 23 Content-Type: multipart/form-data 通常文件上传 现在最主流的是application/json形式axios默认就是这种方式 就像上面我们写的post代码 直接把后该要的参数放到data中就可以了 如图可以发现我们的请求方式 但是有时候后端要求Content-Type必须以application/x-www-form-urlencoded形式,那么通过上面application/json传递的参数,后端是收不到的,我们必须对参数数据进行所谓的序列化处理才行,让它以普通表单形式(键值对)发送到后端,而不是json形式 用qs模块来序列化参数 我们也能通过第三方依赖来序列化参数,就更加方便简洁,下载qs模块。 1.下载 npm install --save qs 2.引用 import qs from “qs” 3.在传递数据的时候使用qs序列化 delete put 等方式 delete 同get put方式 同post 因为浏览器的安全机制 同源策略 不同端口不同域名不同协议 就会造成跨域 面试的时候千万不要先说jsonp 太low 面试的时候千万不要先说jsonp 太low 面试的时候千万不要先说jsonp 太low 面试的时候千万不要先说jsonp 太low 代理:造成跨域的问题是浏览器的安全机制 因为有了这个安全机制我们才要解决跨域 我现在不让浏览器帮我发送请求了 而是让我项目的服务器帮助我绕开浏览器发送请求 nginx反向代理 devServer代理跨域 devServer就是vue脚手架中那个内置的微型开发小服务器--注意 上线之后 该种方式就会失效 1.在项目的根路径下 创建一个vue.config.js 2.写入如下内容 3.修改请求路径 4.千万不要忘了重启 4.千万不要忘了重启 4.千万不要忘了重启 4.千万不要忘了重启 4.千万不要忘了重启 4.千万不要忘了重启 4.千万不要忘了重启 后端解决跨域----在工作的时候 后端解决跨域 大概率都是在项目上线的时候 才会解决 json-server就是为我们创建模拟数据的一个技术 因为在开发的时候 我们不可能写一个功能之前后端都给我们把对应的接口写好 所以有的时候我们需要有模拟数据 来给我们提供页面展示内容 1.安装 2.在项目中新建一个mock文件夹 并且创建json文件用来容纳模拟数据文件 3.cd到mock文件夹下 json-server --watch xxx.json --port 端口号 JSON-server 的增删改查_jsonserver增删改查_悠悠~飘的博客-CSDN博客 在大型项目中 http请求会有很多 而且我们需要区分 所以我们可以把数据请求拆分出来 单独管理这样一来 就可以增强项目的可维护性与可扩展性 大体分为如下内容 api 请求集中式管理 实现请求拦截 实现响应拦截 常见错误信息处理 请求头设置 1.初始化axios实例 编写util文件夹 在编写index.js来容纳封装内容 虽然axios中已经给我们封装好了一些常见的api如上面的axios.get axios.post等 但是我们为了更好的全局控制所有请求的相关配置,所以我们使用 axios.create()创建实例的方法来进行相关配置,这也是封装 axios 的精髓所在。 通过 create() 方法我们得到了一个 axios 的实例,该实例上有很多方法,比如拦截器等等。我们创建实例的时候可以配置一些基础设置,比如基础请求地址,请求超时等等。 2.新建api文件夹 并且编写js文件用来容纳 api请求集中管理 3.在组件中进行使用我们封装的 api请求集中管理 我们就简单的划分出 拦截器 每次发送请求或者请求相应的时候 都会经过拦截器 才会进入到我们的程序(就是对我们的请求和相应进行发送前或者获取前的一个拦截 ) 请求拦截 有的时候我们在发送请求的时候后端需要每次我们携带一些数据(比如token)但是添加在每个请求非常麻烦 所以我们可以添加在请求头中 那么这个配置我们可以在拦截器中进行设置 相应拦截 响应拦截器的作用是在接收到响应后进行一些操作 响应拦截器也是一样如此,就是在请求结果返回后,先不直接显示错误,而是先对响应码等等进行处理,处理好后再导出给页面一个错误提醒 添加请求 响应拦截 响应拦截设置HTTP状态码 得到错误http状态码 设置请求拦截请求头信息 你把用户的token以请求头的方式给后台 地址:/userlist/routertoken 方式 get 参数: 头信息是usertoken 返回值?? ##### 扩展:token的续签问题-token过期了怎么办? 方案一 每次请求都返回新 token 假设一个 token 的签发时间为 12:00,需求为 2h 未进行请求即过期。则设置有效期 2h,那么每次请求都会把一个 token 换成一个新 token。如果 2h 没有进行请求,那么上一次请求的到的 token 就会过期,需要重新登录。不断签就能一直使用下去。 这种方式实现思路很简单,但开销比较大。 方案二 用户登录返回两个 token 第一个方案是 accessToken ,它的过期时间 token 本身的过期时间2个小时,另外一个是 refreshToken 它的过期时间更长一点比如为1天。客户端登录后,将 accessToken和refreshToken 保存在本地,每次访问将 accessToken 传给服务端。服务端校验 accessToken 的有效性,如果过期的话,就将 refreshToken 传给服务端。如果有效,服务端就生成新的 accessToken 给客户端。否则,客户端就重新登录即可。 该方案的不足是: 1、需要客户端来配合; 2、用户注销的时候需要同时保证两个 token 都无效; 3、重新请求获取 token 的过程中会有短暂 token 不可用的情况(可以通过在客户端设置定时器,当accessToken 快过期的时候,提前去通过 refreshToken 获取新的accessToken) 在开发的时候一般会有是三个环境:开发环境 测试环境 线上(生产)环境 vue 中有个概念就是模式,默认先vue cli 有三个模式 但是往往开发的时候可能不止有三种: 本地环境(local) 开发环境(development) 测试环境(devtest) 预发布环境(beta) 生产环境(production) 通过为.env文件增加后缀来设置某个模式下特有的环境变量。 1.在项目根路径下设置 新建对应文件 .env.development(开发环境文件) .env.production(生产环境文件).env.devtest(测试环境文件) 2.在每个文件写入如下内容(VUE_APP_随便写) 使用变量process.env.你的内容即可得到 打包生产环境 1.npm run build 会生成一个dist文件夹 我们点开之后像运行html一样运行项目 2.配置生产环境 (1)在vue.config.js中设置 publicPath:"./" (2)把路由模式设置为hash 3.重新build 因为我们可以配置不同的环境变量 那么我们就可以在设置请求的时候 根据不同的环境来设置不同的请求地址 1.在开发环境文件中配置我们的请求(今后开发的时候可以在其他配置文件中配置你的请求) 2.在封装的拦截器文件中配置baseURL 3.在今后的请求中直接写请求的路由地址就行 就是可以让我们完成一个SPA单页面应用 传统的页面在跳转切换的时候会造成页面加载白屏 这样一来用户体验非常的差 但是 我们通过spa应用 可以达到类似于原生app的切换效果 切换没有白屏 丝滑切换 用户体验更高 路由的本质 就是根据url的不同来渲染不同的组件页面 在创建项目的时候 选中router项即可在项目中集成路由 拿到路由项目之后怎么办? 1.删除掉views文件夹中的内容 2.删除掉components下的helloword.vue 3.在app。vue中删除内容 但是router-view千万千万不要删 稍后在说 是vue路由的默认路由模式 如果你不指定 那么就是hash模式 url带#号 上线之后刷新不会丢失 浏览器兼容性好 url不带#号 上线之后刷新会丢失(需要让你们后端服务器人员给你配置服务器的重定向) 浏览器兼容性一般 1.在views文件夹中创建对应的路由页面组件 2.配置路由 在router下index.js中进行配置 (2-1)先把你要使用的组件页面引用 (2-2) 配置路由规则 3.在app.vue中设置router-view路由出口 路由导航就是在页面中的一些连接通过点击之后完成页面的跳转 不能使用a标签 router-link这个标签来完成页面之间的跳转 其中有一个to属性就是写你的路径 扩展--动态类名 在当前路由页面下 vuerouter会给对应的声明式导航添加一个类名 通过这个类名可以设置当前的样式 this.$router.push("/你要去的路径") push跳转的页面可以回退回来 this.$router.replace('/替换路径') replace是替换 跳转之后不能回退 this.$router.go()正数前进 负数后退 二级路由或者多级路由在创建的时候 使用children关键字来进行规则的配置 1.路由页面创建 views文件夹 2.配置二级路由规则 (2-1)在router文件夹下的index.js中先引用二级路由页面 (2-2)配置路由规则 必须在对应的一级路由规则中使用children关键字来进行配置 3.注意 3.注意 3.注意 3.注意 3.注意 必须设置二级路由的路由出口 router-view (写在对应一级路由的页面中) 在上面的笔记中 会发现 我们在配置二级路由的时候 path路径为 /二级 在路由导航的时候 我们在to中直接就写/二级 那么路由导航 注意 注意 注意 注意 注意 注意 我们在写二级路由的时候 path 也可以写成 不加/的方式 在路由导航的时候 必须写成 /一级/二级 重(重新)定(定位)向(方向) 就是给用户一个页面错误提示的作用 就是把数据从一个路由页面传递到另外一个路由页面中 (新闻列表页面 用户点击之后 会跳转到新闻详情页 但是新闻详情页展示的内容应该是用点击的那一条新闻 所以如何把数据从一个页面传递到另外一个页面) params方式 1.在需要接受数据的路由页面规则上 设置接受参数 2.发送 声明式 编程式 3.接受 在接受的页面使用 this.$route.params.xxx 注意单词 query方式 1.发送 声明式 编程式 课下自行尝试 2.接受 在想使用数据的页面中 使用 this.$route.query.xxxx 路由传参或者是动态路由匹配 他是vue-router 给我们提供的一种 把数据从一个路由页面传递到另外一个路由页面的技术 他有两种方式 第一种params 这种方式在传递参数的时候共3步 1在接收的路由页面规则中配置接收参数 2发送 我是使用name作为跳转地址 使用params来设置发送参数 3接收 能想起来就说想不起来就不说 this.$route.params.xxxx 第二种方式 query 他在传递数据的时候共两部 发送和接收 和params方式基本相同 唯独在发送数据的时候 他不但可以使用name作为跳转地址 还可以使用path 在接收的时候 语法也略有不同 他使用this.$route.query.xxxx 1.url展示形态 params方式 传递参数的时候相对安全一些 因为不显示传递的数据key query方式 在传递的时候会显示数据key和val 相对来说没有parmas安全 2.刷新丢失 parmas方式 刷新会丢失 (上线之后) query方式 刷新不会丢失 3.语法区别 $router代表的是 路由对象 他所涉及的范围是全部项目页面 $route代表 当前路由页面对象 在没有使用路由懒加载的时候 第一次加载会把所有的路由页面全部加载完 在显示页面 有的时候 用户的设备网络不好的时候 导致渲染时间过长 用户就会在第一次进入我们项目的时候 有白屏问题(用户体验不好) 异步组件方式 import导入方式 在路由跳转的特定时期自动触发的一些函数 (这些钩子函数通常是在页面跳转的时候 对项目中的用户权限进行设置的) 全局 所有页面在跳转的时候都会触发 全局前置守卫---beforeEach() 在所有路由跳转之前 触发的钩子函数(此时路由还没有跳转完成) 全局后置守卫--- afterEach(() 所有路由跳转之后 触发的钩子函数 (此时已经进入到了跳转之后的新页面) 路由独享 指定页面在跳转的时候触发 路由独享---beforeEnter 只会对一个路由规则生效(路由独享写在那个规则之上 就对哪一个生效) 组件内 仅仅对某些组件在路由跳转的时候生效 进入组件时候---beforeRouteEnter 离开组件的时候---beforeRouteLeave 因为使用传统的路由来进行开发的时候可能会出现首页加载白屏问题( 因为如果我们有200个页面 那么在第一次加载的时候 会把这200个页面都先加载出来然后再把指定的哪一个显示到页面上 ) 白屏问题对于用户体验很不好 所以为了解决首页加载白屏 那么我们可以使用路由懒加载方式 懒(按需)加载 1.hash模式 默认模式 带#号 hash模式上线正常 hash兼容性好 2.history模式 必须手工指定 不带#号 history上线之后刷新会丢失页面(让后端给你们配置服务器的重定向) 是H5新增 所以兼容性比较差 修改 使用mode属性来进行修改 就是vue组件中的一个封装技巧 他的作用就是对组件中重复出现的数据 方法 生命周期等内容进行封装方便其他组件复用 在组件中引用了混入内容之后 就可以直接在当前组件中进行使用。 全局混入 能不用尽量不要用 因为会造成全局的污染 1.新建一个文件夹 mixin 用来存储混入的代码 2.main.JS中进行配置 1.新建一个文件夹 mixin 用来存储混入的代码 2.在你想使用的组件中 引用 使用 主要分下面几种情况: 1.对于created,mounted 等生命周期函数 mixin文件中的代码先执行,组件中的后执行 2.对于data中定义的字段,组件中定义组件覆盖mixin中同名字段 3.对于 method中的同名方法,组件内的同名方法覆盖mixin中的方法 ref就是对页面的dom进行操作 进行基本的页面dom操作。 1.在标签的dom之上使用ref=“随便起个名字” 2.this.$refs.你的那个名字即可找到指定元素 因为组件的核心就是自定义标签 上面的ref可以绑定到标签之上 我们的自定义标签能不能使用ref 如果可以使用那么得到的是什么? 可以得到绑定的那个组件所有的信息(得到了vue组件对象) 1.ref可以完成逆向传值 2.父组件如何触发子组件的方法? 使用ref绑定到组件即可访问 在vue中 数据改变试图不变怎么解决? 在vue数据改变试图有的时候是不会更新的如下代码 在vue2.0中 数据的双向绑定 是基于数据劫持与发布者订阅者模式的 其中数据劫持是通过Object.defineProperty()这个方法来拦截劫持data中的数据的 因为有了这个方法 所以数据改变试图也会更新 但是 Object.defineProperty()有个问题 他是会监听初始化的数据 如果中途给数组或者对象 添加新属性的时候 Object.defineProperty() 就不会监听到 不会监听到就没有数据劫持 没有数据劫持就没有双向绑定 没有双向绑定就没有数据变试图变 如果我就是想在程序运行的时候 给vue的data中对象或者数组添加新属性 并且让试图改变怎么办? $set() 就是在程序运行的时候给对象或者数组添加新属性并且让试图改变 语法: This.$set("你要给谁添加","你要添加的key","你要添加的val") 多个组件 使用同一个挂载点 并且动态切换 1.需要有多个组件 2.设置挂载点 3.动态切换 修改挂载点上绑定的变量即可 在路由或者动态组件中 页面用户填写的数据可能会随着页面的切换 而丢失(原因是因为 在我们每次切换的时候 vue都会创建一个新的组件实例 ) 如果我就是想在切换的时候保存之前的页面内容呢? keep-alive 用来保存组件切换的时候页面状态内容 使用keep-alive包裹的组件 不会随着页面的切换 而数据丢失 因为当前组件在切换的时候会被缓存起来 那么这样一来 在组件切换的时候能减低性能上的损耗 动态组件 包裹挂载点即可 路由 包裹路由出口即可 属性 incloud 你要缓存谁 excloud 你不想缓存谁 如果 我把两个都写了 excloud的优先级大于incloude 钩子函数 activated 在进入被kepp-alive管理的组件时候触发 deactivated 在离开被kepp-alive管理的组件时候触发 这两个钩子应该写在被keep-alive管理的组件中 与data等属性同级 观察下面得代码 分析上面的代码发现:Vue 实现响应式并不是数据发生变化之后 DOM 立即变化,而是按一定的策略进行 DOM 的更新 简单来说,Vue 在修改数据后,视图不会立刻更新,而是等同一事件循环中的所有数据变化完成之后,再统一进行视图更新。 原理: 1.首先修改数据,这是同步任务。同一事件循环的所有的同步任务都在主线程上执行,形成一个执行栈,此时还未涉及DOM。 2.Vue开启一个异步队列,并缓冲在此事件循环中发生的所有数据变化。 3.同步任务执行完毕,开始执行异步队列的任务,更新DOM 为了在数据变化之后等待 Vue 完成更新 DOM 可以在数据变化之后立即使用Vue.nextTick(callback)。这样回调函数在 DOM 更新完成后就会调用。 在下次 DOM 更新循环结束之后执行延迟回调。在修改数据之后立即使用这个方法,获取更新后的 DOM。 pnpm是 Node.js 的替代包管理器。它是 npm 的直接替代品,但速度更快、效率更高。 为什么效率更高 ? 当您安装软件包时, 我们会将其保存在您机器上的全局存储中,然后我们会从中创建一个硬链接(pnpm 通过硬链接的方式保证了相同的包不会被重复下载,比如说我们已经在 A 中下载过一次 [email protected] 版本,那我们后续在 B 中安装 [email protected] 的时候是会被复用的,具体就是 A 中的 xxx 中的文件和 B 中的 xx 中的文件指向的是同一个 ) PNPM不是进行复制。对于模块的每个版本,磁盘上只保留一个副本。例如,当使用 npm 或 yarn 时,如果您有 100 个使用的包,则磁盘上将有 100 个 lodash 副本。pnpm 可让您节省数 GB 的磁盘空间! 原来咱们学习v3的时候 使用的是vue/cli脚手架(实质上是帮助我们封装好了webpack来帮助我们构建与编译项目) Vite 是 vue 的作者尤雨溪在开发 vue3.0的时候开发的一个 web 开发构建工具。其本人在后来对 vue3 的宣传中对自己的新作品 Vite 赞不绝口,并表示自己 ”再也回不去 webpack 了“ 。那么 Vite 究竟有什么魅力 Vite 优势: 1.vite 开发服务器启动速度比 webpack 快 webpack 会先打包,然后启动开发服务器,请求服务器时直接给予打包结果。 vite 在启动开发服务器时不需要打包,也就意味着不需要分析模块的依赖、不需要编译,因此启动速度非常快。当浏览器请求某个模块时,再根据需要对模块内容进行编译。这种按需动态编译的方式,极大的缩减了编译时间,项目越复杂、模块越多,vite的优势越明显。 2.vite 热更新比 webpack 快 当改动了一个模块后,vite仅需让浏览器重新请求该模块即可,不像webpack那样需要把该模块的相关依赖模块全部编译一次,效率更高。 3.vite 构建速度快,比 webpack 的 nodejs,快 10-100 倍。 Vite 劣势: 1.生态不及webpack,加载器、插件不够丰富 2.项目的开发浏览器要支持 ES Module,而且不能识别 CommonJS 语法 兼容性注意 Vite 需要 Node.js 版本 >= 12.0.0。 运行完的结果如下: vite.config.js 在app.vue中设置 router-view 在main.js中引用使用路由 在store/index.js中创建 pinia是一个用于vue的状态管理库,类似于vuex,是vue的另一种状态管理工具 Pinia意为菠萝,表示与菠萝一样,由很多小块组成。在pinia中,每个store都是单独存在,一同进行状态管理。 很多人也将pinia称为vuex5,因为pinia将作为vue3推荐的状态管理库,而vuex将不再迭代。 与 Vuex 相比,Pinia 提供了一个更简单的 API,具有更少的规范,提供了 Composition-API 风格的 API,最重要的是,在与 TypeScript 一起使用时具有可靠的类型推断支持。 1.mutations 不再存在。 2.无需创建自定义复杂包装器来支持 TypeScript,所有内容都是类型化的,并且 API 的设计方式尽可能利用 TS 类型推断。 3.不再需要注入、导入函数、调用函数、享受自动完成功能! 4.无需动态添加 Store,默认情况下它们都是动态的 5.不再有 modules 的嵌套结构 通过你喜欢的包管理器安装 1.创建pinia实例并挂载 在main.ts中 2.在src目录下新建一个stroe文件夹,在文件夹中新建一个index.ts 3.通过import将下载好的pinia引入到index.ts中并导出 并且使用defineStore创建store对象 state 是 store 的核心部分。 我们通常从定义应用程序的状态开始。 在 Pinia 中,状态被定义为返回初始状态的函数。 Pinia 在服务器端和客户端都可以工作。 1.创建state 2.组件中使用 修改方式1 1.组件内创建触发 调用修改方法 store.你在pinia中actions中创建的修改函数 2.pinia中创建修改方法 修改方式2--$patch $patch是pinia的内置方法 通过该方法可以直接修改state的数据 方式1 在组件内调用$parch 便可直接修改数据 方式2 我们正常在取值的时候是使用如下方法 那么大家发现store能点出数据 那么store是一个对象 那么我们可能想到了如果数据很多 每次store点xxx就很麻烦 那么我们使用解构来试试 大家会发现可以正常读取 那么修改下数据试试 发现数据并没有修改 原因是因为使用解构之后 数据就失去了响应式 所以我们使用 顾名思义就是让数据恢复成state的初始值 $subscribe 监听store数据修改 当数据改变了 那么subscribe也会触发 在pinia中acition不但能处理同步操作 同样也可以处理异步操作 使用方式和之前一致 pinia中定义模块 只需要定义多个store即可 因为pinia没有单一数据源这个概念 在其中可以定义多个store对象 数据可视化就是把我们需要给用户传达的 抽象数据以更加直观的方式展示出来 1.下载 npm install --save echarts yarn add echarts pnpm install echarts (vue2与3 下载都是相同的) 2.引用 vue2 vue3 3.使用 vue2 vue3 Documentation - Apache ECharts Documentation - Apache ECharts Documentation - Apache ECharts 就是用来设置echarts到底是用那种图表的 DataV.GeoAtlas地理小工具系列
扩展----项目怎么启动?
"scripts": {
"diaodeyipi": "vue-cli-service serve",
"build": "vue-cli-service build",
"lint": "vue-cli-service lint"
},
扩展----自动开启浏览器与端口修改
const { defineConfig } = require('@vue/cli-service')
module.exports = defineConfig({
transpileDependencies: true,
// 设置浏览器自动开启
devServer:{
open:true,//设置自动开启
port:8888,//修改端口
host:"localhost"
}
})
组件化
组件的基本概念
组件的创建
.vue文件基本页面内容
组件的分类
全局组件---component
import Vue from 'vue'
import App from './App.vue'
// 1.引用
import AllCom from "./components/AllCom.vue"
// 2.配置全局组件
// Vue.component(“给你这个全局组件起个名字”,你所要对应使用的组件)
Vue.component("AllCom",AllCom)
Vue.config.productionTip = false
new Vue({
render: h => h(App),
}).$mount('#app')
局部组件---components
组件样式隔离--scoped
之前学的内容怎么用
vue组件的data为什么是一个函数?
data
,类似于给每个组件实例创建一个私有的数据空间,让各个组件实例维护各自的数据。而单纯的写成对象形式,就使得所有组件实例共用了一份data
,就会造成一个变了全都会变的结果。父子组件
父组件
组件传值
正向传值---父组件给子组件数据---props
逆向传值--- 子组件给父组件数据
同胞传值--- 兄弟组件(有一个共同的父组件)传值
跨层级传值 --- 爷爷组件给孙子组件---vuex
vuex是什么?
vuex创建
vuex5大属性---state---数据源
import Vue from 'vue'
import Vuex from 'vuex'
Vue.use(Vuex)
export default new Vuex.Store({
state: {//就是数据源 存放数据的地方
text:"我是字符串",
num:18,
bool:true,
arr:[1111,2222,3333],
obj:{
name:"xixi",
age:18
}
},
getters: {
},
mutations: {
},
actions: {
},
modules: {
}
})
This is an about page---{{this.$store.state.text}}
This is an about page---{{this.$store.state.text}}
计算属性方式-- {{newbool}}
vuex5大属性---module---模块
可以直接从store下的index里面复制过来
let aboutm={
state: {//就是数据源 存放数据的地方
text:"我是字符串",
bool:true,
},
getters: {
},
mutations: {
},
actions: {
},
}
// 必须暴露
export default aboutm
import Vue from 'vue'
import Vuex from 'vuex'
// 1.引用模块
import homem from "./modules/homem.js"
import aboutm from "./modules/aboutm.js"
Vue.use(Vuex)
export default new Vuex.Store({
modules: {//2配置模块
homem,
aboutm
}
})
This is an about page---{{this.$store.state.aboutm.text}}
计算属性方式-- {{newbool}}
{{this.$store.state.homem.num}}
vuex5大属性---mutations---修改数据的
let aboutm={
state: {//就是数据源 存放数据的地方
text:"我是字符串",
bool:true,
},
getters: {
},
mutations: {
我是修改函数1(){
},
我是修改函数1(){
},
我是修改函数1(){
},
我是修改函数1(){
},
我是修改函数1(){
},
},
actions: {
},
}
// 必须暴露
export default aboutm
methods:{
add(){
// 调用vuex的修改
// this.$store.commit("你调用的mutations名字随便写",你想给mutations的数据 可选)
this.$store.commit("NUM_LIST_ADD_DATA")
},
del(){
this.$store.commit("NUM_LIST_DEL_DATA")
}
},
let homem={
state: {//就是数据源 存放数据的地方
num:666
},
getters: {
},
mutations: {
// state是一个形参可以随便写但是建议写state
// 这个形参的作用就是代表上面的数据源
NUM_LIST_ADD_DATA(state){
state.num++
},
NUM_LIST_DEL_DATA(state){
state.num--
},
},
actions: {
},
}
// 必须暴露
export default homem
methods:{
add(){
// 调用vuex的修改
// this.$store.commit("你调用的mutations名字随便写",你想给mutations的数据 可选)
this.$store.commit("NUM_LIST_ADD_DATA",{inputval:this.inputval})
},
del(){
this.$store.commit("NUM_LIST_DEL_DATA",{inputval:this.inputval})
}
},
mutations: {
// state是一个形参可以随便写但是建议写state
// 这个形参的作用就是代表上面的数据源
// 第二个参数就是payload (载荷) payload就是接受commit的第二个参数
NUM_LIST_ADD_DATA(state,payload){
state.num=state.num+payload.inputval
},
NUM_LIST_DEL_DATA(state,payload){
state.num=state.num-payload.inputval
},
},
created () {
//判断是否有store这个本地存储的数据
if (sessionStorage.getItem("store") ) {
// 如果有 那么把vuex的数据替换 把当前的state 和上次刷新存储的state合并起来
this.$store.replaceState(Object.assign({}, this.$store.state,JSON.parse(sessionStorage.getItem("store"))))
sessionStorage.removeItem("store")
}
//在页面监听绑定一个beforeunload事件(页面刷新事件)
// 当页面刷新的时候使用本地存储存一个store的数据 把vuex的数据全部取出来 转成字符串存起来
window.addEventListener("beforeunload",()=>{
sessionStorage.setItem("store",JSON.stringify(this.$store.state))
})
}
vuex5大属性---actions---异步触发器
actions: {
// actions中也是一个个的方法 每个方法就是一个异步触发器
// context代表的就是vuex store对象
AXIOS_CESHI(context,payload){
// 在actions中就可以写请求
$http({
url:payload.url,
method:"get"
}).then((ok)=>{
console.log(ok.data)
})
}
},
actions: {
// actions中也是一个个的方法 每个方法就是一个异步触发器
// context代表的就是vuex store对象
AXIOS_CESHI(context,payload){
// 在actions中就可以写请求
$http({
url:payload.url,
method:"get"
}).then((ok)=>{
console.log(ok.data)
context.commit("AXIOSDATA",{data:ok.data})//把请求来的数据通过commit触发mutations
})
}
},
mutations: {
AXIOSDATA(state,payload){
state.arr=payload.data
}
},
vuex5大属性---getters---vuex的计算属性
getters: {
// state就是上面的数据源
newtext(state){
return state.text.toUpperCase()
}
},
扩展:vuex存储和本地存储(localstorage、sessionstorage)的区别
slot槽口/插槽
引子
<组件>
引子
你好么么哒
你好么么哒
你好么么哒
你好么么哒
你好么么哒
你好么么哒
基本slot
具名槽口--带有名字的槽口
你好么么哒1
你好么么哒2
你好么么哒3
你好么么哒4
你好么么哒5
你好么么哒6
生命周期的钩子函数
作用
8大钩子
什么是生命周期的钩子函数?
生命周期第一次执行那些?
生命周期几个阶段
第一次页面加载触发那些
dom在那个阶段渲染完毕?
请您介绍一下生命周期的每个钩子?(请您给我说一下vue实例创建的流程与原理)
父子组件的生周期顺序是什么?
前后台交互
什么是前台什么是后台 什么是前端什么事后端?
分类
fetch VS axios VS ajax区别
jqueryajax请求数据
jqueryajax请求数据
{{data}}
// 引用axios
import axios from "axios"
axios({
// 地址
url:"/movie/list/data_list",
// 方式 有的同学写的时候写成methods了 有s也可以 原因是因为默认get 他不认识你带s的这个属性所以执行默认了
method:"get"
}).then((ok)=>{
console.log(ok.data.subjects)
// 把请求来的数据赋值给arr
this.arr=ok.data.subjects
}).catch((err)=>{
console.log(err)
})
axios
axios常见api
// axios.get("请求地址",{params:{发送数据key:发送的val}}).then((ok)=>{
// // 成功回调
// }).catch((err)=>{
// // 失败回调
// })
axios.get("/api/userlist/get",{params:{name:"xixi"}}).then((ok)=>{
console.log(ok)
}).catch((err)=>{
console.log(err)
})
}
// axios.post("请求地址",{data:{发送的key:发送的val}}).then((ok)=>{
// // 成功回调
// }).catch((err)=>{
// // 失败回调
// })
axios.post("/api/userlist/post",{data:{name:"xixi"}}).then((ok)=>{
console.log(ok)
}).catch((err)=>{
console.log(err)
})
let key=qs.stringify({
key:val
})
// 引用qs
import qs from 'qs';
// 序列化数据
let key=qs.stringify({
name:"xixi"
})
// 传递
axios.post("/api/userlist/post",key).then((ok)=>{
console.log(ok)
}).catch((err)=>{
console.log(err)
})
}
axios.delete("/api/userlist/delete",{params:{name:"xixi"}}).then((ok)=>{
console.log(ok)
}).catch((err)=>{
console.log(err)
})
import axios from "axios"
// 引用qs
import qs from 'qs';
// 序列化数据
let key=qs.stringify({
name:"xixi"
})
// 传递
axios.put("/api/userlist/put",key).then((ok)=>{
console.log(ok)
}).catch((err)=>{
console.log(err)
})
}
跨域
jsonp
代理跨域
module.exports={
devServer: {
proxy: { //配置跨域
'/api': {
target: 'http://www.weather.com.cn/', //需要解决跨域的地址
pathRewrite: {
'^/api': ''
}
},
}
},
}
大纲要修写法
cors
json-server模拟数据
npm i json-server -g
axios封装
api请求集中管理
import axios from "axios"
// 创建 axios 请求实例
const serviceAxios = axios.create({
baseURL: "", // 基础请求地址
timeout: 10000, // 请求超时设置
});
export default serviceAxios
import request from "../util/index.js"
export let funget=()=>{
return request({
url:"/api/userlist/get",
method:"GET",
params:{
name:"xixi"
}
})
}
// 引用
import {funget} from "../api/index.js"
// 使用
funget().then((ok)=>{
console.log(ok)
})
}
API
管理层了,这样我们每次新增加一个 API
,只需要找到对应模块的 API
文件去添加即可,然后再到具体页面导入使用就行啦。axios拦截器(不是所有公司都用)
import axios from "axios"
// 创建 axios 请求实例
const serviceAxios = axios.create({
baseURL: "", // 基础请求地址
timeout: 10000, // 请求超时设置
});
// 添加请求拦截器
serviceAxios.interceptors.request.use(function (config) {
// 在发送请求之前做些什么
return config;
}, function (error) {
// 对请求错误做些什么
return Promise.reject(error);
});
// 添加响应拦截器
serviceAxios.interceptors.response.use(function (response) {
// 对响应数据做点什么
return response;
}, function (error) {
// 对响应错误做点什么
return Promise.reject(error);
});
export default serviceAxios
// 添加响应拦截器
serviceAxios.interceptors.response.use(function (response) {
// 对响应数据做点什么
return response;
}, function (error) {
// 对响应错误做点什么
// 得到错误http状态码
console.log("响应",error.response.status)
return Promise.reject(error);
});
// 添加响应拦截器
service.interceptors.response.use(function (response) {
// 对响应数据做点什么
return response;
}, function (error) {
console.log("error", error.response.status)
switch (error.response.status) {
case 302: alert('接口重定向了!'); break;
case 400: alert('参数不正确!'); break;
case 401: alert('您未登录,或者登录已经超时,请先登录!'); break;
case 403: alert('您没有权限操作!'); break;
case 404: alert('请求地址出错'); break; // 在正确域名下
case 408: alert('请求超时!'); break;
case 409: alert('系统已存在相同数据!'); break;
case 500: alert('服务器内部错误!'); break;
case 501: alert('服务未实现!'); break;
case 502: alert('网关错误!'); break;
case 503: alert('服务不可用!'); break;
case 504: alert('服务暂时无法访问,请稍后再试!'); break;
case 505: alert('HTTP版本不受支持!'); break;
default: alert('异常问题,请联系管理员!'); break
}
// 对响应错误做点什么
return Promise.reject(error);
});
vue环境部署与baseurl配置
环境变量
development
开发环境模式用于 vue-cli-service serveproduction
生产环境模式用于 vue-cli-service build 和 vue-cli-service test:e2etest
测试环境模式用于 vue-cli-service test:unit
创建不同环境变量文件
VUE_APP_XIAOMING = "开发模式"
package.json环境对应的执行语句
"scripts": {
"serve": "vue-cli-service serve",//开发模式
"build": "vue-cli-service build",//生产模式
"dev_test_build": "vue-cli-service build --mode development_test",//测试模式
"lint": "vue-cli-service lint"
},
mounted()=>{
console.log(process.env.VUE_APP_XIAOMING);
}
默认请求地址baseurl
VUE_APP_XIAOMING = "开发模式"
VUE_APP_API = "http://localhost:8888"
let axiosurl = ""
// 如果为开发模式的话执行url
if(process.env.NODE_ENV === 'development' ){
axiosurl=process.env.VUE_APP_API
}
else{
// 否则设置成其他的模式(这里今后有很多个判断)
axiosurl=process.env.VUE_APP_API
}
// 创建axios 赋值给常量service
const service = axios.create({
baseURL: axiosurl
});
// 发送请求 url中请求地址前缀已经在baseURL中配置了
link("/one").then((ok)=>{
console.log(ok)
})
路由
路由是什么?
路由基本创建
方式1 脚手架自动创建
方式2 手工创建方式
路由模式
hash
history
一级路由创建
import Vue from 'vue'
import VueRouter from 'vue-router'
// 1.把你要使用的路由页面引用
import Fenlei from '../views/fenlei.vue'
import Gouwuche from '../views/gouwuche.vue'
import Home from '../views/home.vue'
import Jingxi from '../views/jingxi.vue'
import Wode from '../views/wode.vue'
Vue.use(VueRouter)
const routes = [
{
path: '/',//url路径
name: 'home',//给这个路由规则起个名字
component: HomeView //引用组件
},
// 下面是后面要学的
// {
// path: '/about',
// name: 'about',
// component: () => import('../views/AboutView.vue')
// }
]
const router = new VueRouter({
mode: 'history',
base: process.env.BASE_URL,
routes
})
export default router
import Vue from 'vue'
import VueRouter from 'vue-router'
// 1.把你要使用的路由页面引用
import Fenlei from '../views/fenlei.vue'
import Gouwuche from '../views/gouwuche.vue'
import Home from '../views/home.vue'
import Jingxi from '../views/jingxi.vue'
import Wode from '../views/wode.vue'
Vue.use(VueRouter)
const routes = [
{
path: '/fenlei',//url路径
name: 'fenlei',//给这个路由规则起个名字
component: Fenlei //引用组件
},
{
path: '/gouwuche',//url路径
name: 'gouwuche',//给这个路由规则起个名字
component: Gouwuche //引用组件
},
{
path: '/home',//url路径
name: 'home',//给这个路由规则起个名字
component: Home //引用组件
},
{
path: '/jingxi',//url路径
name: 'Jingxi',//给这个路由规则起个名字
component: Jingxi //引用组件
},
{
path: '/wode',//url路径
name: 'Wode',//给这个路由规则起个名字
component: Wode //引用组件
},
// 下面是后面要学的
// {
// path: '/about',
// name: 'about',
// component: () => import('../views/AboutView.vue')
// }
]
const router = new VueRouter({
mode: 'history',
base: process.env.BASE_URL,
routes
})
export default router
路由导航
标签的方式---声明式导航
js的方式---编程式导航
二级或者多级路由创建
{
path: '/home',//url路径
name: 'home',//给这个路由规则起个名字
component: Home, //引用组件
children:[//配置二级路由规则
{
path: '/era',
name: 'era',
component: Era
},
{
path: '/erc',
name: 'erc',
component: Erc
},
{
path: '/erd',
name: 'erd',
component: Erd
},
]
},
扩展---二级路由path设置
{
path: '/home',//url路径
name: 'home',//给这个路由规则起个名字
component: Home, //引用组件
children:[//配置二级路由规则
{
path: '/era',
name: 'era',
component: Era
},
{
path: '/erc', path路径 是直接/二级
name: 'erc',
component: Erc
},
{
path: '/erd',
name: 'erd',
component: Erd
},
]
},
{
path: '/meishi',
name: 'meishi',
component: Meishi,
children:[
{
path: 'da',//这里二级路由的path没有加/
name: 'da',
component: Da
},
{
path: 'db',
name: 'db',
component: Db
},
{
path: 'dc',
name: 'dc',
component: Dc
},
]
},
路由重定向---redirect
import Vue from 'vue'
import VueRouter from 'vue-router'
import Meishi from '../views/MeiShi.vue'
import Shouye from '../views/ShouYe.vue'
import Da from "@/views/MeiShichil/DemoA.vue"
import Db from "@/views/MeiShichil/DemoB.vue"
import Dc from "@/views/MeiShichil/DemoC.vue"
Vue.use(VueRouter)
const routes = [
{
path: '/meishi',
name: 'meishi',
component: Meishi,
children:[
{
path: 'da',//这里二级路由的path没有加/
name: 'da',
component: Da
},
{
path: 'db',
name: 'db',
component: Db
},
{
path: 'dc',
name: 'dc',
component: Dc
},
]
},
{
path: '/shouye',
name: 'shouye',
component: Shouye
},
// 重定向
{
path:"/",
redirect:"/shouye"
}
// {
// path: '/about',
// name: 'about',
// // route level code-splitting
// // this generates a separate chunk (about.[hash].js) for this route
// // which is lazy-loaded when the route is visited.
// component: () => import(/* webpackChunkName: "about" */ '../views/AboutView.vue')
// }
]
const router = new VueRouter({
mode: 'history',
base: process.env.BASE_URL,
routes
})
export default router
404页面
import Vue from 'vue'
import VueRouter from 'vue-router'
import Meishi from '../views/MeiShi.vue'
import Shouye from '../views/ShouYe.vue'
import No from '../views/NoView.vue'
import Da from "@/views/MeiShichil/DemoA.vue"
import Db from "@/views/MeiShichil/DemoB.vue"
import Dc from "@/views/MeiShichil/DemoC.vue"
Vue.use(VueRouter)
const routes = [
{
path: '/meishi',
name: 'meishi',
component: Meishi,
children:[
{
path: 'da',//这里二级路由的path没有加/
name: 'da',
component: Da
},
{
path: 'db',
name: 'db',
component: Db
},
{
path: 'dc',
name: 'dc',
component: Dc
},
]
},
{
path: '/shouye',
name: 'shouye',
component: Shouye
},
// 重定向
{
path:"/",
redirect:"/shouye"
},
// 404页面必须在所有规则的最下面
{
path: '*',
name: 'no',
component: No
}
// {
// path: '/about',
// name: 'about',
// // route level code-splitting
// // this generates a separate chunk (about.[hash].js) for this route
// // which is lazy-loaded when the route is visited.
// component: () => import(/* webpackChunkName: "about" */ '../views/AboutView.vue')
// }
]
const router = new VueRouter({
mode: 'history',
base: process.env.BASE_URL,
routes
})
export default router
路由传参 动态路由匹配
{
path: '/all/:xiaoming', //设置接收参数
name: 'All',
component: All
},
{{v.title}}
<
新闻详情页---{{this.$route.params.xiaoming}}
总结路由传参/动态路由匹配
query方式与params方式传参区别
$router与$route的区别
扩展----路由懒加载
component: (resolve) => require(['你引用组件页面的路径'], resolve)
component: () => import('../views/About.vue')
路由钩子/路由守卫/导航守卫/路由卫士
import Vue from 'vue'
import VueRouter from 'vue-router'
import Home from '../views/Home.vue'
Vue.use(VueRouter)
const routes = [
{
path: '/home',
name: 'Home',
component: Home
},
{
path: '/about',
name: 'About',
component: () => import('../views/About.vue')
},
{
path: '/phone',
name: 'phone',
component: () => import('../views/phone.vue')
},
{
path: '/shop',
name: 'shop',
component: () => import('../views/shop.vue')
},
]
const router = new VueRouter({
mode: 'history',
base: process.env.BASE_URL,
routes
})
// 全局前置
// to 去哪里
// from 从哪个路由来的
// next 下一步(必须要写 不写的话 就会卡在这个钩子中不想下进行了)
router.beforeEach((to,from,next)=>{
console.log(to);
console.log(from);
if(to.path=="/phone"||to.path=="/shop"){
next()
}else{
alert("您没有登录请您登录后再访问")
next("/phone")
}
})
// 全局后置
router.afterEach((to,from)=>{
console.log("全局后置守卫")
})
export default router
{
path: '/about',
name: 'About',
component: () => import('../views/About.vue'),
// 路由独享守卫
beforeEnter(to,from,next){
console.log(to);
console.log(from);
next()
}
},
// 进入组件
beforeRouteEnter(to,from,next){
console.log("我进来了")
console.log(to)
console.log(from)
next()
},
beforeRouteLeave(to,from,next){
console.log(to)
console.log(from)
if(confirm("您确定离开吗?")){
next()
}else{
next(false)
}
},
路由懒加载
实现1 import方式
{
path: '/about',
name: 'about',
component: () => import('你的路由页面路径')
}
实现2 异步组件方式
component: (resolve) => require(['你的路由页面路径'], resolve),
路由模式
ui库--vantui
mixin混入
全局混入---mixin
let demo={
methods:{
fun(){
console.log("你好")
}
}
}
export default demo
import Vue from 'vue'
import App from './App.vue'
import router from './router'
// 全局混入
import demo from "./mixin/demo.js"
// 全局混入
Vue.mixin(demo)
// 全局过滤器
// Vue.filter("xiaoming",(val)=>{
// if(val.length>6){
// return val.slice(0,5)+"..."
// }else{
// return val
// }
// })
// 全局过滤器
Vue.config.productionTip = false
new Vue({
router,
render: h => h(App)
}).$mount('#app')
局部混入---mixins
let demo={
methods:{
fun(){
console.log("你好")
}
}
}
export default demo
扩展:mixin文件中定义字段/方法/钩子函数与引入到组件相应方法/字段/钩子函数的优先级顺序
ref
使用场景1--绑定到DOM元素身上
找到我
使用场景2--绑定到组件身上
我是home
$set
数据变试图不会改变
{{obj.age}}
数据变试图不会改变
{{obj.age}}
动态组件
动态组件
动态组件
keep-alive--保存组件的状态
使用
keep-alive 属性与钩子
$nexttick
什么时候用
nexttick
{{ text }}
PNPM
安装
npm install -g pnpm
vite
Vite 和 Webpack 区别
1、搭建第一个Vite项目
pnpm create vite
felixlu 2206 $ npm init vite
Need to install the following packages:
create-vite
Ok to proceed? (y) y
✔ Project name: … vue3-basics
? Select a framework: › - Use arrow-keys. Return to submit.
? Select a framework: › - Use arrow-keys. Return to submit.
? Select a framework: › - Use arrow-keys. Return to submit.
? Select a framework: › - Use arrow-keys. Return to submit.
? Select a framework: › - Use arrow-keys. Return to submit.
✔ Select a framework: › vue
✔ Select a variant: › vue
Scaffolding project in /Users/felixlu/Desktop/workspace/st-test/2206/vue3-basics...
Done. Now run:
cd vue3-basics
npm install
npm run dev
import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'
import { resolve } from 'path'
// https://vitejs.dev/config/
export default defineConfig({
plugins: [vue()],
resolve: {
alias: {
'@': resolve(__dirname, './src')
}
}
})
路由使用
1.创建文件夹用来容纳 页面views与路由规则router
2. 新建路由页面
3.配置路由规则--与下载路由
pnpm install --save vue-router@4
import { createRouter, createWebHistory } from 'vue-router'
const routes = [
{
path: '/home',
name: 'home',
component: () => import('../views/HomeView.vue')
},
{
path: '/shop',
name: 'shop',
component: () => import('../views/ShopView.vue')
},
{
path: '/user',
name: 'user',
component: () => import('../views/UserView.vue')
}
]
const router = createRouter({
history: createWebHistory(),
routes
})
export default router
4 设置路由出口
5 注入路由
import { createApp } from 'vue'
import './style.css'
import App from './App.vue'
import router from './router'
createApp(App).use(router).mount('#app')
vuex
1.下载
pnpm install --save vuex
2,创建对应容纳文件夹 store
3.编写vuex相关内容
import { createStore } from 'vuex'
export default createStore({
state: {
},
getters: {
},
mutations: {
},
actions: {
},
modules: {
}
})
4.在项目中配置vuex
import { createApp } from 'vue'
import './style.css'
import App from './App.vue'
import router from './router'
import store from './store'
createApp(App).use(store).use(router).mount('#app')
pinia
与vuex有什么不同
安装(仅限于vue3)
pnpm install pinia yarn add pinia npm install --save pinia
pinia基本使用
import { createApp } from 'vue'
import './style.css'
import App from './App.vue'
// 引用
import {createPinia } from 'pinia'
// 创建Pinia实例
const Pinia = createPinia()
// 挂载pinia
createApp(App).use(Pinia).mount('#app')
import { defineStore } from 'pinia'
//
// 第一个参数是应用程序中 store 的唯一 id
export const useStore = defineStore('main', {
})
pinia--state
import { defineStore } from 'pinia'
// useStore 可以是 useUser、useCart 之类的任何东西
// 第一个参数是应用程序中 store 的唯一 id
export const useStore = defineStore('main', {
state:()=>{//创建状态
return {
text:"我是pinia的一个状态变量"
}
}
})
pinia--修改--action
import { defineStore } from 'pinia'
export const useStore = defineStore('main', {
state:()=>{
return {
text:"我是pinia的一个状态变量"
}
},
actions:{
updatetext(){//修改方法
this.text="我变了"
}
}
})
let update=()=>{
store.$patch({
//你要修改state的数据: 你要修改的value值
text:"我是$patch直接修改"
})
}
let update=()=>{
store.$patch((state)=>{
console.log(state)
state.text="你又被改了"
})
}
pinia--修改常见问题
pinia--重置数据--$reset
pinia--监听数据修改--$subscribe
pinia--处理异步操作--action
import { defineStore } from 'pinia'
import $http from "axios"
export const useStore = defineStore('main', {
state:()=>{
return {
text:"我是pinia的一个状态变量"
}
},
actions:{
async link(){
try {
let data= await $http({
url:"http://localhost:8888/userlist/demoget",
method:"get"
})
console.log(data.data.data.name)
this.text=data.data.data.name
} catch (error) {
console.log("失败")
}
}
}
})
pinia--getters
import { defineStore } from 'pinia'
export const useStoreb = defineStore('mainb', {
state:()=>{
return {
text:"abcdefg"
}
},
actions:{
},
getters:{
newtext():any{
return this.text.toUpperCase()
}
}
})
pinia--模块化
echarts--大数据可视化
使用
vue2使用echarts
我是vue3使用echarts
vue2使用echarts
我是vue3使用echarts
title配置项--标题组件,包含主标题和副标题
我是vue3使用echarts
tooltip---提示框组件
xAxis yAxis
series--系列
地图