이 글은 Vue의 반응형 원칙과 의존성 컬렉션(코드 포함)을 소개합니다. 도움이 필요한 친구들이 참고할 수 있기를 바랍니다.
Vue는 객체 속성의 setter/getter 메서드를 설정하여 데이터 변경 사항을 모니터링하고 getter를 통해 종속성을 수집합니다. 각 setter 메서드는 관찰자로서 데이터가 변경되면 뷰를 업데이트하도록 구독자에게 알립니다.
데이터를 관찰 가능으로 전환
그렇다면 Vue는 어떻게 데이터 아래의 모든 속성을 관찰 가능으로 전환할까요?
function obsever(value,cb){ Object.keys(value).forEach((key)=>defineReactive(value,key,value[key],cb)) } function defineReactive(obj,key,val,cb){ Object.defineProperty(obj,key,{ enumerable;true, configurable:true, get:()=>{ /*依赖收集*/ return val; }, set:newVal=>{ val=newVal; cb(); } }) } class Vue{ constructor(options){ this._data = options.data; obsever(this._data,options.render) } } let app = new Vue({ el:'#app', data:{ text:'text', text2:'text2' }, render(){ console.log('render') } })
이해를 돕기 위해 먼저 배열이나 기타 상황을 고려하지 않고 가장 간단한 경우를 생각해 보겠습니다. initData에서 관찰 함수는 Vue 데이터를 관찰 가능으로 설정하기 위해 호출됩니다. _data 데이터가 변경되면 set이 트리거되고 구독자에게 콜백이 수행됩니다(이 경우 렌더링).
그런 다음 문제가 발생합니다. app._data.text를 실행하여 설정을 실행해야 합니다. 게으르게 하기 위해서는 app.text를 통해 직접 설정하여 뷰를 다시 그리도록 설정을 트리거하는 편리한 방법이 필요합니다. 그러면 상담원에게 가셔야 합니다.
Proxy
Vue의 생성자 생성자에서 데이터에 대한 프록시를 실행할 수 있습니다. 이러한 방식으로 데이터의 속성을 vm 인스턴스로 프록시합니다.
_proxy.call(this,options.data);//构造函数 //代理 function _proxy(data){ const that = this; Object.keys(data).forEach(key=>{ configurable:true, enumerable:true, get:function proxyGetter(){ return that._data[key] }, set:function proxySetter(val){ that._data[key] = val; } }) }
app._data.text 대신 app.text를 사용할 수 있습니다.
컬렉션에 의존하는 이유
다음 코드를 먼저 보세요
new Vue({ template:`<p> <span>text1:</span>{{text1}} <span>text2:</span>{{tetx2}} </p>`, data:{ text1:'text1', text2:'text2', text3:'text3' } })
위의 반응형 원칙에 따라 바인딩하면 문제가 발생합니다. 실제 템플릿에서는 text3이 사용되지 않지만 text3의 데이터가 수정되면 text3의 setter도 트리거되어 렌더링이 다시 실행되는데 이는 분명히 잘못된 것입니다.
먼저 Dep에 대해 이야기합시다
데이터의 객체 값이 수정되면 해당 setter가 트리거됩니다. 그런 다음 값을 얻으면 getter 이벤트가 자연스럽게 트리거되므로 렌더링만 하면 됩니다. 처음에 한 번, 렌더링이 의존하는 데이터의 모든 데이터는 getter에 의해 Dep의 하위 항목으로 수집됩니다. 데이터의 데이터를 수정할 때 setter는 Dep의 subs 기능만 트리거합니다.
컬렉션 클래스에 따라 Dep를 정의합니다.
class Dep{ constructor(){ this.subs = []; } addSub(sub:Watcher){ this.subs.push(sub) } removeSub(sub:Watcher){ remove(this.subs,sub) } notify(){ const subs = this.subs.slice() for(let i = 0;l=subs.length;i<1;i++){ subs[i].update() } } } function remove(arr,item){ if(arr.length){ const index = arr.indexOf(item) if(index>-1){ return arr.splice(index,1) } } }
Watcher
구독자는 종속성이 수집되면 sub에 Sub를 추가합니다. 데이터의 데이터가 수정되면 dep 개체에 대한 알림이 트리거되어 모든 Watcher 개체에 해당 뷰를 수정하도록 알립니다.
class Watcher { constructor (vm, expOrFn, cb, options) { this.cb = cb; this.vm = vm; /*在这里将观察者本身赋值给全局的target,只有被target标记过的才会进行依赖收集*/ Dep.target = this; /*Github:https://github.com/answershuto*/ /*触发渲染操作进行依赖收集*/ this.cb.call(this.vm); } update () { this.cb.call(this.vm); } }
종속성 수집 시작
class Vue { constructor(options) { this._data = options.data; observer(this._data, options.render); let watcher = new Watcher(this, ); } } function defineReactive (obj, key, val, cb) { /*在闭包内存储一个Dep对象*/ const dep = new Dep(); Object.defineProperty(obj, key, { enumerable: true, configurable: true, get: ()=>{ if (Dep.target) { /*Watcher对象存在全局的Dep.target中*/ dep.addSub(Dep.target); } }, set:newVal=> { /*只有之前addSub中的函数才会触发*/ dep.notify(); } }) } Dep.target = null;
Watcher 인스턴스를 전역 Dep.target에 할당한 다음 렌더링 작업을 트리거하면 Dep.target으로 표시된 종속성만 수집됩니다. Dep.target이 있는 개체는 Watcher 인스턴스를 subs로 푸시합니다. 개체가 수정되고 setter 작업을 트리거하면 dep는 렌더링을 위해 subs에서 Watcher 인스턴스의 업데이트 메서드를 호출합니다.
이 기사는 여기서 끝났습니다. 더 흥미로운 내용을 보려면 PHP 중국어 웹사이트의 JavaScript 비디오 튜토리얼 칼럼을 주목하세요!위 내용은 Vue 응답성 원칙 및 종속성 수집 소개(코드 포함)의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!