摘要:許多流行的框架,像或者都會把雙向數據綁定作為其中的主要特性來宣傳。盡管有好多種方式去實現這幾點,一種簡單高效的方法是我們通過發布訂閱者模式來實現。方法很簡單我們可以使用定制的屬性作為代碼中需要綁定的屬性。
原文:Easy Two-Way Data Binding in Javascript
雙向數據綁定指的是當對象的屬性發生變化時能夠同時改變對應的UI,反之亦然。換句話說,如果我們有一個user對象,這個對象有一個name屬性,無論何時你對user.name設置了一個新值,UI也會展示這個新的值。同樣的,如果UI包含一個用于數據用戶名字的輸入框,輸入一個新值也會導致user對象的name屬性發生相應的改變。
許多流行的javascript框架,像Ember.js,Angular.js或者KnockoutJS都會把雙向數據綁定作為其中的主要特性來宣傳。這并不意味著從頭開始實現它很難,也不意味著當我們需要這種功能的時候,使用這些框架是我們唯一的選擇。內部的潛在思想事實上是相當基礎的,實現它可以歸納為以下三點:
我們需要一種方式確定哪個UI元素綁定在哪個屬性上。
我們需要監控屬性和UI的變化
我們需要把所有綁定的對象和UI元素的變化傳播出去。
盡管有好多種方式去實現這幾點,一種簡單高效的方法是我們通過發布訂閱者模式來實現。方法很簡單:我們可以使用定制的data屬性作為HTML代碼中需要綁定的屬性。所有的綁定在一起的Javascript對象和DOM元素將會訂閱這個發布訂閱對象。任何時候我們檢測到無論是Javascript對象亦或是HTML的input元素的變化,我們都是把事件代理傳遞給發布訂閱對象,然后通過它把所有發生在綁定的對象和元素的的變化傳遞和廣播出去。
一個用jQuery實現的簡單例子通過jQuery實現我們上面討論的東西是相當簡單明了的,因為作為一個流行的庫,它讓我們很簡單的實現訂閱和發布DOM事件,同時我們也可以定制一個:
function DataBinder(object_id){ // Use a jQuery object as simple PubSub var pubSub=jQuery({}); // We expect a `data` element specifying the binding // in the form:data-bind-=" " var data_attr="bind-"+object_id, message=object_id+":change"; // Listen to chagne events on elements with data-binding attribute and proxy // then to the PubSub, so that the change is "broadcasted" to all connected objects jQuery(document).on("change","[data-]"+data_attr+"]",function(eve){ var $input=jQuery(this); pubSub.trigger(message,[$input.data(data_attr),$input.val()]); }); // PubSub propagates chagnes to all bound elemetns,setting value of // input tags or HTML content of other tags pubSub.on(message,function(evt,prop_name,new_val){ jQuery("[data-"+data_attr+"="+prop_name+"]").each(function(){ var $bound=jQuery(this); if($bound.is("")){ $bound.val(new_val); }else{ $bound.html(new_val); } }); }); return pubSub; }
至于javascript對象,下面是最小化的user數據模型實現的例子:
function User(uid){ var binder=new DataBinder(uid), user={ attributes:{}, // The attribute setter publish changes using the DataBinder PubSub set:function(attr_name,val){ this.attributes[attr_name]=val; binder.trigger(uid+":change",[attr_name,val,this]); }, get:function(attr_name){ return this.attributes[attr_name]; }, _binder:binder }; // Subscribe to PubSub binder.on(uid+":change",function(evt,attr_name,new_val,initiator){ if(initiator!==user){ user.set(attr_name,new_val); } }); return user; }
現在,無論何時我們想要綁定一個對象的屬性到UI上,我們只要在對應的HTML元素上設置合適的data屬性。
// javascript var user=new User(123); user.set("name","Wolfgang"); // html
input輸入框上值得變化會自動的映射到user的name屬性,反之亦然。大功告成!
不需要jQuery的實現方式現在的大部分項目一般jQuery都已經在使用啦,所以上面的例子是完全可以接受的。但是如果我們需要完全不依賴jQuery,那么該怎么實現呢?好吧,事實上其實也不難辦到(特別是當我們把對IE的支持只提供IE8以上的支持)。最后,我們只是要通過發布訂閱者模式來觀察DOM事件而已。
function DataBinder( object_id ) { // Create a simple PubSub object var pubSub = { callbacks: {}, on: function( msg, callback ) { this.callbacks[ msg ] = this.callbacks[ msg ] || []; this.callbacks[ msg ].push( callback ); }, publish: function( msg ) { this.callbacks[ msg ] = this.callbacks[ msg ] || [] for ( var i = 0, len = this.callbacks[ msg ].length; i < len; i++ ) { this.callbacks[ msg ][ i ].apply( this, arguments ); } } }, data_attr = "data-bind-" + object_id, message = object_id + ":change", changeHandler = function( evt ) { var target = evt.target || evt.srcElement, // IE8 compatibility prop_name = target.getAttribute( data_attr ); if ( prop_name && prop_name !== "" ) { pubSub.publish( message, prop_name, target.value ); } }; // Listen to change events and proxy to PubSub if ( document.addEventListener ) { document.addEventListener( "change", changeHandler, false ); } else { // IE8 uses attachEvent instead of addEventListener document.attachEvent( "onchange", changeHandler ); } // PubSub propagates changes to all bound elements pubSub.on( message, function( evt, prop_name, new_val ) { var elements = document.querySelectorAll("[" + data_attr + "=" + prop_name + "]"), tag_name; for ( var i = 0, len = elements.length; i < len; i++ ) { tag_name = elements[ i ].tagName.toLowerCase(); if ( tag_name === "input" || tag_name === "textarea" || tag_name === "select" ) { elements[ i ].value = new_val; } else { elements[ i ].innerHTML = new_val; } } }); return pubSub; }
數據模型可以保持不變,除了在setter中對jQuery中trigger方法的調用,我們可以通過我們在PubSub中自定義的publish方法來代替。
// In the model"s setter: function User( uid ) { // ... user = { // ... set: function( attr_name, val ) { this.attributes[ attr_name ] = val; // Use the `publish` method binder.publish( uid + ":change", attr_name, val, this ); } } // ... }
我們又一次通過一百行不到,又可維護的純javascript完成了我們想要的結果。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/86142.html
摘要:比如,我們可以監聽事件由實例發出,然后在任何瀏覽器中就是變化的時候都會得到通知,如下所示每一個作用域對象都會有這個方法,可以用來注冊一個作用域事件的偵聽器。這個函數所扮演的偵聽器在被調用時會有一個對象作為第一個參數。 上一篇:【譯】《精通使用AngularJS開發Web App》(二) 下一篇:【譯】《精通使用AngularJS開發Web App》(四) 書名:Mastering W...
摘要:本書的這一部分將為隨后的章節打下基礎,會涵蓋模板,模塊化,和依賴注入。本書的小例子中我們會使用未經壓縮的,開發友好的版本,在的上。作用域也可以針對特定的視圖來擴展數據和特定的功能。 上一篇:【譯】《精通使用AngularJS開發Web App》(一) 下一篇:【譯】《精通使用AngularJS開發Web App》(三) 原版書名:Mastering Web Application D...
摘要:它不過是硬幣的另一面。因此,既然我們能夠接受與通過這種方式混合在一塊兒,那么是時候讓介入并向我們展示硬幣的另一面了第三階段的并不是一個激進的改變,是因為我們這個行業從一開始就注定和應該是在一起的。 React框架剛剛發布的時候,JSX顛覆了很多人的想法。習慣了HTML標簽與JavaScript代碼分離的前端工程師們,看到JSX大概都會不禁吐槽:這些奇怪的標簽出現在JavaScript里...
摘要:避免脆弱的基類問題。紅牌警告沒有提到上述任何問題。單向數據流意味著模型是單一的事實來源。單向數據流是確定性的,而雙向綁定可能導致更難以遵循和理解的副作用。原文地址 1. 你能說出兩種對 JavaScript 應用開發者而言的編程范式嗎? 希望聽到: 2. 什么是函數編程? 希望聽到: 3. 類繼承和原型繼承的不同? 希望聽到 4. 函數式編程和面向對象編程的優缺點? ...
摘要:自己英語一般,水平有限,獻上原文地址,還有我翻譯的中文地址,歡迎大家勘誤下面是自己的一點感想先說一下,我們知道,前端優化有這么幾步,第一步首先呢我們知道,一個應用要依賴好多條文件,而瀏覽器加載完一條,要執行完這條才加載下一條,所以呢,就很慢 自己英語一般,水平有限,獻上原文地址,還有我翻譯的中文地址,歡迎大家勘誤 下面是自己的一點感想 先說一下webpack,我們知道,前端優化有這么幾...
閱讀 2442·2019-08-30 15:52
閱讀 2242·2019-08-30 12:51
閱讀 2837·2019-08-29 18:41
閱讀 2823·2019-08-29 17:04
閱讀 817·2019-08-29 15:11
閱讀 1727·2019-08-28 18:02
閱讀 3606·2019-08-26 10:22
閱讀 2514·2019-08-26 10:12