摘要:基礎布局的中主要為部分,分別是用于搜索篩選和分頁的表單控件用于排序表格的表頭以及用于展示數據的。這也是前瞻發布之后,提出廢棄部分功能后許多人反應較為強烈的原因。
需求分析與上周的第一篇實踐教程一樣,在這篇文章中,我將繼續從一種常見的功能——表格入手,展示Vue.js中的一些優雅特性。同時也將對filter功能與computed屬性進行對比,說明各自的適用場景,也為vue2.0版本中即將刪除的部分filter功能做準備。
還是先從需求入手,想想實現這樣一個功能需要注意什么、大致流程如何、有哪些應用場景。
表格本身是一種非常常用的組件,用于展示一些復雜的數據時表現很好。
當數據比較多時,我們需要提供一些篩選條件,讓用戶更快列出他們關注的數據。
除了預設的一些篩選條件,可能還需要一些個性化的輸入搜索功能。
對于有明顯順序關系的數據,例如排名、價格等,還需要排序功能方便快速倒置數據。
如果數據量較大,需要分頁展示表格。
需要注意的是,上述的這些需求其實和大部分數據庫提供的功能是非常一致的,而且由于數據庫擁有索引等優化方式以及服務器更好的性能,更加適合處理這些需求。不過現在流行的前后端分離,也是希望讓客戶端在合理的范圍內,更多的分擔服務器端的壓力,所以當找到一個平衡時,在前端處理適量的需求是正確的選擇。
接下來就嘗試用vue完成這些需求吧。
完成Table.vue因為這樣一個多功能表格可能會應用在多個項目中,所以設計思路上盡量將表格相關的內容放在Table.vue組件中,減少耦合,方便復用。
獲取測試數據為了更好的對比前端實現以上需求的利與弊,我們需要一份較大較復雜的測試數據。幸運的是我之前的一個項目中,設計的一份API正好滿足這一需求,數據為魔獸世界競技場的天梯排行API,目前這個API處于開放狀態,接口詳見Myarena介紹。
與上一篇教程相類似,還是新建一個api文件夾以及一個arena.js用于管理API接口。再在App.vue中引入arena.js,在created階段獲取數據。作為一個demo,我們只獲取region為CN、laddar為3v3的數據,不過只要將兩個參數通過v-model綁定給對應的表單控件,就能很輕松的實現不同地區數據的切換。
引入table.vue組件如之前所說,思路上我們希望減少table組件與外部環境的耦合,所以我們給Table.vue設置一個props屬性rows,用于獲取App.vue取回的數據。
在App.vue中注冊table組建時要注意,命名不能用默認的table,所以注冊為vTable,就能用
目前為止,我們的App.vue完成了它所有的功能,代碼如下:
實際的App.vue中還有一個獲取API中的最后更新時間的操作,以及一些css設置,篇幅考慮這里進行了省略,對完整代碼有興趣的可以移步文章末尾的Github倉庫。
基礎布局Table.vue的template中主要為3部分,分別是用于搜索、篩選和分頁的表單控件、用于排序表格的表頭thead以及用于展示數據的tbody。
首先來完成tbody的部分,基本思路就是用v-for遍歷數據,再通過模板填入,需要注意以下幾個重點:
返回的數據不一定完全符合要求。例如我希望實現通過勝率排序,但數據中只包含了勝負場數,需要先計算一次。2. 數據中用于表現玩家職業的數據為classId這個屬性,但在實際項目中我想要用各職業的icon展示職業,所以我在utils.js中實現了各一個classIdToIcon的工具函數,用于映射classId至sprite圖中的background-position。
以上兩點說明我們最好不要遍歷props獲得的rows這一原始數據。因此另建了一個computed屬性players,并在其中完成了前期處理,我把所有的前期處理放在了handleBefore中。
由于即將使用的各種filters操作比較復雜,所以在handlebefore中進行了console.log("before handle"),方便我們驗證handlebefore在什么階段被執行了。
完成布局之后,目前Table.vue中的重點代碼如下:
{{ player.ranking }} {{ player.rating }} {{ player.name }} {{ player.realmName }}
可以看到,我還引入了一個Bar.vue組件用于展示勝率,這是因為我希望最終的實際效果是這樣的:
一開始我直接在勝率所在的
更好理解和維護了,不是嗎?
用filters實現需求在使用vue的過程中,需要注意的是框架中許多方法其實在內部最終是殊途同歸。
例如我們可以直接在元素中執行一些對數據的操作,例如@click="show = !show",同樣的我們也可以對事件綁定方法,再在方法中操作數據,例如@click="toggle", toggle () { this.show = !this.show }。還比如我們可以用computed屬性和watch屬性實現很多相同的功能,接下來還將用computed去實現和filters相同的功能。
vue設計中的靈活性讓我們有了更多的可能性,但在學習時,應該以搞明白不同方式在不同場景中的優劣為目標,實際運用時選擇最好的那一種。
在例子中,players實際是一個5000條數據的數組,在不做任何處理時,將直接渲染出5000個
對于v-for循環,vue中提供了3中filters過濾數組,分別為filterBy, orderBy, limitBy,其功能對應了搜索/篩選、排序和分頁,實現分別是使用了Array.filter, Array.sort(), Array.slice()。
這三種filters在使用時非常便利,只要在v-for后用|分離再添加對應的filters即可,這3中filter的具體參數可以查看官方API,這里不多做贅述。
需要注意的是,實際的過程是先將被遍歷的數組(例子中的players)依次通過過濾器,再將最后一個過濾器返回的數組進行v-for操作。
因此,filters放置的順序是需要根據需求來調整的,也因為每種過濾器的內部實現效率不同,所以在需求優先級不明顯時,應該以效率為優先。
注意:實際測試時,發現不論怎么過濾數組,handleBefore方法都沒有再次執行,也就是說players數組并沒有被改動過。
例如在我的例子中,我希望可以篩選出名字或者服務器包含了我所輸入內容的玩家,并且將他們按照某種方式排序,最后的結果每頁只顯示20條。
那么顯然剪切數組永遠應該放在最后一步,而排序和過濾在需求中沒有明顯的優先級。但是大部分情況下,sort的效率都要低于filter,所以我們先進行filter,減少數組長度,再sort。
有了這一思路之后,用于v-for的
這里直接將各個變量動態化,再通過Table.vue中的input綁定v-model以及表頭thead綁定@click事件來改變篩選的條件,就已經實現了大部分的搜索、過濾、分頁功能。
表頭改變sort排序我是通過以下代碼實現的,方式可能不是太好,特此列出:
排名 分數 資料 服務器 本周戰績 賽季戰績 可以看到,通過vue的filters功能,已經可以輕松完成我們的大部分功能,代碼量極少。這也是vue2.0前瞻發布之后,提出廢棄部分filters功能后許多人反應較為強烈的原因。
但是如同作者在改動說明中所說,filters對于初學者來說不易理解,并且filters的功能都可以用computed屬性進行更靈活、更好把控的實現。而且在一些復雜條件下,堆疊過濾器會造成一些額外的復雜性以及不方便之處。那么何為復雜條件呢?例如我增添兩個需求,一是按職業篩選玩家,而是篩選出一定分數以上的玩家,那么后者用filterBy就不太好實現了。
我們需要將對分數段的過濾放在filters之前進行,但又要注意不破壞players數組本身。在實際完成時,會發現這個過程還是比較糾結的。除此之外,我們還會發現分頁中最重要的一個信息——總頁數我們獲取不到。因為vue并沒有把一串過濾管道中產出的最終用于v-for的數組暴露出來,所以我們無法獲得這個實際被循環的數組的長度。
在實際hack這些需求時,發現很容易與filters的執行順序發生沖突,因此決定重新用computed屬性來實現一遍所有功能,不借助自帶的filters。
當然,在這一段的前半部分中,我們顯而易見的感受到了來自filters的便利性。如果需求中filters可以滿足,那么在1.x版本中使用filters還是十分明智的!
用computed屬性完成需求在Github倉庫中,我用Table.vue.bak文件儲存了之前一段中用filters實現的代碼,方便與我們接下里的實現進行比較。
首先整理一下用computed屬性來實現的思路:
首先要實現filterBy, orderBy, limitBy這三個filter的功能,上文中已經提到了他們的內部實現,所以分別用Array.filter, Array.sort和Array.slice重寫一遍并不復雜。
說是computed屬性實現,其實也還是只有players這個computed屬性,只是在其內部執行了所有的過濾動作,我們實際是把各種過濾器的邏輯放置在各個method中。
不建議把各個過濾method寫的過于抽象,因為就是內置filters高度抽象導致一些特殊需求無法實現,所以不妨就以最針對性的方式:一個method對應一種過濾。
在執行各個過濾method時,依然有最初提到的順序帶來的效率問題。因為vue牽一發而動全身的特性,任何一個過濾條件改變時,所有過濾method都會執行一遍,所以盡快用高效的過濾器縮短數組長度顯得更為重要。
我嘗試過通過watch屬性實現最小化method調用,但無奈功力不夠沒能實現。同時我也認為前端處理大量數據的情況很少見,并且用第4點中的數據進行優化后,執行效率不算太低,所以沒必要在這個方面做過多糾結。真有性能瓶頸時,從服務器端尋求解決會更簡單。
注意:在實現各種過濾method時,建議閱讀vue中filterBy, orderBy, limitBy三部分的實現源碼,其本身對于數組的操作就有一些優化,非常值得學習。在一些特殊情況中,例如數組中大量相等值時,過于簡單的sort function會導致執行步數激增,vue中的一些處理都予以了避免。
根據需求目標,我設置了以下這些method(順序即為執行順序):
classFilter:過濾玩家職業,通過item.classId === this.class進行判斷,this.class綁定的是一個select控件。
queryFilter:匹配玩家姓名中的字段,通過item.name.indexOf(this.query)判斷,this.query則綁定一個input控件。
ratingFilter:篩選玩家分數段,通過item.rating >= this.rating進行判斷,this.rating綁定了一個類型為range的input控件,range的范圍則是用computed屬性進行計算。
sortTable:因為Array.sort進行的步數較多,所以放在數組被上述3個method處理的較短后進行。
paginate:所有過濾操作完畢之后,就可以進行分頁了。在使用Array.slice()之前,先將數組的長度傳給this.total儲存起來,用于在分頁后計算總的頁數。
除了以上幾個過濾method以外,當然也還有handleBefore方法對數組進行前期處理。但是由于players每次都會重新計算,所以為了放止handleBefore被重復執行,應該加上一定的判斷條件,例如handleBefore添加的屬性是否已經存在了等等。
同時,還可以把一些不需要在過濾之前執行的動作從handleBefore中拿出,例如例子中的classId轉換為Icon,可以在過濾之后對最終要展示的數據進行即可,減少一些步數。所以又設置了一個handleAfter方法,用于在分頁完成之后進行后續操作,當然在handleAfter中也可能重復執行,所以如果執行的操作消耗很大,建議同樣添加判斷,避免重復執行。在例子代碼中,我在每個方法中都統計了執行的步數,實際結果顯示設置一個合理的過濾順序可以避免一些性能問題,結果如下:
可以看出初始化時,在沒有任何過濾的情況下,sort的步數較高。而一旦添加了一些過濾條件之后,順位靠后的filter和sort的步數都會大幅度減少。
DEMO地址由于工作比較忙,暫時沒有打算將開頭中展示的MyArena項目重構,不過可以想象那會是一個很好的用vue制作單頁應用的示例,后續的教程中可能會用來做例子。
本次教程中的例子,專注于展示多功能表格本身
寫作計劃DEMO地址點我
Github倉庫上周是Vue.js開發實踐的第一篇文章,也是我第一次在SF社區的個人專欄里發表文章,希望能夠把平時遇到的一些問題和解決的思路分享給大家,自己也進行一個梳理。
開發實踐這個系列會用一些小例子,展示一些思路,實現一些有用、可復用的常見功能。計劃中,還會有Vue.js實戰系列和Sails.js實戰系列兩個系列的文章。
前者從較完整的項目出發,分析技術選型、vue-router和vuex的使用、多端共用代碼、后期維護等方面的一些考量。后者則是用Sails.js這個框架構建企業級Node.js后端的一些嘗試和心得,包括框架的優缺點、橫向對比以及細節摸索等等。
目前也在關注阿里的開源項目Weex的內測進展,理想中的狀態是用Weex實現項目在移動端App的開發,真正完成JS全棧,不過Weex還沒正式開源,有待觀望,所以只是后期設想,暫時不在計劃內。文章目前就只發在SF的專欄里,所以有意見建議都請在文章底部留言。同時由于以上所說的所有工作都由我一個人在負責,所以文章的更新可能時快時慢,爭取做到一周一篇。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/49863.html
相關文章
Vue.js 實踐(2):實現多條件篩選、搜索、排序及分頁的表格功能
摘要:基礎布局的中主要為部分,分別是用于搜索篩選和分頁的表單控件用于排序表格的表頭以及用于展示數據的。這也是前瞻發布之后,提出廢棄部分功能后許多人反應較為強烈的原因。 與上周的第一篇實踐教程一樣,在這篇文章中,我將繼續從一種常見的功能——表格入手,展示Vue.js中的一些優雅特性。同時也將對filter功能與computed屬性進行對比,說明各自的適用場景,也為vue2.0版本中即將刪除的部...
【從零入門系列-5】Spring Boot 之 前端展示
摘要:務必在之前引入最新的核心文件為了偷懶,我們這里引入的第三方庫文件都是采用的方式,也可以選擇把庫下載到本地然后再引用。 文章系列 【從零入門系列-0】Spring Boot 之 Hello World 【從零入門系列-1】Spring Boot 之 程序結構設計說明 【從零入門系列-2】Spring Boot 之 數據庫實體類 【從零入門系列-3】Spring Boot 之 數據庫操作...
【從零入門系列-5】Spring Boot 之 前端展示
摘要:務必在之前引入最新的核心文件為了偷懶,我們這里引入的第三方庫文件都是采用的方式,也可以選擇把庫下載到本地然后再引用。 文章系列 【從零入門系列-0】Spring Boot 之 Hello World 【從零入門系列-1】Spring Boot 之 程序結構設計說明 【從零入門系列-2】Spring Boot 之 數據庫實體類 【從零入門系列-3】Spring Boot 之 數據庫操作...
表格組件 GridManager Angular 1.x
摘要:非必設項篩選條件列表數組對象。格式在使用時該參數為必設項。前端雞湯前端框架前端相關篩選選中項,字符串默認為。非必設項,選中的過濾條件將會覆蓋否為多選布爾值默認為。刷新更新查詢條件其它更多請直接訪問查看當前版本 GridManager Angular 1.x 基于 Angular 1.x 的 GridManager 封裝, 用于便捷的在 Angular 中使用GridManager. s...
發表評論
0條評論
ChanceWong
男|高級講師
TA的文章
閱讀更多
SSL證書有效期是多久?SSL證書不能用是哪些原因導致的?
閱讀 923·2021-11-22 13:54
?Java枚舉實例---交通燈,一文帶你深入理解枚舉,了解融會貫通JAVA SE所該具備的能力【Ja
閱讀 2851·2021-09-28 09:36
偽類與偽元素
閱讀 2989·2019-08-30 15:55
A human being,who loves football and music.
閱讀 1957·2019-08-30 15:44
前端面試寶典
閱讀 551·2019-08-29 12:31
一個簡單的瀑布流實現。
閱讀 2568·2019-08-28 18:18
vue強制重新渲染組件
閱讀 1207·2019-08-26 13:58
react本質:JSX如何轉化為javascript
閱讀 1393·2019-08-26 13:44
閱讀需要支付1元查看