1. Vue基础
01 初始vue
1. 想让vue工作,就必须创建一个Vue实例,且要传入一个配置对象;
2. root容器里的代码依然符合html规范,只不过混入了一些特殊的Vue语法;
3. root容器里的代码被称为【Vue模板】;
4. Vue实例和容器是一一对应的;
5. 真实开发中只有一个Vue实例并且会配合着组件一起使用;
6. {{xxx}}中的xxx要写js表达式,且xxx可以自动读取到data中的所对应的属性;
7. 一旦data中的数据发生改变,那么模板中用到该数据的地方也会自动更新;
注意区分:js表达式 和 js代码(又称为“语句”)
1. 表达式:一个表达式会产生一个值,可以放在任何一个需要值的地方,如:
Date.now() // 获取当前时间戳
a+b
demo(1)
x === y ? 'a' : 'b'
2. js代码(语句)
if(){}
for(){}
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<script src="../js/vue.js"></script>
<title>初始vue</title>
</head>
<body>
<div id="root">
<h1>hello, {{name.toUpperCase()}}, {{address}}</h1>
</div>
<script>
Vue.config.productionTip = false;
new Vue({
el: '#root',
data: {
name: 'badi',
address: "贵州花溪"
}
});
</script>
</body>
</html>
02 Vue模板语法
1. 插值语法:
功能:用于解析标签体内容;
写法:{{xxx}},xxx是js表达式,且可以直接读取到data中的所有属性。
2. 指令语法:
功能:用于解析标签(包括:标签属性、标签体内容、绑定事件...)
举例:v-bind:href="xxx" 或简写为 :href="xxx", xxx同样需要写为js表达式。且可以直接读取到data中所有的属性。
注:Vue中有很多的指令,且形式都是:v-???,此处我们只是拿v-bind举个例子。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>模板语法</title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="root">
<h1>插值语法</h1>
<h3>你好, {{name}}</h3>
<hr>
<h1>指令语法</h1>
<a v-bind:href="school.url">点击我去{{school.name}}学习1</a>
<a :href="school.url">点击我去{{school.name}}学习2</a>
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
name: 'zhangsan',
school: {
name: "尚硅谷",
url: 'http://www.atguigu.com/'
}
}
});
</script>
</body>
</html>
03 数据绑定
Vue当中有两种数据绑定的方式:
1. 单向绑定(v-bind):数据只能从data流向页面;
2. 双向绑定(v-model):数据不仅能从data流向页面,还可以从页面流向data
注:
3. 双向绑定一般都应用在表单类元素上(如:input、select等)
4. v-model:value可以简写为v-model,因为v-model默认收集的就是value的值。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="root">
单向的数据绑定:<input type="text" :value="name"><br>
双向的数据绑定:<input type="text" v-model="name"><br>
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
name: "尚硅谷"
}
});
</script>
</body>
</html>
04 el与data的两种写法
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实例了。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="root">
<h1>hello, {{name}}</h1>
</div>
<script>
Vue.config.productionTip=false;
const x = new Vue({
data: function(){
return {
name: 'zhangsan'
}
}
});
x.mount("#root");
</script>
</body>
</html>
05 MVVM模型
MVVM模型总结:
1. M模型(Model):data中的数据
2. V视图(View):模板代码
3. VM视图模型(ViewModel):Vue实例
观察发现:
1. data中所有的属性,最后都出现再了vm身上;
2. vm身上所有的属性以及Vue原型上所有属性,在Vue模板中都可以直接使用。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="root">
<h1>学校名称:{{name}}</h1>
<h1>学校地址:{{url}}</h1>
<h1>测试1:{{1+1}}</h1>
<h1>测试2:{{$options}}</h1>
</div>
<script>
Vue.config.productionTip=false;
const vm = new Vue({
el: "#root",
data:{
name: "尚硅谷",
url: "北京"
}
});
console.log(vm);
</script>
</body>
</html>
06 数据代理
06.1 回顾Object.defineproperty
<script>
let number = 20;
let person = {
name: 'zhangsan',
sex: 'male'
}
Object.defineProperty(person, 'age', {
get(){
console.log('有人读取age属性了');
return number;
},
set(value){
console.log('有人修改了age属性,且值为:', value);
number = value;
}
});
console.log(person);
</script>
06.2 何为数据代理
<script>
let obj = {x: 100};
let obj2 = {y: 200};
Object.defineProperty(obj2, 'x', {
get(){
return obj.x;
},
set(value){
obj.x = value;
}
});
</script>
06.3 Vue当中的数据代理
1. Vue中的数据代理:通过vm对象来代理data对象中属性的操作(读/写);
2. Vue中数据代理的好处:更加方便的操作data中的数据;
3. 基本原理:
通过Object.defineProperty()把data对象中所有属性添加到vm上;
为每一个添加到vm上的属性,都指定一个getter/setter,在getter/setter内部去操作data中对应的属性;
<body>
<div id="root">
<h1>学校名称:{{name}}</h1>
<h1>学校地址:{{address}}</h1>
</div>
<script>
Vue.config.productionTip=false;
const vm = new Vue({
el: "#root",
data: {
name: '尚硅谷',
address: '滨江区'
}
});
</script>
</body>
07 事件处理
07.1 事件的基本使用
事件的基本使用:
1. 使用v-on:xxx 或 @xxx 绑定事件,其中xxx是事件名称;
2. 时间的回调函数需要配置在methods对象中,最终会在vm上;
3. methods中配置的函数,不要使用箭头函数!否则this就不再是vm了;
4. methods中配置的函数,都是被Vue所管理的函数,this的只想是vm 或 组件实例对象;
5. @click="demo" 和 @click="demo($event)" 效果是一样的,但后者可以传指定的参数;
<body>
<div id="root">
<h1>欢迎来到{{name}}学习</h1>
<button @click="showInfo1">点我弹出提示信息1(不传递参数)</button>
<button @click="showInfo2(666, $event)">点我弹出提示信息2(传递参数)</button>
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
name: '尚硅谷'
},
methods:{
showInfo1(event){
alert("同学你好!");
},
showInfo2(number, event){
console.log(number);
console.log(event);
}
}
});
</script>
</body>
07.2 事件修饰符
Vue中的时间修饰符:
1. prevent:阻止默认事件(常用);
2. stop:阻止时间冒泡(常用);
3. once:事件只触发一次(常用);
4. self:只有event.target是当前操作的元素时才触发事件;
5. passive:事件的默认行为立即执行,无需等待时间回调执行完毕;
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
<script src="../js/vue.js"></script>
<style>
*{
margin-top: 5px;
}
#wrapper{
height: 100px;
background-color: #bfa;
line-height: 100px;
}
#box1, #box2{
padding: 5px;
}
#box1{
height: 100px;
background-color: #bfa;
}
#box2{
height: 40px;
background-color: orange;
}
#list{
height: 100px;
background-color: skyblue;
overflow: auto;
}
#list li{
margin: 20px;
}
</style>
</head>
<body>
<div id="root">
<h2>欢迎来到{{name}}学习</h2>
<a href="http://www.atguigu.com" @click.prevent="showInfo">点我提示信息</a>
<div id="wrapper" @click="wrapperInfo">
<a href="#" @click.stop.prevent="wrapperInner">去百度</a>
</div>
<button @click.once="showInfo">我只会触发一次</button>
<div id="box1" @click.capture="showMsg(1)">
div1
<div id="box2" @click="showMsg(2)">
div2
</div>
</div>
<div id="demo1" @click.self="showEvent">
div1
<div id="demo1" @click="showEvent">
div2
</div>
</div>
<ul id="list" @scroll="demo">
<li>1</li>
<li>2</li>
<li>3</li>
<li>4</li>
</ul>
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
name: '尚硅谷'
},
methods:{
showInfo(evnet){
alert("同学你好!");
},
wrapperInfo(){
alert("wrapper");
},
wrapperInner(){
alert("wrapper inner");
},
showMsg(msg){
console.log(msg);
},
showEvent(event){
console.log(event.target);
},
demo(){
for(i = 0; i < 1000000; i++){
console.log(i);
}
console.log('累坏了');
}
}
});
</script>
</body>
</html>
07.3 键盘事件
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去指定具体的按键(不推荐)
5.Vue.config.keyCodes.自定义键名 = 键码,可以去定制按键别名(不推荐使用,因为一般情况下默认的都已经够使用了)
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="root">
<h1>键盘事件测试</h1>
<input type="text" placeholder="请输入你的字符" @keyup.ctrl.y="demo">
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
},
methods:{
demo(event){
console.log(event.key,event.keyCode);
}
}
});
</script>
</body>
</html>
08 计算属性
08.1 姓名按钮_插值语法实现
<div id="root">
姓:<input type="text" v-model="firstName"><br>
名:<input type="text" v-model="lastName"><br>
全名:<span>{{firstName.substring(0,3)}}-{{lastName}}</span>
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
firstName: '张',
lastName: '三'
}
});
</script>
08.2 姓名按钮_methods实现
<div id="root">
姓:<input type="text" v-model="firstName"><br>
名:<input type="text" v-model="lastName"><br>
全名:<span>{{fullName()}}</span>
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
firstName: '张',
lastName: '三'
},
methods:{
fullName(){
return this.firstName + '-' + this.lastName;
}
}
});
</script>
08.3 姓名按钮_计算属性实现
计算属性:
1.定义:要用的属性不存在,要通过已有属性计算得来。
2.原理:底层借助了Objcet.defineProperty方法提供的getter和setter。
3.get函数什么时候执行?
(1).初次读取时会执行一次。
(2).当依赖的数据发生改变时会被再次调用。
4.优势:与methods实现相比,内部有缓存机制(复用),效率更高,调试方便。
5.备注:
1.计算属性最终会出现在vm上,直接读取使用即可{{fullName}}。
2.如果计算属性要被修改,那必须写set函数去响应修改(否则会报错),且set中要引起计算时依赖属性数据发生改变后,该计算属性的值才有效。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="root">
姓:<input type="text" v-model="firstName"><br>
名:<input type="text" v-model="lastName"><br>
全名:<span>{{fullName}}</span> <br><br>
</div>
<script>
Vue.config.productionTip=false;
const vm = new Vue({
el: "#root",
data:{
firstName: '张',
lastName: '三'
},
computed:{
fullName:{
get(){
return this.firstName + '-' + this.lastName;
},
set(value){
const temp = value.split("-");
this.firstName = temp[0];
this.lastName = temp[1];
}
}
}
});
</script>
</body>
</html>
08.4 计算属性的简写
Vue.config.productionTip=false;
const vm = new Vue({
el: "#root",
data:{
firstName: '张',
lastName: '三'
},
computed:{
fullName(){
return this.firstName + "-" + this.lastName;
}
}
});
09 监视属性
09.1 天气案例
如果methods中的方法实现的功能很简单,如这里只是对当前的天气取反,那么可以直接在@click后面进行编写,即@后面可以编写一些简单的js语句。
但是不建议编写较为复杂的语句。
@xxx="yyy";xxx事件名,yyy执行的语句。
<button @click="isHot = !isHot">切换天气</button>
以下的代码会报错:
<button @click="alert(1)">切换天气</button>
因为被Vue所管辖的模板当中,所有的方法、变量都是在vue实例当中取寻找,如果找不到则报错,而不会跳转到上一级作用域当中去寻找(window)。
<div id="root">
<h2>今天天气很{{info}}</h2>
<button @click="changeWeather">切换天气</button>
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
isHot: true
},
computed:{
info(){
return this.isHot ? '炎热' : '凉爽';
}
},
methods: {
changeWeather(){
this.isHot = !this.isHot;
}
},
});
</script>
09.2 天气案例_属性监视
监视属性watch:
1.当被监视的属性变化时, 回调函数自动调用, 进行相关操作
2.监视的属性必须存在(不存在也不会报错,但是没有意义),才能进行监视!!
3.监视的两种写法:
(1).new Vue时传入watch配置
(2).通过vm.$watch监视
<div id="root">
<h2>今天天气很{{info}}</h2>
<button @click="changeWeather">切换天气</button>
</div>
<script>
Vue.config.productionTip=false;
const vm = new Vue({
el: "#root",
data:{
isHot: true
},
computed:{
info(){
return this.isHot ? '炎热' : '凉爽';
}
},
methods: {
changeWeather(){
this.isHot = !this.isHot;
}
},
});
vm.$watch('isHot', {
immediate: true,
handler(newValue, oldValue){
console.log("handler()被执行了:" + newValue, oldValue);
}
});
</script>
09.4 天气案例_深度监视
深度监视:
1. Vue中的watch默认不监视对象内部值的改变(默认只能监视一层);
2. 配置deep:true可以检测对象内部值的改变(可以监视多层);
注:
1. Vue自身可以检测对象内部值的改变,但Vue提供的watch默认不可以;
2. 使用watch时根据数据的具体结构,决定是否采用深度监视;
<div id="root">
<h2>今天天气很{{info}}</h2>
<button @click="changeWeather">切换天气</button>
<hr>
<h2>a的值是{{numbers.a}}</h2>
<button @click="numbers.a++">点我给a+1</button>
<h2>b的值是{{numbers.b}}</h2>
<button @click="numbers.b++">点我给b+1</button>
</div>
<script>
Vue.config.productionTip=false;
const vm = new Vue({
el: "#root",
data:{
isHot: true,
numbers: {
a:1,
b:1
}
},
computed:{
info(){
return this.isHot ? '炎热' : '凉爽';
}
},
methods: {
changeWeather(){
this.isHot = !this.isHot;
}
},
watch:{
isHot:{
immediate: true,
handler(newValue, oldValue){
console.log("handler()被执行了:" + newValue, oldValue);
}
},
numbers: {
deep: true,
handler(){
console.log("numbers属性变化了");
}
}
}
});
</script>
09.5 天气案例_深度监视_简写属性
<div id="root">
<h2>今天天气很{{info}}</h2>
<button @click="changeWeather">切换天气</button>
</div>
<script>
Vue.config.productionTip=false;
const vm = new Vue({
el: "#root",
data:{
isHot: true,
},
computed:{
info(){
return this.isHot ? '炎热' : '凉爽';
}
},
methods: {
changeWeather(){
this.isHot = !this.isHot;
}
},
});
vm.$watch('isHot', {
handler(newValue, oldValue){
console.log("handler()被执行了:" + newValue, oldValue);
}
});
</script>
09.5 姓名案例_watch实现
computed和watch之间的区别:
1.computed能完成的功能,watch都可以完成。
2.watch能完成的功能,computed不一定能完成,例如:watch可以进行异步操作。
两个重要的小原则:
1.所被Vue管理的函数,最好写成普通函数,这样this的指向才是vm 或 组件实例对象。
2.所有不被Vue所管理的函数(如定时器的回调函数、ajax的回调函数等、Promise的回调函数),最好写成箭头函数,
这样this的指向才是vm 或 组件实例对象。
3. 其实以上两个步骤的目的就是想让函数的this指向vm。
<div id="root">
姓:<input class="firstName" type="text" v-model="firstName"><br>
名:<input class="lastName" type="text" v-model="lastName"><br>
全名:<span>{{fullName}}</span> <br><br>
</div>
<script>
Vue.config.productionTip=false;
const vm = new Vue({
el: "#root",
data:{
firstName: '张',
lastName: '三',
fullName: '姓-名'
},
watch:{
firstName(val){
setTimeout(() => {
this.fullName = val + '-' + this.lastName;
}, 1000);
},
lastName(val){
this.fullName = this.firstName + '-' + val;
}
}
});
</script>
10 绑定样式
绑定样式:
1. class样式
写法:class="xxx" xxx可以是字符串、对象、数组。
字符串写法适用于:类名不确定,要动态获取。
对象写法适用于:要绑定多个样式,个数不确定,名字也不确定。
数组写法适用于:要绑定多个样式,个数确定,名字也确定,但不确定用不用。
2. style样式(了解)
:style="{fontSize: xxx}"其中xxx是动态值。
:style="[a,b]"其中a、b是样式对象。
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8" />
<title></title>
<style>
.basic{
width: 400px;
height: 100px;
border: 1px solid black;
}
.happy{
border: 4px solid red;;
background-color: rgba(255, 255, 0, 0.644);
background: linear-gradient(30deg,yellow,pink,orange,yellow);
}
.sad{
border: 4px dashed rgb(2, 197, 2);
background-color: gray;
}
.normal{
background-color: skyblue;
}
.atguigu1{
background-color: yellowgreen;
}
.atguigu2{
font-size: 30px;
text-shadow:2px 2px 10px red;
}
.atguigu3{
border-radius: 20px;
}
</style>
<script type="text/javascript" src="../js/vue.js"></script>
</head>
<body>
<div id="root">
<div class="basic" :class="mood" @click="changeMood">{{name}}</div>
<br><br>
<div class="basic" :class="classArr">{{name}}</div>
<br><br>
<div class="basic" :class="classObj">{{name}}</div>
<br><br>
<div class="basic" :style="styleObj">{{name}}</div>
<br><br>
<div class="basic" :style="styleArr">{{name}}</div>
</div>
</body>
<script type="text/javascript">
Vue.config.productionTip = false;
const vm = new Vue({
el: '#root',
data:{
name: 'my mood',
mood: 'normal',
classArr: ['atguigu1', 'atguigu2', 'atguigu3'],
classObj:{
atguigu1: false,
atguigu2: false
},
styleObj:{
fontSize: '40px',
color: 'blue',
},
styleObj2:{
backgroundColor: '#bfa'
},
styleArr:[
{
fontSize: '40px',
color: 'blue',
},
{
backgroundColor: '#bfa'
}
]
},
methods: {
changeMood(){
const arr = ['happy', 'normal', 'sad'];
this.mood = arr[Math.floor(Math.random()*3)];
}
},
});
</script>
</html>
11 条件渲染
条件渲染:
1.v-if
写法:
(1).v-if="表达式"
(2).v-else-if="表达式"
(3).v-else="表达式"
适用于:切换频率较低的场景。
特点:不展示的DOM元素直接被移除。
注意:v-if可以和:v-else-if、v-else一起使用,但要求结构不能被“打断”。
2.v-show
写法:v-show="表达式"
适用于:切换频率较高的场景。
特点:不展示的DOM元素未被移除,仅仅是使用样式隐藏掉,底层为display:none
3.备注:使用v-if的时,元素可能无法获取到,而使用v-show一定可以获取到。
<div id="root">
<h2>n的值是:{{n}}</h2>
<button @click="n++">点我n的值会加1</button>
<template v-if="n===1">
<h2>html</h2>
<h2>css</h2>
<h2>js</h2>
</template>
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
name: '尚硅谷',
n: 0
}
});
</script>
12 列表渲染
12.1 基本列表
v-for指令:
1.用于展示列表数据
2.语法:v-for="(item, index) in xxx" :key="yyy",这里的key要求是唯一的;
3.可遍历:数组、对象、字符串(用的很少)、指定次数(用的很少)
<div id="root">
<h2>人员列表</h2>
<ul>
<li v-for="(person, index) in persons" :key="person.id">
{{person.name}}-{{person.age}}--{{index}}
</li>
</ul>
<h2>汽车信息</h2>
<ul>
<li v-for="(car, key) of cars" :key="key">
{{car}}---{{key}}
</li>
</ul>
<h2>遍历字符串(使用的少)</h2>
<ul>
<li v-for="(char,index) of strs" :key="index">
{{char}}---{{index}}
</li>
</ul>
<h2>直接遍历次数(使用的少)</h2>
<ul>
<li v-for="(number, index) of 10" :key="index">
{{number}}---{{index}}
</li>
</ul>
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
persons:[
{id: '001',name: '张三', age:18},
{id: '002',name: '李四', age:19},
{id: '003',name: '王五', age:16},
],
cars:{
name: 'audi a6',
price: '50w',
color: 'black'
},
strs: 'hello'
}
});
</script>
12.2 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是没有问题的。
5. 值得注意的是:如果么有指定key,key的默认值为index;
<div id="root">
<h2>人员列表</h2>
<button @click.once="add">添加人员信息</button>
<ul>
<li v-for="(person, index) in persons" :key="person.id">
{{person.name}}-{{person.age}}--{{index}}
<input type="text">
</li>
</ul>
</div>
<script>
Vue.config.productionTip=false;
new Vue({
el: "#root",
data:{
persons:[
{id: '001',name: '张三', age:18},
{id: '002',name: '李四', age:19},
{id: '003',name: '王五', age:16},
],
cars:{
name: 'audi a6',
price: '50w',
color: 'black'
},
strs: 'hello'
},
methods:{
add(){
const liu = {
id:'004',
name: '老刘',
age: 40
};
this.persons.unshift(liu);
}
}
});
</script>
|