国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

在knockoutjs 上自己實現(xiàn)的flux(實例講解)

2019-11-19 14:42:53
字體:
供稿:網(wǎng)友

在knockoutjs 上實現(xiàn) Flux 單向數(shù)據(jù)流 狀態(tài)機,主要解決多個組件之間對數(shù)據(jù)的耦合問題。

一、其實簡單

flux的設(shè)計理念和實現(xiàn)方案,很大程度上人借鑒和參考了Vuex的實現(xiàn),只是簡化了某些過程,數(shù)據(jù)流向圖如下:

image

從上圖,中以看出數(shù)據(jù)的改變是單向循環(huán)的。我想這就是Flux理念的核心所在吧。Vuex中對Action規(guī)范為Action和Mutation,由action去觸發(fā)Mutation,action是可以異步的,而Mutation則是同步更新。而我在設(shè)計ko的Flux時,去掉了Mutation這個環(huán)節(jié),是因為我理解為,異步的請求一般情況下都是與api接口有關(guān)系,這塊內(nèi)容存在極大的變化性,應(yīng)該從業(yè)務(wù)或項目構(gòu)架上做一層區(qū)分。

二、如果使用

當(dāng)然,flux只是針對knockoutjs的,所以你使用之前必須引入knockoutjs。flux主要的方法和對象

2.1 靜態(tài)方法

方法 說明
flux.use 在require模式下,將flux與ko做關(guān)聯(lián)的方法,當(dāng)然他必須先與createStore方法調(diào)用。
flux.createStore 創(chuàng)建一個store(狀態(tài)器)實例,當(dāng)然此方法是有返回值,他的返回值可以調(diào)用register方法注冊到指定的域上,但第一次調(diào)用此方法時是創(chuàng)建rootStore(根狀態(tài)器),他不允許被注冊到域的。

2.1.1 flux.createStore參數(shù)格式

參數(shù)名稱 說明
state 狀態(tài)器相關(guān)狀態(tài)數(shù)據(jù)
actions 更改state上的狀態(tài)方法,方法的第一個參數(shù)為state,第二參數(shù)開始則為傳入的相關(guān)內(nèi)容
getters 獲取state上的相關(guān)狀態(tài)數(shù)據(jù),當(dāng)然返回是一個ko監(jiān)控對象。

2.2 實例方法

createStore方法的執(zhí)行,會在ko實例上增加$store屬性,此屬性是狀態(tài)器的實例對象,在任何位置都可以調(diào)用他的dispatch來觸發(fā)事件。

方法 說明
register 創(chuàng)建和注冊一個狀態(tài)域,域與域之間是相互獨立存儲的,域之間action或get名稱是可以重復(fù)的
unRegister 移除一個狀態(tài)域
dispatch 根據(jù)actionName調(diào)用指定的action,無返回值
get 根據(jù)getName調(diào)用指定的get,有返回值

三、簡單的使用

本示例定義了四個ko綁定區(qū)域,分別是:app1, app2, app3, app4。實現(xiàn)app4中對name的改變自動影響到app1,而app3對列表的改變自動影響到app2。

image

3.1 定義vm并初始化store

function ViewModel(){ this.list = ko.observableArray(); this.name = ko.observable('無名氏'); this.count = ko.computed(function(){  //必須用this,這個時候ko.$store還沒創(chuàng)建完成,應(yīng)該ko.computed創(chuàng)建時會執(zhí)行一次此處  //如果是子vm依賴主vm,還是可以用ko.$store的  return this.list().length + '個數(shù)'; //需要對監(jiān)控對象求值,否則computed不能有效 },this);}var fullVm = new ViewModel(); var index = 1;fullVm.vf={ add: function(){  ko.$store.dispatch('addClass',{title: 'title' + (index++)}); }}var opt = { state: {  class: fullVm }, actions:{  "setName":function(state, name){   state.class.name(name);  },  "addClass":function(state, classInfo){   state.class.list.push(classInfo);  } }, getters:{  "getName":function(state){   return state.class.name;  } }}flux.createStore(opt);

根據(jù)上述代碼,首先定義了ViewModel的一個類,并創(chuàng)建了一個fullVm的一個實例,然后直接在fullVm實例上增加了add方法。

opt的state引用的是fullVm,其中還配置了actions和getters相關(guān)對象,然后調(diào)用flux.createStore(opt)方法。創(chuàng)建一個store,并關(guān)聯(lián)到ko.$store對象上。

3.2 與視圖綁定

html代碼:

<div id="app1"> app1: <span data-bind="text:ko.$store.get('getName')"></span></div><div id="app4"> app4: <input type="text" data-bind="value:name" /> <button type="text" data-bind="click:changeName" >改變名字</button> <span data-bind="text:ko.$store.state.class.name"></span></div><hr><div id="app2"> app2: <ul data-bind="foreach:list" >  <li data-bind="text:title" ></li> </ul></div><div id="app3"> app3:  <button type="button" data-bind="click:vf.add" >添加</button> <span data-bind="text:count"></span></div>

js代碼:

var app1 = ko.applyBindings(fullVm, document.getElementById("app1"));var app2 = ko.applyBindings(fullVm, document.getElementById("app2"));var app3 = ko.applyBindings(fullVm, document.getElementById("app3"));//測試兩個vm之間的依賴 解藕var app4 = ko.applyBindings({ name: ko.observable(), changeName:function(data,event){  ko.$store.dispatch('setName', this.name()); }}, document.getElementById("app4"));

四、域的實例

html代碼:

<div id="app1"> <span data-bind="text:name" ></span></div><div id="app2"> <span data-bind="text:name"></span> <span data-bind="text:full"></span> <button type="button" data-bind="click:changeName" >換名</button></div>

js代碼:

function rootViewModel(){ this.name = ko.observable('root'); }var rVM = new rootViewModel();flux.createStore({ state: rVM}); //創(chuàng)建root狀態(tài)器var treeNode={ name: ko.observable('node'), changeName:function(){  ko.$store.areas.treeNode.state.name('新名字'); }, full: ko.computed(function(){  //computed的職責(zé):1. 監(jiān)控其他對象屬性的變化,而影響自身對象(flux解決);2. 合并自身對象的幾個屬性(在function下,有this可解)  //不能通過ko.$store訪問對象本身,因為首次對象本身還沒初始化好  var store = ko.$store;  //(store.areas.treeNode? store.areas.treeNode.state.name() : '') 這樣也是不行,因為解決第一次通不過,后面肯定不行  return store.state.name(); })}ko.$store.register('treeNode', flux.createStore({ state: treeNode})); //創(chuàng)建子狀態(tài)機var app1 = ko.applyBindings(rVM, document.getElementById("app1"));var app2 = ko.applyBindings(treeNode, document.getElementById("app2"));

五、其他

當(dāng)然模塊化的引用,也是支持。具體實例細節(jié)可參考test中的測試示例。

項目的git地址:https://gitee.com/ko-plugins/flux.git,歡迎大家指正和提出寶貴的意見。

以上這篇在knockoutjs 上自己實現(xiàn)的flux(實例講解)就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持武林網(wǎng)。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 来凤县| 鄂州市| 大方县| 专栏| 威信县| 喀什市| 慈利县| 蒙阴县| 正宁县| 武宁县| 堆龙德庆县| 吉木萨尔县| 屏山县| 吴忠市| 大同市| 岱山县| 安图县| 化隆| 德江县| 安吉县| 交口县| 监利县| 泾川县| 鹤峰县| 当阳市| 九寨沟县| 大安市| 长垣县| 信宜市| 聂拉木县| 萨迦县| 阿克陶县| 嘉禾县| 本溪市| 日喀则市| 泗水县| 五家渠市| 休宁县| 集安市| 江安县| 哈密市|