data:image/s3,"s3://crabby-images/1ff79/1ff7933c5cae7fdd0b19ad98ef6cda7892d15850" alt="實現一個簡易的vue"
1./compiler ⽬目錄是編譯模版;
2./core ⽬目錄是 Vue.js 的核⼼心(也是後⾯面的重點);
3./platforms ⽬目錄是針對核⼼心模組的 ‘平臺’ 模組;
4./server ⽬目錄是處理理服務端渲染;
5./sfc ⽬目錄處理理單⽂檔案 .vue;
6./shared ⽬目錄提供全域性⽤用到的⼯工具函式。
Vue.js 的組成是由 core + 對應的 ‘平臺’ 補充程式碼構成(獨立構建和執行時構建 只是 platforms 下 web 平臺的兩種選擇)。
data:image/s3,"s3://crabby-images/3611c/3611c6755b14d3e04e1fa7d160504fc8c88674cb" alt="實現一個簡易的vue"
vue的雙向資料繫結
雙向繫結(響應式原理)所涉及到的技術
1. Object.defineProperty
2. Observer
3. Watcher
4. Dep
5. Directive
複製程式碼
1. Object.defineProperty
var obj = {};
var a;
Object.defineProperty(obj,'a',{
get: function(){
console.log('get val');
return a;
},
set: function(newVal){
console.log('set val:' + newVal);
a = newVal;
}
});
obj.a // get val; 相當於<span>{{a}}</span>
obj.a = '111'; // set val:111 相當於<input v-model="a">
複製程式碼
data:image/s3,"s3://crabby-images/cf0bf/cf0bf4dc55df9719ce42fccd34140369e7c335ff" alt="實現一個簡易的vue"
2. Observer
觀察者模式是軟體設計模式的一種。
在此種模式中,一個目標物件管理所有相依於它的觀 察者物件,並且在它本身的狀態改變時主動發出通知。
這通常透過呼叫各觀察者所提供的 方法來實現。此種模式通常被用來實時事件處理系統。
訂閱者模式涉及三個物件:
釋出者、主題物件、訂閱者,三個物件間的是一對多的關係,
每當主題物件狀態發生改變時,其相關依賴物件都會得到通知,並被自動更新。
看一個簡單的示例:
複製程式碼
data:image/s3,"s3://crabby-images/c15fe/c15fe56b75fb122f37fe71df87b4eaba0c472c6f" alt="實現一個簡易的vue"
data:image/s3,"s3://crabby-images/33053/33053116305a24ff50fbe96dd3b6b709661ef573" alt="實現一個簡易的vue"
data:image/s3,"s3://crabby-images/e1382/e138293b41225221382cd72c529898666d2ef2a2" alt="實現一個簡易的vue"
data:image/s3,"s3://crabby-images/36608/36608f60b664ec78881f26ec243bdb702f3f4785" alt="實現一個簡易的vue"
data:image/s3,"s3://crabby-images/e7437/e7437a241f678e8a759e23ed861b82eddf20ed68" alt="實現一個簡易的vue"
data:image/s3,"s3://crabby-images/4b0c1/4b0c1498268c0cf1183efa698a1bc241b8cd487c" alt="實現一個簡易的vue"
data:image/s3,"s3://crabby-images/4df87/4df87ce0c750b669d92b27b5ed08f889b61e9d20" alt="實現一個簡易的vue"
- Dep
data:image/s3,"s3://crabby-images/64aa2/64aa2227871c3e8361a9316e7d71f9d0fb47c7cb" alt="實現一個簡易的vue"
- Directive
data:image/s3,"s3://crabby-images/57a05/57a05a5d85c2ad94cee87adb6f3bc7c385add17e" alt="實現一個簡易的vue"
data:image/s3,"s3://crabby-images/fd4e4/fd4e437ac509032e0b8d28326d16a424e1159e26" alt="實現一個簡易的vue"
data:image/s3,"s3://crabby-images/5f5bc/5f5bc4db775f9569439c21e97c204b6e343f13ca" alt="實現一個簡易的vue"
data:image/s3,"s3://crabby-images/c7bc1/c7bc1c9b603e8dfd22b8be8b96d56b354617cf5d" alt="實現一個簡易的vue"
弄明白原理和架構之後,我們來實現一個簡單的vue雙向資料繫結
data:image/s3,"s3://crabby-images/e7d65/e7d656e6d2b6b7f3adf0dfcfd56da5bb9fdcd4f0" alt="實現一個簡易的vue"
1.這個Vue是從哪裡來的呢?
data:image/s3,"s3://crabby-images/2a2ad/2a2ad6f5a439b298e3c97cd262f7f751449f0256" alt="實現一個簡易的vue"
data:image/s3,"s3://crabby-images/66a51/66a51aeaa8710229e6c391354b252c3a252a62e6" alt="實現一個簡易的vue"
通過set,get來設定值與獲取值
把text屬性繫結到vue例項上面去使用
那其中的Dep又是什麼呢?
data:image/s3,"s3://crabby-images/8460d/8460d860343c77b293231274549ece9fe9475802" alt="實現一個簡易的vue"
再來看一下Compile中寫的什麼吧
function Compile(node, vm) {
if (node) {
this.$frag = this.nodeToFragment(node, vm);
return this.$frag;
}
}
Compile.prototype = {
nodeToFragment: function (node, vm) {
var self = this;
var frag = document.createDocumentFragment(); // 建立一段html文件片段
var child;
while (child = node.firstChild) {
self.compileElement(child, vm);
frag.append(child); // 將所有子節點新增到fragment中
}
return frag;
},
compileElement: function (node, vm) {
var reg = /\{\{(.*)\}\}/;
//節點型別為元素
if (node.nodeType === 1) {
var attr = node.attributes;
// 解析屬性
for (var i = 0; i < attr.length; i++) {
if (attr[i].nodeName == 'v-model') {
var name = attr[i].nodeValue; // 獲取v-model繫結的屬性名
node.addEventListener('input', function (e) {
// 給相應的data屬性賦值,進而觸發該屬性的set方法
// 觸發set vm[name]
vm[name] = e.target.value;
});
// node.value = vm[name]; // 將data的值賦給該node
new Watcher(vm, node, name, 'value');
}
};
}
//節點型別為text
if (node.nodeType === 3) {
if (reg.test(node.nodeValue)) {
var name = RegExp.$1; // 獲取匹配到的字串
name = name.trim();
// node.nodeValue = vm[name]; // 將data的值賦給該node
new Watcher(vm, node, name, 'nodeValue');
}
}
},
}
複製程式碼
哦,原來Compile中是渲染html的啊。其中的Watcher是不是監控節點變化,然後給Dep通知的呢?
function Watcher(vm, node, name, type) {
Dep.target = this;
this.name = name; //text
this.node = node; // 節點
this.vm = vm; // vue例項
this.type = type; //nodeValue 當前節點的值
this.update();
Dep.target = null;
}
Watcher.prototype = {
update: function() {
this.get();
var batcher = new Batcher();
batcher.push(this);
// this.node[this.type] = this.value; // 訂閱者執行相應操作
},
cb:function(){
this.node[this.type] = this.value; // 訂閱者執行相應操作
},
// 獲取data的屬性值
get: function() {
this.value = this.vm[this.name]; //觸發相應屬性的get
}
}
複製程式碼
哎呦,我們們猜對了呢,這樣雙向資料繫結馬上就要完成了,只剩一個Vue.nextTick()的地方了
/**
* 批處理建構函式
* @constructor
*/
function Batcher() {
this.reset();
}
/**
* 批處理重置
*/
Batcher.prototype.reset = function () {
this.has = {};
this.queue = [];
this.waiting = false;
};
/**
* 將事件新增到佇列中
* @param job {Watcher} watcher事件
*/
Batcher.prototype.push = function (job) {
if (!this.has[job.name]) {
this.queue.push(job);
this.has[job.name] = job;
if (!this.waiting) {
this.waiting = true;
setTimeout(() => {
this.flush();
});
}
}
};
/**
* 執行並清空事件佇列
*/
Batcher.prototype.flush = function () {
this.queue.forEach((job) => {
job.cb();
});
this.reset();
};
複製程式碼
看完後是不是覺得超簡單呢?
vue3版本將做出巨大的變化,把Dep跟Watcher都幹掉了,html直接跟資料進行繫結,等vue3出來後,在寫一篇關於vue的文章吧
看完後能幫我點個贊嗎?