輕松實現(xiàn)javascript數(shù)據(jù)雙向綁定
雙向數(shù)據(jù)綁定指的是當(dāng)對象的屬性發(fā)生變化時能夠同時改變對應(yīng)的UI,反之亦然。換句話說,如果我們有一個user對象,這個對象有一個name屬性,無論何時你對user.name設(shè)置了一個新值,UI也會展示這個新的值。同樣的,如果UI包含一個用于數(shù)據(jù)用戶名字的輸入框,輸入一個新值也會導(dǎo)致user對象的name屬性發(fā)生相應(yīng)的改變。
許多流行的javascript框架,像Ember.js,Angular.js或者KnockoutJS都會把雙向數(shù)據(jù)綁定作為其中的主要特性來宣傳。這并不意味著從頭開始實現(xiàn)它很難,也不意味著當(dāng)我們需要這種功能的時候,使用這些框架是我們唯一的選擇。內(nèi)部的潛在思想事實上是相當(dāng)基礎(chǔ)的,實現(xiàn)它可以歸納為以下三點(diǎn):
- 我們需要一種方式確定哪個UI元素綁定在哪個屬性上。
- 我們需要監(jiān)控屬性和UI的變化
- 我們需要把所有綁定的對象和UI元素的變化傳播出去。
盡管有好多種方式去實現(xiàn)這幾點(diǎn),一種簡單高效的方法是我們通過發(fā)布訂閱者模式來實現(xiàn)。方法很簡單:我們可以使用定制的data屬性作為HTML代碼中需要綁定的屬性。所有的綁定在一起的Javascript對象和DOM元素將會訂閱這個發(fā)布訂閱對象。任何時候我們檢測到無論是Javascript對象亦或是HTML的input元素的變化,我們都是把事件代理傳遞給發(fā)布訂閱對象,然后通過它把所有發(fā)生在綁定的對象和元素的的變化傳遞和廣播出去。
一個用jQuery實現(xiàn)的簡單例子
通過jQuery實現(xiàn)我們上面討論的東西是相當(dāng)簡單明了的,因為作為一個流行的庫,它讓我們很簡單的實現(xiàn)訂閱和發(fā)布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-<object_id>="<property_name>"
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數(shù)據(jù)模型實現(xiàn)的例子:
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;
}
現(xiàn)在,無論何時我們想要綁定一個對象的屬性到UI上,我們只要在對應(yīng)的HTML元素上設(shè)置合適的data屬性。
// javascript
var user=new User(123);
user.set("name","Wolfgang");
// html
<input type="number" data-bind-123="name" />
input輸入框上值得變化會自動的映射到user的name屬性,反之亦然。大功告成!
不需要jQuery的實現(xiàn)方式
現(xiàn)在的大部分項目一般jQuery都已經(jīng)在使用啦,所以上面的例子是完全可以接受的。但是如果我們需要完全不依賴jQuery,那么該怎么實現(xiàn)呢?好吧,事實上其實也不難辦到(特別是當(dāng)我們把對IE的支持只提供IE8以上的支持)。最后,我們只是要通過發(fā)布訂閱者模式來觀察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;
}
數(shù)據(jù)模型可以保持不變,除了在setter中對jQuery中trigger方法的調(diào)用,我們可以通過我們在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 );
}
}
// ...
}
通過實例講解,并又一次通過一百行不到,又可維護(hù)的純javascript完成了我們想要的結(jié)果,希望對大家實現(xiàn)javascript數(shù)據(jù)雙向綁定有所幫助。
- javascript實現(xiàn)數(shù)據(jù)雙向綁定的三種方式小結(jié)
- Vue.js每天必學(xué)之?dāng)?shù)據(jù)雙向綁定
- 深入學(xué)習(xí)AngularJS中數(shù)據(jù)的雙向綁定機(jī)制
- 淺談AngularJs 雙向綁定原理(數(shù)據(jù)綁定機(jī)制)
- Vue.js第一天學(xué)習(xí)筆記(數(shù)據(jù)的雙向綁定、常用指令)
- 深入理解Angularjs向指令傳遞數(shù)據(jù)雙向綁定機(jī)制
- AngularJS學(xué)習(xí)筆記(三)數(shù)據(jù)雙向綁定的簡單實例
- JS原生數(shù)據(jù)雙向綁定實現(xiàn)代碼
- 實例剖析AngularJS框架中數(shù)據(jù)的雙向綁定運(yùn)用
- js實現(xiàn)數(shù)據(jù)雙向綁定(訪問器監(jiān)聽)
相關(guān)文章
JS求1到任意數(shù)之間的所有質(zhì)數(shù)的方法詳解
這篇文章主要介紹了JS求1到任意數(shù)之間的所有質(zhì)數(shù),本文通過實例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價值 ,需要的朋友可以參考下2019-05-05
教你如何解密js/vbs/vbscript加密的編碼異處理小結(jié)
教你如何解密js/vbs/vbscript加密的編碼異處理加密代碼 是一篇非常不錯的加密解密原理,希望大家仔細(xì)研究2008-06-06
Javasript設(shè)計模式之鏈?zhǔn)秸{(diào)用詳解
這篇文章主要為大家詳細(xì)介紹了Javasript設(shè)計模式之鏈?zhǔn)秸{(diào)用的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-04-04
談?wù)勎覍avaScript中typeof和instanceof的深入理解
這次主要說說javascript的類型判斷函數(shù)typeof和判斷構(gòu)造函數(shù)原型instanceof的用法和注意的地方,對本文感興趣的朋友一起看看吧2015-12-12
使用 JavaScript如何獲取當(dāng)月的第一天和最后一天
這篇文章主要介紹了使用 JavaScript如何獲取當(dāng)月的第一天和最后一天,通過本文學(xué)習(xí)了如何使用 JavaScript 中的Date.getFullYear()和?Date.getMonth()方法獲得某個特定月份的第一天和最后一天,需要的朋友可以參考下2023-05-05

