成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

[譯]簡單的用javascript實(shí)現(xiàn)的數(shù)據(jù)雙向綁定

pf_miles / 1743人閱讀

摘要:許多流行的框架,像或者都會把雙向數(shù)據(jù)綁定作為其中的主要特性來宣傳。盡管有好多種方式去實(shí)現(xiàn)這幾點(diǎn),一種簡單高效的方法是我們通過發(fā)布訂閱者模式來實(shí)現(xiàn)。方法很簡單我們可以使用定制的屬性作為代碼中需要綁定的屬性。

原文:Easy Two-Way Data Binding in Javascript

雙向數(shù)據(jù)綁定指的是當(dāng)對象的屬性發(fā)生變化時(shí)能夠同時(shí)改變對應(yīng)的UI,反之亦然。換句話說,如果我們有一個user對象,這個對象有一個name屬性,無論何時(shí)你對user.name設(shè)置了一個新值,UI也會展示這個新的值。同樣的,如果UI包含一個用于數(shù)據(jù)用戶名字的輸入框,輸入一個新值也會導(dǎo)致user對象的name屬性發(fā)生相應(yīng)的改變。

許多流行的javascript框架,像Ember.js,Angular.js或者KnockoutJS都會把雙向數(shù)據(jù)綁定作為其中的主要特性來宣傳。這并不意味著從頭開始實(shí)現(xiàn)它很難,也不意味著當(dāng)我們需要這種功能的時(shí)候,使用這些框架是我們唯一的選擇。內(nèi)部的潛在思想事實(shí)上是相當(dāng)基礎(chǔ)的,實(shí)現(xiàn)它可以歸納為以下三點(diǎn):

我們需要一種方式確定哪個UI元素綁定在哪個屬性上。

我們需要監(jiān)控屬性和UI的變化

我們需要把所有綁定的對象和UI元素的變化傳播出去。

盡管有好多種方式去實(shí)現(xiàn)這幾點(diǎn),一種簡單高效的方法是我們通過發(fā)布訂閱者模式來實(shí)現(xiàn)。方法很簡單:我們可以使用定制的data屬性作為HTML代碼中需要綁定的屬性。所有的綁定在一起的Javascript對象和DOM元素將會訂閱這個發(fā)布訂閱對象。任何時(shí)候我們檢測到無論是Javascript對象亦或是HTML的input元素的變化,我們都是把事件代理傳遞給發(fā)布訂閱對象,然后通過它把所有發(fā)生在綁定的對象和元素的的變化傳遞和廣播出去。

一個用jQuery實(shí)現(xiàn)的簡單例子

通過jQuery實(shí)現(xiàn)我們上面討論的東西是相當(dāng)簡單明了的,因?yàn)樽鳛橐粋€流行的庫,它讓我們很簡單的實(shí)現(xiàn)訂閱和發(fā)布DOM事件,同時(shí)我們也可以定制一個:

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數(shù)據(jù)模型實(shí)現(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)在,無論何時(shí)我們想要綁定一個對象的屬性到UI上,我們只要在對應(yīng)的HTML元素上設(shè)置合適的data屬性。

// javascript 
var user=new User(123);
user.set("name","Wolfgang");

// html

input輸入框上值得變化會自動的映射到username屬性,反之亦然。大功告成!

不需要jQuery的實(shí)現(xiàn)方式

現(xiàn)在的大部分項(xiàng)目一般jQuery都已經(jīng)在使用啦,所以上面的例子是完全可以接受的。但是如果我們需要完全不依賴jQuery,那么該怎么實(shí)現(xiàn)呢?好吧,事實(shí)上其實(shí)也不難辦到(特別是當(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é)果。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/86142.html

相關(guān)文章

  • 】《精通使用AngularJS開發(fā)Web App》(三)--- 深入scope,繼承結(jié)構(gòu),事件系

    摘要:比如,我們可以監(jiān)聽事件由實(shí)例發(fā)出,然后在任何瀏覽器中就是變化的時(shí)候都會得到通知,如下所示每一個作用域?qū)ο蠖紩羞@個方法,可以用來注冊一個作用域事件的偵聽器。這個函數(shù)所扮演的偵聽器在被調(diào)用時(shí)會有一個對象作為第一個參數(shù)。 上一篇:【譯】《精通使用AngularJS開發(fā)Web App》(二) 下一篇:【譯】《精通使用AngularJS開發(fā)Web App》(四) 書名:Mastering W...

    wind5o 評論0 收藏0
  • 】《精通使用AngularJS開發(fā)Web App》(二) --- 框架概覽,雙向數(shù)據(jù)綁定,MVC

    摘要:本書的這一部分將為隨后的章節(jié)打下基礎(chǔ),會涵蓋模板,模塊化,和依賴注入。本書的小例子中我們會使用未經(jīng)壓縮的,開發(fā)友好的版本,在的上。作用域也可以針對特定的視圖來擴(kuò)展數(shù)據(jù)和特定的功能。 上一篇:【譯】《精通使用AngularJS開發(fā)Web App》(一) 下一篇:【譯】《精通使用AngularJS開發(fā)Web App》(三) 原版書名:Mastering Web Application D...

    geekidentity 評論0 收藏0
  • []JSX:硬幣另一面

    摘要:它不過是硬幣的另一面。因此,既然我們能夠接受與通過這種方式混合在一塊兒,那么是時(shí)候讓介入并向我們展示硬幣的另一面了第三階段的并不是一個激進(jìn)的改變,是因?yàn)槲覀冞@個行業(yè)從一開始就注定和應(yīng)該是在一起的。 React框架剛剛發(fā)布的時(shí)候,JSX顛覆了很多人的想法。習(xí)慣了HTML標(biāo)簽與JavaScript代碼分離的前端工程師們,看到JSX大概都會不禁吐槽:這些奇怪的標(biāo)簽出現(xiàn)在JavaScript里...

    mudiyouyou 評論0 收藏0
  • 】每個JavaScript 開發(fā)者應(yīng)該了解10個面試題

    摘要:避免脆弱的基類問題。紅牌警告沒有提到上述任何問題。單向數(shù)據(jù)流意味著模型是單一的事實(shí)來源。單向數(shù)據(jù)流是確定性的,而雙向綁定可能導(dǎo)致更難以遵循和理解的副作用。原文地址 1. 你能說出兩種對 JavaScript 應(yīng)用開發(fā)者而言的編程范式嗎? 希望聽到: 2. 什么是函數(shù)編程? 希望聽到: 3. 類繼承和原型繼承的不同? 希望聽到 4. 函數(shù)式編程和面向?qū)ο缶幊痰膬?yōu)缺點(diǎn)? ...

    mykurisu 評論0 收藏0
  • 前端優(yōu)化感想以及[]redux 教程 第一部分(共四部分

    摘要:自己英語一般,水平有限,獻(xiàn)上原文地址,還有我翻譯的中文地址,歡迎大家勘誤下面是自己的一點(diǎn)感想先說一下,我們知道,前端優(yōu)化有這么幾步,第一步首先呢我們知道,一個應(yīng)用要依賴好多條文件,而瀏覽器加載完一條,要執(zhí)行完這條才加載下一條,所以呢,就很慢 自己英語一般,水平有限,獻(xiàn)上原文地址,還有我翻譯的中文地址,歡迎大家勘誤 下面是自己的一點(diǎn)感想 先說一下webpack,我們知道,前端優(yōu)化有這么幾...

    snowell 評論0 收藏0

發(fā)表評論

0條評論

pf_miles

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<