Heim >Web-Frontend >View.js >Was kann die berechnete Vue-Eigenschaft bewirken? Kurze Analyse von Anwendungsszenarien
Was können von Vue berechnete Eigenschaften? Auf welche Szenarien kann es angewendet werden? Der folgende Artikel wird Ihnen helfen, die von Vue berechneten Eigenschaften zu verstehen und sie mit Methoden und Uhren zu vergleichen. Ich hoffe, er wird Ihnen hilfreich sein!
Die berechneten Attribute in Vue sind die Berechnungen aller Attribute, und diese Berechnungen erfolgen alle in Filterrichtung. Durch die kontinuierliche Änderung der Daten werden unterschiedliche Werte berechnet und unterschiedliche Betriebsmethoden berechnet. In Vue werden häufige Szenarien verwendet, in denen berechnete Eigenschaften verwendet werden:
Ausdrücke in Vorlagen
Ausdrücke, die im Attribut v-bind
ausgeführt werden können. (Teilen von Lernvideos: vue-Video-Tutorial) v-bind
里可以进行的表达式。(学习视频分享:vue视频教程)
指令中可以进行的表达式
以上三者的主要优势是简洁。如果只是一个小操作,比如说一些简单的数值++
,字符拼接,三元表达式,那么使用相当方便。当然,以上方式也有自己的劣势。一旦要处理的逻辑复杂,代码量就会变得大得多,而且难于维护。比如说用到if语句来控制流程。那么这个时候可能会想到用filter
,咱们来看一个简单的示例。
<div id="app"> <button @click="count++">{{count + '分'}}</button> <div> <input v-model="message" /> </div> <p>{{ message.split(' ').reverse().join(' ') }}</p> </div>
let app = new Vue({ el: '#app', data () { return { count: 0, message: '' } } })
效果如下:
从上面的示例中我们可以看出:
我们在模板内使用字符串拼接,{{count + '分'}}
通过v-bind
指定绑定了一个click
事件,它只做了一件事情,就是count++
,button
每点击一次,count
值增加1
,注意,这里并没有使用methods
input
通过v-model
做了数据双向绑定,绑定了message
,并且在p
标签中对message
字符串进行反转。
在p
标签中对message
进行计算转换的时候,是不是觉得语义不是很强烈,那么用什么办法更好呢?这就需要我们使用到前面提到的filter。
我们可以使用filter
来修改前面的示例。事实下,在Vue中使用filter
具有自己的优势,也具有自己的劣势:
优势:filter
给我们用于计算和过滤一些模板表达式和v-bind
属性表达式一些弊端的地方进行计算,他们会返回当前计算的值,可以进行传参在多地方共用这个过滤方法
劣势
:如果我们要计算多个数据不同变化结合而成的地方,那么filter
就比较难过滤到了,本质上filter
就是一个一对一的行为,对单个数据进行过滤,可以进行传参,同方法,但不同参
把上面的示例,用filter
修改之后就像下面这样:
<div id="app"> <button @click="count++">{{count + '分'}}</button> <div> <input v-model="message" /> </div> <p>{{ message | reverseString }}</p> </div> let app = new Vue({ el: '#app', data () { return { count: 0, message: '' } }, filters: { reverseString (value) { if (!value) return '' value = value.split('').reverse().join('') return value } } })
在这个示例中,我们使用filter来实现,很明显代码量多了那么一点点,但整体的语意化就变得相当明显了,让人一看这里就要进行一些过滤计算,看到reverseString就知道是字符串反转。
前面说了这么多,其实我们都是为了给computed
的应用场景做一个铺垫。那问题又来了,computed
可以做些什么?又能应用于什么场景呢?
Vue中的computed
其实规避了模板语法和filter
两个所有的劣势,他的优势在于通过计算所有依赖的数据进行计算,然后返回一个值,记住可以依赖方法里所有的数据,只要一个数据发生变化,则会重新计算,来更新视图的改变。是不是很意思,迫切的想知道它是怎么玩。
咱们还是拿使用场景来说事吧。先给大家演示一个简单实用的应用场景,后面再做一个更好玩一点的应用场景。
大家是否还记得,我们在玩微博的时候,发微博会有一个字数限制,比如说只限输入140
++
, das Zusammenfügen von Zeichen und ternäre Ausdrücke, ist die Verwendung recht praktisch. Natürlich haben die oben genannten Methoden auch ihre eigenen Nachteile. Sobald die zu verarbeitende Logik komplex ist, wird die Codemenge viel größer und schwieriger zu warten. Wenn beispielsweise Anweisungen zur Steuerung des Prozesses verwendet werden. Dann denken Sie vielleicht an die Verwendung von filter
. Schauen wir uns ein einfaches Beispiel an. 🎜<div id="app"> <div class="twitter"> <img :src="imgUrl" / alt="Was kann die berechnete Vue-Eigenschaft bewirken? Kurze Analyse von Anwendungsszenarien" > <div class="content"> <textarea v-model="content" :maxlength="totalcount">有什么新鲜事情?</textarea> <p>您还可以输入{{ reduceCount }}字</p> </div> </div> </div> let app = new Vue({ el: '#app', data () { return { imgUrl: '//pbs.twimg.com/profile_images/468783022687256577/eKHcWEIk_normal.jpeg', totalcount: 140, // 总共只给输入140字 content: '' } }, computed: { reduceCount () { return this.totalcount - this.content.length } } })rrree🎜Der Effekt ist wie folgt: 🎜🎜🎜🎜Aus dem obigen Beispiel können wir sehen: 🎜🎜🎜🎜Wir verwenden String-Verkettung innerhalb der Vorlage,
{{count + 'minutes'}
🎜🎜🎜🎜Ein click
-Ereignis wird über v-bind
gebunden. Es macht nur eines, nämlich count++ code>. Jedes Mal, wenn auf die <code>Schaltfläche
geklickt wird, erhöht sich der count
-Wert um 1
. Beachten Sie, dass methods
hier nicht verwendet wird🎜🎜 🎜🎜input
hat eine bidirektionale Datenbindung über v-model
, gebundene message
und in p
Reverse durchgeführt die Zeichenfolge message
im Tag. 🎜🎜🎜Bei der Berechnung und Konvertierung von message
im p
-Tag haben Sie das Gefühl, dass die Semantik nicht sehr stark ist. Was ist also eine bessere Methode? Dazu müssen wir den zuvor erwähnten Filter verwenden. 🎜filter
verwenden, um das vorherige Beispiel zu ändern. Tatsächlich hat die Verwendung von filter
in Vue ihre eigenen Vor- und Nachteile: 🎜🎜🎜🎜Vorteile: filter
ist für uns zu verwenden. Es wird verwendet zum Berechnen und Filtern einiger Vorlagenausdrücke und v-bind
-Attributausdrücke, bei denen es einige Nachteile gibt. Sie geben den aktuell berechneten Wert zurück, und Parameter können übergeben werden, um diese Filtermethode an mehreren Stellen zu teilen🎜🎜 🎜 🎜Nachteil
: Wenn wir den Ort berechnen möchten, an dem mehrere Daten mit unterschiedlichen Änderungen kombiniert werden, ist filter
im Wesentlichen schwieriger zu filtern /code >Es handelt sich um ein Eins-zu-eins-Verhalten, bei dem einzelne Daten gefiltert werden. Sie können Parameter, dieselbe Methode, aber unterschiedliche Parameter übergeben🎜🎜🎜Nachdem Sie das obige Beispiel mit filter
geändert haben > So: 🎜<div id="app"> <h1>比赛时间:{{ time }}s</h1> <h2>直播播报:{{ result }}</h2> <div class="team"> <div> <span>中国队进球数:{{ team.china }}</span> <button @click="team.china++">点击中国队进一球</button> </div> <div> <span>韩国队进球数:{{ team.korea }}</span> <button @click="team.korea++">点击韩国队进一球</button> </div> </div> </div> let app = new Vue({ el: '#app', data () { return { time: 0, team: { china: 0, korea: 0 } } }, created () { let time = setInterval(() => { this.time++ if (this.time == 90) { clearInterval(time) } }, 1000) }, computed: { result () { if (this.time < 90) { if (this.team.china > this.team.korea) { return '中国队领先' } else if (this.team.china < this.team.korea) { return '韩国领先' } else { return '双方僵持' } } else { if (this.team.china > this.team.korea) { return '中国队赢' } else if (this.team.china < this.team.korea) { return '韩国队赢' } else { return '平局' } } } } })🎜🎜🎜In diesem Beispiel verwenden wir einen Filter, um ihn zu implementieren. Natürlich ist die Codemenge etwas größer, aber die Gesamtsemantik wird ziemlich offensichtlich, was die Leute auf den ersten Blick erkennt Hier sind einige Filterberechnungen erforderlich. Wenn Sie reverseString sehen, wissen Sie, dass es sich um eine String-Umkehr handelt. 🎜
computed
zu ebnen. Es stellt sich wieder die Frage: Was kann computed
tun? Auf welche Szenarien kann es angewendet werden? 🎜🎜berechnet
vermeidet tatsächlich alle Nachteile der Vorlagensyntax und des Filters
. Sein Vorteil liegt in der Berechnung aller abhängigen Daten und der anschließenden Rückgabe eines Werts Sie können sich auf alle Daten in der Methode verlassen. Solange sich Daten ändern, werden sie neu berechnet, um die Ansichtsänderung zu aktualisieren. Ist es nicht interessant? Ich bin gespannt, wie es spielt. 🎜🎜Lassen Sie uns über Nutzungsszenarien sprechen. Ich werde Ihnen zunächst ein einfaches und praktisches Anwendungsszenario demonstrieren und später ein interessanteres Anwendungsszenario erstellen. 🎜🎜Erinnerst du dich noch daran, dass es beim Posten auf Weibo eine Zeichenbeschränkung gab, als wir Weibo spielten, zum Beispiel konnten nur 140
Zeichen eingegeben werden. Um das Benutzererlebnis zu verbessern, gibt es bei der Eingabe von Inhalten in das Textfeld auch eine Eingabeaufforderung, die dem Benutzer mitteilt, wie viele Zeichen Sie eingeben können. Dann wird es viel einfacher sein, Vue für solche Dinge zu verwenden. Zum Beispiel das folgende Beispiel: 🎜<div id="app"> <h1>比赛时间:{{ time }}s</h1> <h2>直播播报:{{ result }}</h2> <div class="team"> <div> <span>中国队进球数:{{ team.china }}</span> <button @click="team.china++">点击中国队进一球</button> </div> <div> <span>韩国队进球数:{{ team.korea }}</span> <button @click="team.korea++">点击韩国队进一球</button> </div> </div> </div> let app = new Vue({ el: '#app', data () { return { time: 0, team: { china: 0, korea: 0 }, result: '双方僵持' } }, created () { let time = setInterval(() => { this.time++ if (this.time == 90) { clearInterval(time) } }, 1000) }, wacth: { time (value, oldval) { if (value < 90) { if (this.team.china > this.team.korea) { this.result = '中国队领先' } else if (this.team.china < this.team.korea) { this.result = '韩国队领先' } else { this.result = '双方僵持' } } else { if (this.team.china > this.team.korea) { this.result = '中国队赢' } else if (this.team.chian < this.team.korea) { this.result = '韩国队赢' } else { this.result = '平局' } } }, team (value, oldval) { if (this.time < 90) { if (value.china > value.korea) { this.result = '中国队领先' } else if (value.china < value.korea) { this.result = '韩国队领先' } else { this.result = '双方僵持' } } else { if (value.china > value.korea) { this.result = '中国队赢' } else if(value.chian < value.korea) { this.result = '韩国队赢' } else { this.result = '平局' } } } } })🎜Der Effekt ist wie folgt:🎜
你可以尝试在文本域中输入内容,你将体验的效果类似下图一样:
在computed
创建了一个reduceCount
,一直在监听文字的字符长度,来再次进行计算,返回值给视图,让视图进行变化。这也是一个很简单的示例。前面也提到过,我们可以监听多个数据,只要一个数据变了,整个方法就会重新计算,然后反馈到视图,这个方法只是一个简单的应用。咱们再来看一个示例,这样会更好的帮助我们理解。
这个示例是一个足球比赛记分的示例。简单的对示例进行分析:
比赛时间,用time
来维护
比赛双方的进球数,用对象team
来维护
比赛的播报情况,在90分钟内,要显示中国领先或者韩国领先或者双方僵持,如果到了90分钟我们要显示中国队赢还是韩国队赢,还是平局
第三个数据相对而言比较复杂,也是比较关键的数据。那么我们用什么方式来维护,可以说比赛情况是多样化的,用一个数据去定死,这样不符合我们的场景。那么我们先列出将会改变的地方:
我们需要检测双方的进球数
通过时间来比对,确定比赛是否结果,然后显示对应的方案
这样我们就要不断的监听两个维护的数据,一是比赛时间,二是比赛两队进球数。
<div id="app"> <h1>比赛时间:{{ time }}s</h1> <h2>直播播报:{{ result }}</h2> <div class="team"> <div> <span>中国队进球数:{{ team.china }}</span> <button @click="team.china++">点击中国队进一球</button> </div> <div> <span>韩国队进球数:{{ team.korea }}</span> <button @click="team.korea++">点击韩国队进一球</button> </div> </div> </div> let app = new Vue({ el: '#app', data () { return { time: 0, team: { china: 0, korea: 0 } } }, created () { let time = setInterval(() => { this.time++ if (this.time == 90) { clearInterval(time) } }, 1000) }, computed: { result () { if (this.time < 90) { if (this.team.china > this.team.korea) { return '中国队领先' } else if (this.team.china < this.team.korea) { return '韩国领先' } else { return '双方僵持' } } else { if (this.team.china > this.team.korea) { return '中国队赢' } else if (this.team.china < this.team.korea) { return '韩国队赢' } else { return '平局' } } } } })
看到的效果如下:
注:这里时间90分钟是一个写死的时间值。如果要让Demo更为完美,这个时间我们可以写一个90分钟的倒计时效果。如果你感兴趣的话,可以自己动手修改上面的Demo,然后在下面的评论中与我们一起分享。
这个示例中,用了点击事件来进行双方进球数,这个Demo帮助我们能能更充分的理解Vue中的computed
的含义。说到底是观察一个或多个数据,每当其中一个数据改变的时候,这个函数就会重新计算,还有就是通过观察所有数据来维护一个状态,就是所谓的返回一个状态值。从上面这个Demo,我们就可以很容易的知道computed
到底用在什么场景,如何去维护返回一个多状态的场景。
在Vue中,使用methods
可以做computed
同样的事情,不同的是,computed
可以进行缓存。什么意思呢?就是在上个例子中我们对比赛时间和两个球队的进球数进行了检测数据。如果随着时间的改变,但是球数没动,对于computed
来说只会重新计算这个球数,进入缓存,而不会再次计算,而重新计算的是这个时间,而且页面的DOM更新也会触发methods
来重新计算属性。所以,如果不想让计算属性进入缓存,请使用methods
,但我个人更推荐使用computed
,语义化会更好一点。毕竟是什么选项里就应该做什么事,methods
里面就是应该来管事件的。
computed
和watch
都可以做同一件事,就像跑步运动员都可以跑步,但是分100米和1000米,术业有专攻嘛,两个选项都是对数据进行时时监听,但是两个的适用场景就不一样了:
computed
前面说了是适用于对多数据变动进行监听,然后来维护一个状态,就是返回一个状态
wacth
是对一个数据监听,在数据变化时,会返回两个值,一个是value
(当前值),二是oldvalue
是变化前的值
我们可以通过这些变化也可以去维护一个状态,但是不符合场景。那么watch
主要用于什么地方呢?其主要用于监听一个数据来进行复杂的逻辑操作。
<div id="app"> <h1>比赛时间:{{ time }}s</h1> <h2>直播播报:{{ result }}</h2> <div class="team"> <div> <span>中国队进球数:{{ team.china }}</span> <button @click="team.china++">点击中国队进一球</button> </div> <div> <span>韩国队进球数:{{ team.korea }}</span> <button @click="team.korea++">点击韩国队进一球</button> </div> </div> </div> let app = new Vue({ el: '#app', data () { return { time: 0, team: { china: 0, korea: 0 }, result: '双方僵持' } }, created () { let time = setInterval(() => { this.time++ if (this.time == 90) { clearInterval(time) } }, 1000) }, wacth: { time (value, oldval) { if (value < 90) { if (this.team.china > this.team.korea) { this.result = '中国队领先' } else if (this.team.china < this.team.korea) { this.result = '韩国队领先' } else { this.result = '双方僵持' } } else { if (this.team.china > this.team.korea) { this.result = '中国队赢' } else if (this.team.chian < this.team.korea) { this.result = '韩国队赢' } else { this.result = '平局' } } }, team (value, oldval) { if (this.time < 90) { if (value.china > value.korea) { this.result = '中国队领先' } else if (value.china < value.korea) { this.result = '韩国队领先' } else { this.result = '双方僵持' } } else { if (value.china > value.korea) { this.result = '中国队赢' } else if(value.chian < value.korea) { this.result = '韩国队赢' } else { this.result = '平局' } } } } })
以上代码和computed
产生的效果是一模一样,但是很明显,就像我对computed
和watch
阐述过了应用场景,这个场景只是维护了一个比赛状态,而不牵扯到逻辑操作。虽然也能完成,但无论从代码量的比对还是可读性,还是可维护性的都不胜于computed
。但说到底谁更强大呢?我还是老实说watch
更强大,哪里有然他有场景的局限性,但是他可以做牵扯到计算属性的一切操作,缺点是watch
只能一个一个监听。
我相信图片预加载大家肯定都有接触过,当图片量大的时候,为了保证页面图片都加载出来的时候,才渲染页面,再进行一些 Ajax请求,或者逻辑操作。那些时你用computed
对这种监听一个数据然后进行一系列逻辑操作和Ajax
请求,那watch
再适合不过了,如果用computed
的话,那是无法实现的。
<template> <div v-show=show> <img src="https://img.alicdn.com/simba/img/TB14sYVQXXXXXc1XXXXSutbFXXX.jpg" alt=""> <img src="//img.alicdn.com/tfs/TB1iZ6EQXXXXXcsXFXXXXXXXXXX-520-280.jpg_q90_.webp" alt=""> <img src="https://img.alicdn.com/simba/img/TB1C0dOPXXXXXarapXXSutbFXXX.jpg" alt=""> <img src="//img.alicdn.com/tfs/TB1iZ6EQXXXXXcsXFXXXXXXXXXX-520-280.jpg_q90_.webp" alt=""> </div> </template> <script> export default { mounted () { var _this = this let imgs = document.querySelectorAll('img') console.log(imgs) Array.from(imgs).forEach((item)=>{ let img = new Image() img.onload = ()=>{ this.count++ } img.src=item.getAttribute('src') }) }, data () { return { count : 0, show : false } }, watch : { count (val,oldval) { if(val == 4){ this.show = true alert("加载完毕") //然后可以对后台发送一些ajax操作 } } } } </script>
我们可以发现四张图片都加载完毕的时候,页面才渲染出来。
Vue官方有一句话说得很重要:
虽然计算属性在大多数情况下更合适,但有时也需要一个自定义的
watcher
。这是为什么Vue提供一个更通用的方法,使用wacth
来响应数据的变化。当你想要在数据变化响应时,执行异步操作或开销较大的操作,这是很有用的。
基于这个描述,我对computed
和watch
的总结:
computed
:监听多个数据或者一个数据来维护返回一个状态值,只要其中一个或多个数据发生了变化,则会重新计算整个函数体,重新返回状态值
watch
:只能一个一个监听数据,只要这个数据发生变化,就会返回两个参数,第一个是当前的值,第二个是变化前的值。每当变化的时候,则会触发函数体的逻辑行为,根据逻辑行为做后续的操作
其实,computed
、watch
这几个都不是有多难,如果从表层上来说都很易理解,但从深层面上看,很多时候还是会存在问题。比如说会滥用,混用。这篇文章通过一些示例来讲解Vue的计算属性,并且对其做了一定的分析。我想这些对学习Vue的初级人员或许会有一些帮助。
原文地址:https://www.w3cplus.com/vue/vue-computed.html
Das obige ist der detaillierte Inhalt vonWas kann die berechnete Vue-Eigenschaft bewirken? Kurze Analyse von Anwendungsszenarien. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!