検索
ホームページウェブフロントエンドVue.jsVue で双方向データ バインディングを実装する方法について話しましょう

Vue で双方向データ バインディングを実装するにはどうすればよいですか? Vue.jsのデータ双方向バインディングの実装方法は以下の記事で紹介していますので、参考になれば幸いです!

Vue で双方向データ バインディングを実装する方法について話しましょう

vue を使用すると、データが変更されるとインターフェースも更新されますが、これは当然のことではありません。 ? データは変更されますが、データが変更されたときに Vue はどのようにインターフェイスを更新しますか?

データを変更すると、vue は es5 の Object.defineProperty メソッドを通じてデータの変更を監視します。データが変更されると、パブリッシュおよびサブスクライブ モードを使用します。統計サブスクライバ インターフェイスが更新されました。これは設計パターンです。 [学習ビデオ共有: vue ビデオ チュートリアル web フロントエンド ビデオ ]

以下に示すように、新しい Vue から開始して、Vue インスタンス、el、データを作成します。 , データが渡されます オブザーバー オブジェクトが渡され、Object.definproperty を使用してデータ内のデータをデータ ハイジャック用のゲッター/セッターに変換します。データ内の各プロパティは Dep インスタンスを作成しますウォッチャー インスタンスを保存する

El がコンパイルに渡され、命令がコンパイルで解析されます。data 内のデータが el に解析されると、ゲッターがトリガーされ、それによってウォッチャーが依存関係に追加されます。データが変更されると、セッターがトリガーされて依存関係通知を発行し、ウォッチャーに通知します。通知を受信した後、ウォッチャーはビューに通知を送信し、ビューを更新させます

Vue で双方向データ バインディングを実装する方法について話しましょう

#データ ハイジャック

HTML 部分は、span タグと input タグを含む id アプリを使用して div タグを作成します。span タグは補間式を使用し、input タグは v-model

<div class="container" id="app">
    <span>内容:{{content}}</span>
    <input type="text" v-model="content">
</div>
を使用します。

js 部分では vue.js ファイルが導入され、双方向データ バインディングを実装するコードがその中に記述され、Vue インスタンス vm を作成し、データを div タグにマウントします

const vm=new Vue({
    el:&#39;#app&#39;,
        data:{
        content:&#39;请输入开机密码&#39;
    }
})

new Vue インスタンスは明らかにコンストラクターを使用する必要があります関数は vue のソース コードで

function を使用して定義されていますここでは ES6 の class を使用してこの Vue インスタンス

を作成し、次に

constructor を設定し、新しい Vue インスタンスの作成時に渡されるオブジェクトとして仮パラメータを obj_instance に設定し、渡されたオブジェクト内のデータをインスタンス ## の $data 属性に割り当てます。 #In JavaScript オブジェクトのプロパティが変更された場合、変更されたプロパティを dom ノードに更新できるように通知する必要があるため、インスタンスの初期化時に listen 関数を呼び出しとして定義し、呼び出し時に関数を渡します監視する必要があるデータ内

class Vue{//创建Vue实例
  constructor(obj_instance){
    this.$data=obj_instance.data
    Observer(this.$data)
  }
}
function Observer(data_instance){//监听函数
  
}

このインスタンスを印刷 vm

Vue で双方向データ バインディングを実装する方法について話しましょうインスタンスは作成されましたが、引き続き各属性を監視する必要があります$data。これはデータ監視の実装に使用されます

Object.defineProperty

Object.defineProperty はオブジェクトの既存のプロパティを変更できます。構文形式は Object.defineProperty(obj) 、prop、記述子)

obj: プロパティを定義するオブジェクト
  • prop: 定義または変更するプロパティの名前
  • descriptor: 定義または変更するプロパティ記述子
  • オブジェクト内の各プロパティを監視するには、Object.keys と foreach を使用してオブジェクト内の各プロパティを走査し、各プロパティに対して Object.defineProperty を使用します。データを監視するには
function Observer(data_instance){
  Object.keys(data_instance).forEach(key=>{
    Object.defineProperty(data_instance,key,{
      enumerable:true,//设置为true表示属性可以枚举
      configurable:true,//设置为true表示属性描述符可以被改变
      get(){},//访问该属性的时候触发,get和set函数就是数据监听的核心
      set(){},//修改该属性的时候触发
    })
  })
}

In

Object.defineProperty

属性に対応する値を保存し、それを get 関数で返す必要があります。そうでない場合は、get 関数に到達した後、属性はなくなり、属性に返される値は未定義になります<pre class='brush:php;toolbar:false;'>let value=data_instance[key] Object.defineProperty(data_instance,key,{ enumerable:true, configurable:true, get(){ console.log(key,value); return value }, set(){} })</pre>Click $data の属性名をクリックすると get 関数がトリガーされます

Vue で双方向データ バインディングを実装する方法について話しましょう#次に、set 関数を設定し、set の仮パラメータを設定します。この仮パラメータは新しく渡された属性値を表し、この新しい属性値を変数値に割り当てます。何も返す必要はなく、変更するだけです。 get にアクセスすると return が返されますが、変更後は get は最新の値変数 value

set(newValue){
    console.log(key,value,newValue);
    value = newValue
}

Vue で双方向データ バインディングを実装する方法について話しましょう にもアクセスしますが、現在は get と set のみが最初の層の属性に設定されています$data.

obj:{
    a:&#39;a&#39;,
    b:&#39;b&#39;
}

のような深い層がある場合、get と set は設定されません。層ごとに属性へのデータ ハイジャックを実行する必要があるため、

recursion

を使用しますトラバースする前に自分自身を再度監視し、条件付きの判断を行うためです。サブプロパティがない場合、またはオブジェクトが検出されない場合は、再帰を終了します

function Observer(data_instance){
  //递归出口
  if(!data_instance || typeof data_instance != &#39;object&#39;) return
  Object.keys(data_instance).forEach(key=>{
    let value=data_instance[key]
    Observer(value)//递归-子属性的劫持
    Object.defineProperty(data_instance,key,{
      enumerable:true,
      configurable:true,
      get(){
        console.log(key,value);
        return value
      },
      set(newValue){
        console.log(key,value,newValue);
        value = newValue
      }
    })
  })
}
別の詳細があります。$data の content 属性を書き換えると、文字列からオブジェクトへの変換では、この新しいオブジェクトには get と set がありません

Vue で双方向データ バインディングを実装する方法について話しましょう

因为我们在修改的时候根本没有设置get和set,因此在set里要调用监听函数

set(newValue){
    console.log(key,value,newValue);
    value = newValue
    Observer(newValue)
}

Vue で双方向データ バインディングを実装する方法について話しましょう

模板解析

劫持数据后就要把Vue实例里的数据应用带页面上,得要加一个临时内存区域,将所有数据都更新后再渲染页面以此减少dom操作

创建一个解析函数,设置2个参数,一个是Vue实例里挂载的元素,另一个是Vue实例,在函数里获取获取元素保存在实例了的$el里,获取元素后放入临时内存里,需要用到[createDocumentFragment]创建一个新的空白的文档片段

然后把$el的子节点一个一个加到fragment变量里,页面已经没有内容了,内容都被临时存在fragment里了

class Vue{
  constructor(obj_instance){
    this.$data=obj_instance.data
    Observer(this.$data)
    Compile(obj_instance.el,this)
  }
}
function Compile(ele,vm){
  vm.$el=document.querySelector(ele)
  const fragment=document.createDocumentFragment()
  let child;
  while (child=vm.$el.firstChild){
    fragment.append(child)
  }
  console.log(fragment);
  console.log(fragment.childNodes);
}

Vue で双方向データ バインディングを実装する方法について話しましょう

现在直接把需要修改的内容应用到文档碎片里面,应用后重新渲染,只需修改了fragment的childNodes子节点的文本节点,文本节点的类型是3,可以创建一个函数并调用来修改fragment里的内容

节点里面可能还会有节点,因此判定节点类型是否为3,不是就递归调用这个解析函数

节点类型为3就进行修改操作,但也不行把整个节点的文本都修改,只需修改插值表达式的内容,因此要使用正则表达式匹配,将匹配的结果保存到变量里,匹配的结果是一个数组,而索引为1的元素才是我们需要提取出来的元素,这个元素就是去除了{{}}和空格得到的字符串,然后就可以直接用Vue实例来访问对应属性的值,修改完后return出去结束递归

function Compile(ele,vm){
  vm.$el=document.querySelector(ele) //获取元素保存在实例了的$el里
  const fragment=document.createDocumentFragment() //创建文档碎片
  let child;
  while (child=vm.$el.firstChild){//循环将子节点添加到文档碎片里
    fragment.append(child)
  }
  
  fragment_compile(fragment)
  function fragment_compile(node){ //修改文本节点内容
    const pattern = /\{\{\s*(\S*)\s*\}\}/ //检索字符串中正则表达式的匹配,用于匹配插值表达式
    if(node.nodeType===3){
      const result = pattern.exec(node.nodeValue)
      if(result){
        console.log(&#39;result[1]&#39;)
        const value=result[1].split(&#39;.&#39;).reduce(//split将对象里的属性分布在数组里,链式地进行排列;reduce进行累加,层层递进获取$data的值
          (total,current)=>total[current],vm.$data
        )
        node.nodeValue=node.nodeValue.replace(pattern,value) //replace函数将插值表达式替换成$data里的属性的值
      }
      return 
    }
    node.childNodes.forEach(child=>fragment_compile(child))
  }
  vm.$el.appendChild(fragment) //将文档碎片应用到对应的dom元素里面
}

页面的内容又出来了,插值表达式替换成了vm实例里的数据

Vue で双方向データ バインディングを実装する方法について話しましょう

Vue で双方向データ バインディングを実装する方法について話しましょう

订阅发布者模式

虽然进行了数据劫持,和将数据应用到页面上,但是数据发生变动还不能及时更新,还需要实现订阅发布者模式

首先创建一个类用来收集和通知订阅者,生成实例的时候需要有一个数组存放订阅者的信息,一个将订阅者添加到这个数组里的方法和一个通知订阅者的方法,调用这个方法就回去遍历订阅者的数组,让订阅者调用自身的update方法进行更新

class Dependency{
  constructor(){
    this.subscribers=[] //存放订阅者的信息
  }
  addSub(sub){
    this.subscribers.push(sub) //将订阅者添加到这个数组里
  }
  notify(){
    this.subscribers.forEach(sub=>sub.update()) //遍历订阅者的数组,调用自身的update函数进行更新
  }
}

设置订阅者类,需要用到Vue实例上的属性,需要Vue实例和Vue实例对应的属性和一个回调函数作为参数,将参数都赋值给实例

然后就可以创建订阅者的update函数,在函数里调用传进来的回调函数

class Watcher{
  constructor(vm,key,callback){//将参数都赋值给Watcher实例
    this.vm=vm
    this.key=key
    this.callback=callback
  }
  update(){
    this.callback() 
  }
}

替换文档碎片内容的时候需要告诉订阅者如何更新,所以订阅者实例在模板解析把节点值替换内容的时候创建,传入vm实例,exec匹配成功后的索引值1和回调函数,将替换文本的执行语句复制到回调函数里,通知订阅者更新的时候就调用这个回调函数

回调函数里的nodeValue要提前保存,不然替换的内容就不是插值表达式而是替换过的内容

Vue で双方向データ バインディングを実装する方法について話しましょう

然后就要想办法将订阅者存储到Dependency实例的数组里,我们可以在构造Watcher实例的时候保存实例到订阅者数组里

Dependency.temp=this //设置一个临时属性temp

将新的订阅者添加到订阅者数组里且还要将所有的订阅者都进行同样的操作,那么就可以在触发get的时候将订阅者添加到订阅者数组里,为了正确触发对应的属性get,需要用reduce方法对key进行同样的操作

Vue で双方向データ バインディングを実装する方法について話しましょう

Vue で双方向データ バインディングを実装する方法について話しましょう

可以看到控制台打印出了Wathcer实例,每个实例都不同,都对应不同的属性值

Vue で双方向データ バインディングを実装する方法について話しましょう

Dependency类还没创建实例,里面的订阅者数组是不存在的,所以要先创建实例再将订阅者添加到订阅者数组里

Vue で双方向データ バインディングを実装する方法について話しましょう

Vue で双方向データ バインディングを実装する方法について話しましょう

修改数据的时候通知订阅者来进行更新,在set里调用dependency的通知方法,通知方法就会去遍数组,订阅者执行自己的update方法进行数据更新

Vue で双方向データ バインディングを実装する方法について話しましょう

但是update调用回调函数缺少设定形参,依旧使用split和reduce方法获取属性值

update(){
    const value =this.key.split(&#39;.&#39;).reduce(
        (total,current)=>total[current],this.vm.$data
    )
    this.callback(value)
}

在控制台修改属性值都修改成功了,页面也自动更新了

Vue で双方向データ バインディングを実装する方法について話しましょう

完成了文本的绑定就可以绑定输入框了,在vue里通过v-model进行绑定,因此要判断哪个节点有v-model,元素节点的类型是1,可以使用nodeName来匹配input元素,直接在判断文本节点下面进行新的判断

if(node.nodeType===1&&node.nodeName===&#39;INPUT&#39;){
    const attr=Array.from(node.attributes)
    console.log(attr);
}

节点名字nodeName为v-model,nodeValue为name,就是数据里的属性名

Vue で双方向データ バインディングを実装する方法について話しましょう

因此对这个数组进行遍历,匹配到了v-model根据nodeValue找到对应的属性值,把属性值赋值到节点上,同时为了在数据更新后订阅者知道更新自己,也要在INPUT节点里新增Watcher实例

attr.forEach(i=>{
    if(i.nodeName===&#39;v-model&#39;){
        const value=i.nodeValue.split(&#39;.&#39;).reduce(
            (total,current)=>total[current],vm.$data
        )
        node.value=value
        new Watcher(vm,i.nodeValue,newValue=>{
            node.value=newValue
        })
    }
})

修改属性值,页面也作出修改

Vue で双方向データ バインディングを実装する方法について話しましょう

最后剩下用视图改变数据,在v-model的节点上使用addEventListener增加input监听事件就行了

node.addEventListener(&#39;input&#39;,e=>{
    const arr1=i.nodeValue.split(&#39;.&#39;)
    const arr2=arr1.slice(0,arr1.length - 1)
    const final=arr2.reduce(
        (total,current)=>total[current],vm.$data
    )
    final[arr1[arr1.length - 1]]=e.target.value
})

Vue で双方向データ バインディングを実装する方法について話しましょう

(学习视频分享:web前端开发编程基础视频

以上がVue で双方向データ バインディングを実装する方法について話しましょうの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明
この記事は掘金社区で複製されています。侵害がある場合は、admin@php.cn までご連絡ください。
フレームワークの選択:Netflixの決定を推進するものは何ですか?フレームワークの選択:Netflixの決定を推進するものは何ですか?Apr 13, 2025 am 12:05 AM

Netflixは、主に、パフォーマンス、スケーラビリティ、開発効率、エコシステム、技術的な負債、およびフレームワーク選択におけるメンテナンスコストを考慮しています。 1。パフォーマンスとスケーラビリティ:JavaとSpringbootが選択され、大規模なデータと高い同時リクエストを効率的に処理します。 2。開発効率とエコシステム:Reactを使用して、フロントエンド開発効率を向上させ、その豊富なエコシステムを利用します。 3.技術的な負債とメンテナンスコスト:node.jsを選択してマイクロサービスを構築して、メンテナンスコストと技術的債務を削減します。

Netflixのフロントエンドの反応、Vue、および未来Netflixのフロントエンドの反応、Vue、および未来Apr 12, 2025 am 12:12 AM

Netflixは、主にReactをフロントエンドフレームワークとして使用し、特定の機能のためにVUEによって補足されます。 1)Reactのコンポーネント化と仮想DOMは、Netflixアプリケーションのパフォーマンスと開発効率を向上させます。 2)VueはNetflixの内部ツールと小規模プロジェクトで使用されており、その柔軟性と使いやすさが重要です。

フロントエンドのvue.js:実際のアプリケーションと例フロントエンドのvue.js:実際のアプリケーションと例Apr 11, 2025 am 12:12 AM

Vue.jsは、複雑なユーザーインターフェイスを構築するのに適した進歩的なJavaScriptフレームワークです。 1)そのコア概念には、レスポンシブデータ、コンポーネント、仮想DOMが含まれます。 2)実際のアプリケーションでは、TODOアプリケーションを構築し、Vuerouterを統合することで実証できます。 3)デバッグするときは、vuedevtools and Console.logを使用することをお勧めします。 4)パフォーマンスの最適化は、V-IF/V-Show、リストレンダリング最適化、コンポーネントの非同期負荷などを通じて達成できます。

Vue.jsとReact:重要な違​​いを理解するVue.jsとReact:重要な違​​いを理解するApr 10, 2025 am 09:26 AM

Vue.JSは中小企業から中規模のプロジェクトに適していますが、Reactは大規模で複雑なアプリケーションにより適しています。 1。VUE.JSのレスポンシブシステムは、依存関係追跡を介してDOMを自動的に更新し、データの変更を簡単に管理できるようにします。 2.反応は一方向のデータフローを採​​用し、データは親コンポーネントから子コンポーネントに流れ、明確なデータフローと簡単な抽出構造を提供します。

Vue.js vs. React:プロジェクト固有の考慮事項Vue.js vs. React:プロジェクト固有の考慮事項Apr 09, 2025 am 12:01 AM

VUE.JSは、中小規模のプロジェクトや迅速な反復に適していますが、Reactは大規模で複雑なアプリケーションに適しています。 1)Vue.jsは使いやすく、チームが不十分な状況やプロジェクトスケールが小さい状況に適しています。 2)Reactにはより豊富なエコシステムがあり、高性能で複雑な機能的ニーズを持つプロジェクトに適しています。

Vueにタグをジャンプする方法Vueにタグをジャンプする方法Apr 08, 2025 am 09:24 AM

VUEでタグのジャンプを実装する方法には、HTMLテンプレートでAタグを使用してHREF属性を指定する方法が含まれます。 VUEルーティングのルーターリンクコンポーネントを使用します。 JavaScriptでこれを使用します。$ router.push()メソッド。パラメーターはクエリパラメーターに渡すことができ、ルートは動的ジャンプのルーターオプションで構成されています。

VUEのコンポーネントジャンプを実装する方法VUEのコンポーネントジャンプを実装する方法Apr 08, 2025 am 09:21 AM

VUEでコンポーネントジャンプを実装するための次の方法があります。Router-Linkと&lt; router-view&gt;を使用してください。ハイパーリンクジャンプを実行し、ターゲットパスとして属性を指定するコンポーネント。 &lt; router-view&gt;を使用してください現在ルーティングされているレンダリングされているコンポーネントを表示するコンポーネント。プログラマティックナビゲーションには、router.push()およびrouter.replace()メソッドを使用します。前者は歴史を保存し、後者は記録を残さずに現在のルートに取って代わります。

VueのDivにジャンプする方法VueのDivにジャンプする方法Apr 08, 2025 am 09:18 AM

VUEにDIV要素をジャンプするには、VUEルーターを使用してルーターリンクコンポーネントを追加するには、2つの方法があります。 @clickイベントリスナーを追加して、これを呼び出します。$ router.push()メソッドをジャンプします。

See all articles

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

mPDF

mPDF

mPDF は、UTF-8 でエンコードされた HTML から PDF ファイルを生成できる PHP ライブラリです。オリジナルの作者である Ian Back は、Web サイトから「オンザフライ」で PDF ファイルを出力し、さまざまな言語を処理するために mPDF を作成しました。 HTML2FPDF などのオリジナルのスクリプトよりも遅く、Unicode フォントを使用すると生成されるファイルが大きくなりますが、CSS スタイルなどをサポートし、多くの機能強化が施されています。 RTL (アラビア語とヘブライ語) や CJK (中国語、日本語、韓国語) を含むほぼすべての言語をサポートします。ネストされたブロックレベル要素 (P、DIV など) をサポートします。

WebStorm Mac版

WebStorm Mac版

便利なJavaScript開発ツール

VSCode Windows 64 ビットのダウンロード

VSCode Windows 64 ビットのダウンロード

Microsoft によって発売された無料で強力な IDE エディター

EditPlus 中国語クラック版

EditPlus 中国語クラック版

サイズが小さく、構文の強調表示、コード プロンプト機能はサポートされていません

MantisBT

MantisBT

Mantis は、製品の欠陥追跡を支援するために設計された、導入が簡単な Web ベースの欠陥追跡ツールです。 PHP、MySQL、Web サーバーが必要です。デモおよびホスティング サービスをチェックしてください。