實現之前我們得先看一下Object.defineProperty的實現,因為vue主要是通過數據劫持來實現的,通過get、set來完成數據的讀取和更新。
var obj={name:'wclimb'} var age=24 Object.defineProperty(obj,'age',{ enumerable: true, // 可枚舉 configurable: false, // 不能再define get () { return age }, set (newVal) { console.log('我改變了',age +' -> '+newVal); age=newVal } }) > obj.age > 24 > obj.age=25; > 我改變了 24 -> 25 > 25
從上面可以看到通過get獲取數據,通過set監聽到數據變化執行相應操作,還是不明白的話可以去看看Object.defineProperty文檔。
<div id="wrap"> <p v-html="test"></p> <input type="text" v-model="form"> <input type="text" v-model="form"> <button @click="changeValue">改變值</button> {{form}} </div>
new Vue({ el: '#wrap', data:{ form: '這是form的值', test: '<strong>我是粗體</strong>', }, methods:{ changeValue(){ console.log(this.form) this.form='值被我改變了,氣不氣?' } } })
class Vue{ constructor(){} proxyData(){} observer(){} compile(){} compileText(){} } class Watcher{ constructor(){} update(){} }
class Vue{ constructor(options={}){ this.$el=document.querySelector(options.el); let data=this.data=options.data; // 代理data,使其能直接this.xxx的方式訪問data,正常的話需要this.data.xxx Object.keys(data).forEach((key)=> { this.proxyData(key); }); this.methods=obj.methods // 事件方法 this.watcherTask={}; // 需要監聽的任務列表 this.observer(data); // 初始化劫持監聽所有數據 this.compile(this.$el); // 解析dom } }
上面主要是初始化操作,針對傳過來的數據進行處理
class Vue{ constructor(options={}){ ...... } proxyData(key){ let that=this; Object.defineProperty(that, key, { configurable: false, enumerable: true, get () { return that.data[key]; }, set (newVal) { that.data[key]=newVal; } }); } }
上面主要是代理data到最上層,this.xxx的方式直接訪問data
class Vue{ constructor(options={}){ ...... } proxyData(key){ ...... } observer(data){ let that=this Object.keys(data).forEach(key=>{ let value=data[key] this.watcherTask[key]=[] Object.defineProperty(data,key,{ configurable: false, enumerable: true, get(){ return value }, set(newValue){ if(newValue !==value){ value=newValue that.watcherTask[key].forEach(task=> { task.update() }) } } }) }) } }
同樣是使用Object.defineProperty來監聽數據,初始化需要訂閱的數據。
把需要訂閱的數據到push到watcherTask里,等到時候需要更新的時候就可以批量更新數據了。??下面就是;
遍歷訂閱池,批量更新視圖。
set(newValue){ if(newValue !==value){ value=newValue // 批量更新視圖 that.watcherTask[key].forEach(task=> { task.update() }) } } compile 解析dom class Vue{ constructor(options={}){ ...... } proxyData(key){ ...... } observer(data){ ...... } compile(el){ var nodes=el.childNodes; for (let i=0; i < nodes.length; i++) { const node=nodes[i]; if(node.nodeType===3){ var text=node.textContent.trim(); if (!text) continue; this.compileText(node,'textContent') }else if(node.nodeType===1){ if(node.childNodes.length > 0){ this.compile(node) } if(node.hasAttribute('v-model') && (node.tagName==='INPUT' || node.tagName==='TEXTAREA')){ node.addEventListener('input',(()=>{ let attrVal=node.getAttribute('v-model') this.watcherTask[attrVal].push(new Watcher(node,this,attrVal,'value')) node.removeAttribute('v-model') return ()=> { this.data[attrVal]=node.value } })()) } if(node.hasAttribute('v-html')){ let attrVal=node.getAttribute('v-html'); this.watcherTask[attrVal].push(new Watcher(node,this,attrVal,'innerHTML')) node.removeAttribute('v-html') } this.compileText(node,'innerHTML') if(node.hasAttribute('@click')){ let attrVal=node.getAttribute('@click') node.removeAttribute('@click') node.addEventListener('click',e=> { this.methods[attrVal] && this.methods[attrVal].bind(this)() }) } } } }, compileText(node,type){ let reg=/\{\{(.*)\}\}/g, txt=node.textContent; if(reg.test(txt)){ node.textContent=txt.replace(reg,(matched,value)=>{ let tpl=this.watcherTask[value] || [] tpl.push(new Watcher(node,this,value,type)) return value.split('.').reduce((val, key)=> { return this.data[key]; }, this.$el); }) } } }
這里代碼比較多,我們拆分看你就會覺得很簡單了
首先我們先遍歷el元素下面的所有子節點,node.nodeType===3 的意思是當前元素是文本節點,node.nodeType===1 的意思是當前元素是元素節點。因為可能有的是純文本的形式,如純雙花括號就是純文本的文本節點,然后通過判斷元素節點是否還存在子節點,如果有的話就遞歸調用compile方法。下面重頭戲來了,我們拆開看:
if(node.hasAttribute('v-html')){ let attrVal=node.getAttribute('v-html'); this.watcherTask[attrVal].push(new Watcher(node,this,attrVal,'innerHTML')) node.removeAttribute('v-html') }
上面這個首先判斷node節點上是否有v-html這種指令,如果存在的話,我們就發布訂閱,怎么發布訂閱呢?只需要把當前需要訂閱的數據push到watcherTask里面,然后到時候在設置值的時候就可以批量更新了,實現雙向數據綁定,也就是下面的操作
that.watcherTask[key].forEach(task=> { task.update() })
然后push的值是一個Watcher的實例,首先他new的時候會先執行一次,執行的操作就是去把純雙花括號 -> 1,也就是說把我們寫好的模板數據更新到模板視圖上。
最后把當前元素屬性剔除出去,我們用Vue的時候也是看不到這種指令的,不剔除也不影響
至于Watcher是什么,看下面就知道了
that.watcherTask[key].forEach(task=> { task.update() })
之前發布訂閱之后走了這里面的操作,意思就是把當前元素如:node.innerHTML='這是data里面的值'、node.value='這個是表單的數據'
那么我們為什么不直接去更新呢,還需要update做什么,不是多此一舉嗎?
其實update記得嗎?我們在訂閱池里面需要批量更新,就是通過調用Watcher原型上的update方法。
寫在最后:歡迎留言討論,關于Java架構進階私信“Java”或“架構資料”有驚喜!加關注,持續更新!
ue的使用相信大家都很熟練了,使用起來簡單。但是大部分人不知道其內部的原理是怎么樣的,今天我們就來一起實現一個簡單的vue。實現之前我們得先看一下Object.defineProperty()的實現,因為vue主要是通過數據劫持來實現的,通過get、set來完成數據的讀取和更新。
從上面可以看到通過get獲取數據,通過set監聽到數據變化執行相應操作,還是不明白的話可以去看看Object.defineProperty文檔。
流程圖
html代碼結構
js調用
Vue結構
Vue constructor 初始化
proxyData 代理data
observer 劫持監聽
同樣是使用Object.defineProperty來監聽數據,初始化需要訂閱的數據。把需要訂閱的數據到push到watcherTask里,等到時候需要更新的時候就可以批量更新數據了。
uePress 是一個以 Markdown 為中心的靜態網站生成器。你可以使用 Markdown 來書寫內容(如文檔、博客等),然后 VuePress 會幫助你生成一個靜態網站來展示它們。
VuePress 誕生的初衷是為了支持 Vue.js 及其子項目的文檔需求,但是現在它已經在幫助大量用戶構建他們的文檔、博客和其他靜態網站。
首先,安裝 VuePress 最新版本及相關依賴:
# 安裝 VuePress
npm install -D vuepress@next
# 安裝 Vite 打包工具和默認主題
npm install -D @vuepress/bundler-vite@next @vuepress/theme-default@next
以下是安裝過程中的示例輸出:
virhuiai@virhuiaideMBP vite-project % npm install -D vuepress@next
(#########?????????) ? idealTree:vite-project: timing idealTree:#root Completed in 16997ms
added 112 packages, audited 158 packages in 2m
31 packages are looking for funding
run `npm fund` for details
found 0 vulnerabilities
virhuiai@virhuiaideMBP vite-project %
virhuiai@virhuiaideMBP vite-project % npm install -D @vuepress/bundler-vite@next @vuepress/theme-default@next
added 79 packages, audited 237 packages in 1m
63 packages are looking for funding
run `npm fund` for details
found 0 vulnerabilities
創建必要的目錄和配置文件:
# 創建文檔目錄和配置目錄
mkdir docs
mkdir docs/.vuepress
創建 VuePress 配置文件:
# docs/.vuepress/config.js
import { viteBundler } from '@vuepress/bundler-vite'
import { defaultTheme } from '@vuepress/theme-default'
import { defineUserConfig } from 'vuepress'
export default defineUserConfig({
bundler: viteBundler(),
theme: defaultTheme(),
})
創建你的第一篇文檔:
echo '# Hello VuePress' > docs/README.md
項目目錄結構如下:
├─ docs
│ ├─ .vuepress
│ │ └─ config.js
│ └─ README.md
└─ package.json
建議的 .gitignore 配置如下:
# 忽略 VuePress 生成的臨時文件和緩存
.vuepress/.temp
.vuepress/.cache
.vuepress/dist
在 package.json 中配置啟動和構建腳本:
{
"scripts": {
"docs:dev": "vuepress dev docs",
"docs:build": "vuepress build docs"
}
}
啟動開發服務器:
npm run docs:dev
構建靜態網站:
npm run docs:build
安裝 ElementPlus:
npm i element-plus -S
配置VuePress以使用ElementPlus:
# docs/.vuepress/client.js
import { defineClientConfig } from 'vuepress/client'
import ElementPlus from 'element-plus';
import 'element-plus/dist/index.css'
export default defineClientConfig({
enhance({ app, router, siteData }) {
app.use(ElementPlus);
},
setup() {},
rootComponents: [],
})
在 Markdown 文件中使用 ElementPlus 組件:
<el-button type="primary">Primary</el-button>
<el-button type="success">Success</el-button>
<el-button type="info">Info</el-button>
<el-button type="warning">Warning</el-button>
<el-button type="danger">Danger</el-button>
重新啟動開發服務器查看效果:
*請認真填寫需求信息,我們會在24小時內與您取得聯系。