vue 实站技巧总结
多个页面都使用的到方法,放在
vue.prototype
上会很方便刚接触
vue
的时候做过一件傻事,因为封装了一个异步请求接口post
,放在post.js
文件里面,然后在每个需要使用异步请求的页面引入import port from './xxxx/xxxx/post'
复制代码如果只是这样,还没什么,我们可以写好一个页面以后再复制,可以保证每个页面都有上面的语句。但是如果每个文件所在的目录层级不一样呢?
// 假设正常是这样
import port from '../xxxx/xxxx/post'
// 目录加深一级,就变成这样
import port from '../../xxxx/xxxx/post'
// 再加深一级的样子
import port from '../../../xxxx/xxxx/post'
复制代码当然,这个时候,我们可以用 别名
@/xxxx/post
,但是还是少不了要每个页面引用。 那我们来看看,用vue.prototype
有多方便? 首先,你得在vue
的入口文件(vue-cli
生成的项目的话,默认是/src/main.js
)里面做如下设置import port from './xxxx/xxxx/post' vue.prototype.$post = post
复制代码这样,我们就可以在所有的
vue
组件(页面)里面使用this.post()
方法了,就像vue
的亲儿子一样tip: 把方法挂在到
prototype
上的时候,最好加一个$
前缀,避免跟其他变量冲突til again: 不要挂载太多方法到
prototype
上,只挂载一些使用频率非常高的需要响应的数据,在获取到接口数据的时候,先设置
大家有没有很经常碰到这样都一种情况,在循环列表的时候,我们需要给列表项一个控制显示的属性,如 是否可删除,是否已选中等等,而后端接口一般不会返回这种字段,因为这属于纯前端展示的,跟后端没啥关系,比如后端给的数据如下
[
{name: 'abc', age: 18},
{name: 'def', age: 20},
{name: 'ghi', age: 22},
]
复制代码我们不妨假设以上数据为学生列表
然后我们需要渲染这个列表,在每一项后面显示一个勾选按钮,如果用户打勾,则这个按钮是绿色,默认这个按钮是灰色,这个时候,上表是没有满足这个渲染条件的数据,而如果我们在用户打勾的时候,再去添加这个数据的话,正常的做法是无法及时响应的。
如果我们在获取到数据的时候,先给数组的每一项都加一个是否打勾的标示,就可以解决这个问题,我们假设我们获取到的数据是
res.list
res.list.map(item => {
item.isTicked = false
})
复制代码这么做的原理是
vue
无法对不存在的属性作响应,所以我们在获取到数据的时候,先把需要的属性加上去,然后在赋值给data
, 这样data
接收到数据的时候,已经是存在这个属性了,所以会响应。当然还有其他方法可以实现。不过对于一个强迫症来说,我还是比较倾向于这种做法封装全局基于
promise
的异步请求方法看过很多项目的源码,发现大部分的异步请求都是直接使用
axios
之类的方法,如下axios({
method: 'post',
url: '/user/12345',
data: {
firstName: 'Fred',
lastName: 'Flintstone'
}
})
.then(function (response) {
console.log(response);
})
.catch(function (error) {
console.log(error);
});
复制代码如果有跨域,或者需要设置
http
头等,还需要加入更多的配置,而这些配置,对于同一个项目来说,基本都是一样的,不一样的只有url
跟参数,既然这样,那我吗为什么不把它封装成一个方法呢?function post (url,param) {
return new Promise((resolve, reject) => {
axios({
method: 'post',
url: url,
data: param
})
.then(function (response) {
resolve(response);
})
.catch(function (error) {
reject(error);
});
})
} 复制代码再结合第一点,我们就可以再任意
vue
实例中这样使用let param = {
firstName: 'Fred',
lastName: 'Flintstone'
}
this.post('/user/12345',param)
.then(...)
.catch(...)
复制代码有没有比原始的简单很多呢?如果你的项目支持
async
await
,还可以这样用let param = {
firstName: 'Fred',
lastName: 'Flintstone'
}
let res = await this.post('/user/12345',param)
console.log(res) // res 就是异步返回的数据 复制代码tip:
await
关键字必须在被 async 修饰的函数里面使用
如果你觉得有时候,你真的需要父子组件共享一个值,不如试试传个引用类型过去
vue
的父子组件传值,有好多种方法,这里就不一一列举了,但是今天我们要了解的,是利用javascript
的引用类型特性,还达到另一种传值的目的假设有这么一个需求,父组件需要传 3 个值到子组件,然后再子组件里面改动后,需要立马再父组件上作出响应,我们通常的做法上改完以后,通过
this.$emit
发射事件,然后再父组件监听对应的事件,然而这么做应对一两个数据还好,如果传的数据多了,会累死人。 我们不妨把这些要传递的数据,包再一个对象/数组 里面,然后在传给子组件<subComponent :subData="subData"></subComponent>
复制代码data () {
return {
subData: {
filed1: 'field1',
filed2: 'field2',
filed3: 'field3',
filed4: 'field4',
filed5: 'field5',
}
}
}
复制代码这样,我们在子组件里面改动
subData
的内容,父组件上就能直接作出响应,无需this.$emit
或vuex
而且如果有其他兄弟组件的话,只要兄弟组件也有绑定这个subData
,那么兄弟组件里面的subData
也能及时响应tip: 首先,这么做我个人上感觉有点不符合规范的,如果没有特别多的数据,还是乖乖用
this.$emit
吧,其次,这个数据需要有特定的条件才能构造的出来,并不是所有情况都适用。异步请求的参数在
data
里面构造好,用一个对象包起来,会方便很多有做过类似
ERP
类型的系统的同学,一定碰到过这样的一个场景,一个列表,有 N 个过滤条件,这个时候通常我们这么绑定<input type="text" v-model="field1">
<input type="text" v-model="field2">
<input type="text" v-model="field3">
....
<input type="text" v-model="fieldn">
复制代码data () {
return {
field1: 'value1',
field2: 'value2',
field3: 'value3',
...
fieldn:'valuen'
}
}
复制代码然后提交数据的时候这样:
var param = {
backend_field1: this.field1,
backend_field2: this.field2,
backend_field3: this.field3,
...
backend_fieldn: this.fieldn
}
this.post(url,param)
复制代码如你看到的,每次提交接口,都要去构造参数,还很容易遗漏,我们不妨这样:先去接口文档里面看一下后端需要的字段名称,然后
<input type="text" v-model="queryParam.backend_field1">
<input type="text" v-model="queryParam.backend_field2">
<input type="text" v-model="queryParam.backend_field3">
....
<input type="text" v-model="queryParam.backend_fieldn">
``` ```javascript
data () {
return {
queryParam:{
backend_field1: 'value1'
backend_field2: 'value2'
backend_field3: 'value3'
...
backend_fieldn: 'valuen'
}
}
}
```
然后提交数据的时候这样:
```javascript
this.post(url,this.queryParam)
```
是的,这样做也是有局限性的,比如你一个数据在 2 个地方共用,比如前端组件绑定的是一个数组,你需要提交给后端的是 2 个字符串(例:`element ui` 的时间控件),不过部分特殊问题稍微处理一下,也比重新构建一个参数简单不是吗? 复制代码data
里面的数据多的时候,给每个数据加一个备注,会让你后期往回看的时候很清晰续上一点,
data
里面有很多数据的时候,可能你写的时候是挺清晰的,毕竟都是你自己写的东西,可是过了十天半个月,或者别人看你的代码,相信我,不管是你自己,还是别人,都是一头雾水(记忆力超出常人的除外),所以我们不妨给每个数据后面加一个备注data () {
return {
field1: 'value1', // 控制xxx显示
field2: 'value2', // 页面加载状态
field3: [], // 用户列表
...
fieldn: 'valuen' // XXXXXXXX
}
}
复制代码逻辑复杂的内容,尽量拆成组件
假设我们有一个这样的场景:
<div>
<div>姓名:{{user1.name}}</div>
<div>性别:{{user1.sex}}</div>
<div>年龄:{{user1.age}}</div>
...此处省略999个字段...
<div>他隔壁邻居的阿姨家小狗的名字:{{user1.petName}}</div>
</div>
<-- 当然,显示中我们不会傻到不用 v-for,我们假设这种情况无法用v-for -->
<div>
<div>姓名:{{user2.name}}</div>
<div>性别:{{user2.sex}}</div>
<div>年龄:{{user2.age}}</div>
...此处省略999个字段...
<div>他隔壁邻居的阿姨家小狗的名字:{{user2.petName}}</div>
</div>
复制代码这种情况,我们不妨把[用户]的代码,提取到一个组件里面: 假设如下代码,在
comUserInfo.vue
<template>
<div>
<div>姓名:{{user.name}}</div>
<div>性别:{{user.sex}}</div>
<div>年龄:{{user.age}}</div>
...此处省略999个字段...
<div>他隔壁邻居的阿姨家小狗的名字:{{user.petName}}</div>
</div>
</template> <script >
export default {
props:{
user:{
type:Object,
default: () => {}
}
}
}
</script>
复制代码然后原来的页面可以改成这样(省略掉导入和注册组件,假设注册的名字是
comUserInfo
):<comUserInfo :user="user1"/>
<comUserInfo :user="user2"/>
复制代码这样是不是清晰很多?不用看注释,都能猜的出来,这是2个用户信息模块, 这样做,还有一个好处就是出现错误的时候,你可以更容易的定位到错误的位置。
如果你只在子组件里面改变父组件的一个值,不妨试试
$emit('input')
,会直接改变v-model
我们正常的父子组件通信是 父组件通过
props
传给子组件,子组件通过this.$emit('eventName',value)
通知父组件绑定在@eventName
上的方法来做相应的处理。 但是这边有个特例,vue
默认会监听组件的input
事件,而且会把子组件里面传出来的值,赋给当前绑定到v-model
上的值正常用法 - 父组件
<template>
<subComponent :data="param" @dataChange="dataChangeHandler"></subComponent>
</template> <script >
export default {
data () {
return {
param:'xxxxxx'
}
},
methods:{
dataChangeHandler (newParam) {
this.param = newParam
}
}
}
</script>
复制代码正常用法 - 子组件
<script >
export default {
methods:{
updateData (newParam) {
this.$emit('dataChange',newParam)
}
}
}
</script>
复制代码利用默认
input
事件 - 父组件<template>
<subComponent v-model="param"></subComponent>
</template>
复制代码利用默认
input
事件 - 子组件<script >
export default {
methods:{
updateData (newParam) {
this.$emit('input',newParam)
}
}
}
</script>
复制代码这样,我们就能省掉父组件上的一列席处理代码,
vue
会自动帮你处理好tip: 这种方法只适用于改变单个值的情况,且子组件对父组件只需简单的传值,不需要其他附加操作(如更新列表)的情况。
conponents
放在Vue options
的最上面不知道大家有没有这样的经历: 导入组件,然后在也页面中使用,好的,报错了,为啥?忘记注册组件了,为什么会经常忘记注册组件呢?因为正常的一个
vue
实例的结构大概是这样的:import xxx form 'xxx/xxx'
export default {
name: 'component-name',
data () {
return {
// ...根据业务逻辑的复杂程度,这里省略若干行
}
},
computed: {
// ...根据业务逻辑的复杂程度,这里省略若干行
},
created () {
// ...根据业务逻辑的复杂程度,这里省略若干行
},
mounted () {
// ...根据业务逻辑的复杂程度,这里省略若干行
},
methods () {
// ...根据业务逻辑的复杂程度,这里省略若干行
},
}
复制代码我不知道大家正常是把
components
属性放在哪个位置,反正我之前是放在最底下,结果就是导致经常犯上述错误。后面我把
components
调到第一个去了import xxx form 'xxx/xxx'
export default {
components: {
xxx
},
// 省略其他代码
}
复制代码从此以后,妈妈再也不用担心我忘记注册组件了,导入和注册都在同一个位置,想忘记都难。
大部分情况下,生命周期里面,不要有太多行代码,可以封装成方法,再调用
看过很多代码,包括我自己之前的,在生命周期里面洋洋洒洒的写了一两百行的代码,如:把页面加载的时候,该做的事,全部写在
created
里面,导致整个代码难以阅读,完全不知道你在页面加载的时候,做了些什么, 这个时候,我们不妨把那些逻辑封装成方法,然后在生命周期里面直接调用:created () {
// 获取用户信息
this.getUserInfo()
// 获取系统信息
this.getSystemInfo()
// 获取配置
this.getConfigInfo()
},
methods:{
// 获取用户信息
getUserInfo () {...},
// 获取系统信息
getSystemInfo () {...},
// 获取配置
getConfigInfo () {...},
}
复制代码这样是不是一眼就能看的出,你在页面加载的时候做了些什么?
tip: 这个应该算是一个约定俗成的规范吧,只是觉得看的比较多这样写的,加上我自己初学的时候,也这么做了,所以写出来,希望新入坑的同学能避免这个问题
少用
watch
,如果你觉得你好多地方都需要用到watch
,那十有八九是你对vue
的API
还不够了解vue
本身就是一个数据驱动的框架,数据的变动,能实时反馈到视图上去,如果你想要根据数据来控制试图,正常情况一下配合computed
服用就能解决大部分问题了,而视图上的变动,我们一般可以通过监听input
change
等事件,达到实时监听的目的, 所以很少有需求使用到watch
的时候,至少我最近到的十来个项目里面,是没有用过watch
当然,并不是说watch
是肯定没用处,vue
提供这个api,肯定是有他的道理,也有部分需求是真的需要用到的,只是我觉得应该很少用到才对,如果你觉得到处都得用到的话, 那么我觉得 十有八九你应该多去熟悉一下computed
和vue
的其他api
了
vue 实站技巧总结的更多相关文章
- 使用VUE实现在table中文字信息超过5个隐藏,鼠标移到时弹窗显示全部
使用VUE实现在table中文字信息超过5个隐藏,鼠标移到时弹窗显示全部 <template> <div> <table> <tr v-for="i ...
- Vue + Element 小技巧
说是小技巧 ,其实就是本人 就是一个小菜比 .如有大佬可以纠正,或者再救救我这个小菜比 跪谢 1.当后台返回一个字段需要根据不同字段内容在表格内显示相对应的文字(字段内容是死的,表格内需要显示对 ...
- VUE基础实用技巧
Vue以前听说过,有了解过一点.当时还在热衷于原生JavaScript去写一些方法的封装,不是为啥,就感觉这样很帅,后面多多少少接触了一些JQuery的用法,到现在为止,JavaScript原生封装的 ...
- vue的开发技巧
1.监听组件的生命周期 比如有父组件 Parent和子组件 Child,如果父组件监听到子组件挂载 mounted就做一些逻辑处理,常规的写法可能如下: // Parent.vue <Child ...
- vue 优化小技巧 之 require.context()
1.require.context() 回忆一下 当我们引入组件时 第一步 创建一个子组件 第二步 import ... form ... 第三步 components:{..} 第四步 页面使用 & ...
- 在vuex的mutations中使用vue的小技巧
问题: 在vuex组件中的mutations属性中的定义的函数,有时会要用到vue这个对象.正常在其他的地方使用是通过this这个变量来获取,但是在mutations定义的函数中this指定的是Vue ...
- GO后端开发+VUE实列
因为我是从java转到go,代码结构跟我之前用java的很像 在这里只浅显的实战运用,没有过多理论讲解 工作环境:IDE:Goland , Go 1.17.7 框架 Gin+Gorm ,前端VUE 这 ...
- Maven实站读后感
这本书是一本非常经典的Maven教程,通俗易懂,同时介绍的东西十分实用,在工作上都能用到. 以前在公司里面需要要问同时的有关的Maven的问题,都可以自己解决了. 除了最基本的,以后自己可能要用到的: ...
- vue页面操作技巧
// this.$router.push({ path: "https://www.baidu.com/"}); // POST请求的时候 // this.$router.push ...
随机推荐
- python核心语法
一.语句和语法 #:注释 \:转译回车,继续上一行,在一行语句较长的情况下可以使用其来切分成多行,因其可读性差所以不建议使用 ::将两个语句连接到一行,可读性差,不建议使用 ::将代码的头和体分开 语 ...
- csdn 站点使用
大学时使用csdn下载软件资源,最终csdn成为一个it中很重要站点,csdn的运营方式值得思考.
- 使用C#的is、as操作符来转型
is检查对象是否兼容于指定类型,返回Boolean值true或false.使用is永远不会抛出异常. 例:Object o=new Object(); bool b1=(o is Object);// ...
- 关于visual assist x插件不能用的解决方案
打開VS莫名其妙地彈出下面的錯誤框: "the security key for this program currently stored on your system does not ...
- CodeForces - 1016D 补零思想
题目连接: https://vjudge.net/problem/1753263/origin 其实这道题跟行列式里的分块发有点类似,但也是类似罢了. 主要的思想是每一行,每一列的第一行(或者最后一行 ...
- 2018 EC-Final 部分题解 (A,J)
目录 The 2018 ICPC Asia-East Continent Final A.Exotic - Ancient City(思路 并查集) J.Philosophical - Balance ...
- 纯CSS实现展开列表
效果预览 以下为源码 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> &l ...
- (转)java创建对象的步骤
关于对象的创建过程一般是从new指令(我说的是JVM的层面)开始的(具体请看图1),JVM首先对符号引用进行解析,如果找不到对应的符号引用,那么这个类还没有被加载,因此JVM便会进行类加载过程.符号引 ...
- 杭电acm2059-龟兔赛跑 java
一看题就知道是动态规划,不过这要看下如何设置变化数组了 先分析这道题:兔子到达终点的时间时固定的,因此只需要考虑乌龟了,乌龟骑电车和骑自行车的时间,然后计算,因为中间有N个充电站,可以看做N个点(到起 ...
- 100405之python程序安装
Python的安装 注意安装的path和.exe可执行文件的命名方式 例如py2或者py3 sublime的安装 一.Sublime Text 3安装(64位Windows操作系统): 1.双击 Su ...