摘要:與大多數(shù)全局對象不同,沒有構(gòu)造函數(shù)。為什么要設(shè)計(jì)更加有用的返回值早期寫法寫法函數(shù)式操作早期寫法寫法可變參數(shù)形式的構(gòu)造函數(shù)一般寫法寫法當(dāng)然還有很多,大家可以自行到上查看什么是代理設(shè)計(jì)模式代理模式,為其他對象提供一種代理以控制對這個對象的訪問。
這是專門探索 JavaScript 及其所構(gòu)建的組件的系列文章的第 19 篇。
如果你錯過了前面的章節(jié),可以在這里找到它們:
想閱讀更多優(yōu)質(zhì)文章請猛戳GitHub博客,一年百來篇優(yōu)質(zhì)文章等著你!
JavaScript 是如何工作的:引擎,運(yùn)行時和調(diào)用堆棧的概述!
JavaScript 是如何工作的:深入V8引擎&編寫優(yōu)化代碼的5個技巧!
JavaScript 是如何工作的:內(nèi)存管理+如何處理4個常見的內(nèi)存泄漏!
JavaScript 是如何工作的:事件循環(huán)和異步編程的崛起+ 5種使用 async/await 更好地編碼方式!
JavaScript 是如何工作的:深入探索 websocket 和HTTP/2與SSE +如何選擇正確的路徑!
JavaScript 是如何工作的:與 WebAssembly比較 及其使用場景!
JavaScript 是如何工作的:Web Workers的構(gòu)建塊+ 5個使用他們的場景!
JavaScript 是如何工作的:Service Worker 的生命周期及使用場景!
JavaScript 是如何工作的:Web 推送通知的機(jī)制!
JavaScript 是如何工作的:使用 MutationObserver 跟蹤 DOM 的變化!
JavaScript 是如何工作的:渲染引擎和優(yōu)化其性能的技巧!
JavaScript 是如何工作的:深入網(wǎng)絡(luò)層 + 如何優(yōu)化性能和安全!
JavaScript 是如何工作的:CSS 和 JS 動畫底層原理及如何優(yōu)化它們的性能!
JavaScript 是如何工作的:解析、抽象語法樹(AST)+ 提升編譯速度5個技巧!
JavaScript 是如何工作的:深入類和繼承內(nèi)部原理+Babel和 TypeScript 之間轉(zhuǎn)換!
JavaScript 是如何工作的:存儲引擎+如何選擇合適的存儲API!
JavaScript 是如何工作的:Shadow DOM 的內(nèi)部結(jié)構(gòu)+如何編寫?yīng)毩⒌慕M件!
JavaScript 是如何工作的:WebRTC 和對等網(wǎng)絡(luò)的機(jī)制!
響應(yīng)式原理Proxy 允許我們創(chuàng)建一個對象的虛擬代理(替代對象),并為我們提供了在訪問或修改原始對象時,可以進(jìn)行攔截的處理方法(handler),如 set()、get() 和 deleteProperty() 等等,這樣我們就可以避免很常見的這兩種限制(vue 中):
添加新的響應(yīng)性屬性要使用 Vue.$set(),刪除現(xiàn)有的響應(yīng)性屬性要使用
數(shù)組的更新檢測
Proxylet proxy = new Proxy(target, habdler);
target:用 Proxy 包裝的目標(biāo)對象(可以是數(shù)組對象,函數(shù),或者另一個代理)
handler:一個對象,攔截過濾代理操作的函數(shù)
實(shí)例方法
方法 | 描述 |
---|---|
handler.apply() | 攔截 Proxy 實(shí)例作為函數(shù)調(diào)用的操作 |
handler.construct() | 攔截 Proxy 實(shí)例作為函數(shù)調(diào)用的操作 |
handler.defineProperty() | 攔截 Object.defineProperty() 的操作 |
handler.deleteProperty() | 攔截 Proxy 實(shí)例刪除屬性操作 |
handler.get() | 攔截 讀取屬性的操作 |
handler.set() | 截 屬性賦值的操作 |
handler.getOwnPropertyDescriptor() | 攔截 Object.getOwnPropertyDescriptor() 的操作 |
handler.getPrototypeOf() | 攔截 獲取原型對象的操作 |
handler.has() | 攔截 屬性檢索操作 |
handler.isExtensible() | 攔截 Object.isExtensible() 操作 |
handler.ownKeys() | 攔截 Object.getOwnPropertyDescriptor() 的操作 |
handler.preventExtension() | 截 Object().preventExtension() 操作 |
handler.setPrototypeOf() | 攔截Object.setPrototypeOf()操作 |
Proxy.revocable() | 創(chuàng)建一個可取消的 Proxy 實(shí)例 |
Reflect 是一個內(nèi)置的對象,它提供攔截 JavaScript 操作的方法。這些方法與處理器對象的方法相同。Reflect不是一個函數(shù)對象,因此它是不可構(gòu)造的。
與大多數(shù)全局對象不同,Reflect沒有構(gòu)造函數(shù)。你不能將其與一個new運(yùn)算符一起使用,或者將Reflect對象作為一個函數(shù)來調(diào)用。Reflect的所有屬性和方法都是靜態(tài)的(就像Math對象)。
為什么要設(shè)計(jì) Reflect ?
1. 更加有用的返回值
早期寫法:
try { Object.defineProperty(target, property, attributes); // success } catch (e) { // failure }
Reflect 寫法:
if (Reflect.defineProperty(target, property, attributes)) { // success } else { // failure }
2. 函數(shù)式操作
早期寫法:
"name" in Object //true
Reflect 寫法:
Reflect.has(Object,"name") //true
3. 可變參數(shù)形式的構(gòu)造函數(shù)
一般寫法:
var obj = new F(...args)
Reflect 寫法:
var obj = Reflect.construct(F, args)
當(dāng)然還有很多,大家可以自行到 MND 上查看
什么是代理設(shè)計(jì)模式代理模式(Proxy),為其他對象提供一種代理以控制對這個對象的訪問。代理模式使得代理對象控制具體對象的引用。代理幾乎可以是任何對象:文件,資源,內(nèi)存中的對象,或者是一些難以復(fù)制的東西。現(xiàn)實(shí)生活中的一個類比可能是銀行賬戶的訪問權(quán)限。
例如,你不能直接訪問銀行帳戶余額并根據(jù)需要更改值,你必需向擁有此權(quán)限的人(在本例中 你存錢的銀行)詢問。
var account = { balance: 5000 } var bank = new Proxy(account, { get: function (target, prop) { return 9000000; } }); console.log(account.balance); // 5,000 console.log(bank.balance); // 9,000,000 console.log(bank.currency); // 9,000,000
在上面的示例中,當(dāng)使用 bank 對象訪問 account 余額時,getter 函數(shù)被重寫,它總是返回 9,000,000 而不是屬性值,即使屬性不存在。
var bank = new Proxy(account, { set: function (target, prop, value) { // Always set property value to 0 return Reflect.set(target, prop, 0); } }); account.balance = 5800; console.log(account.balance); // 5,800 bank.balance = 5400; console.log(account.balance); // 0
通過重寫 set 函數(shù),可以修改其行為。可以更改要設(shè)置的值,更改其他屬性,甚至根本不執(zhí)行任何操作。
響應(yīng)式現(xiàn)在已經(jīng)對代理設(shè)計(jì)模式的工作方式有了基本心,讓就開始編寫 JavaScript 框架吧。
為了簡單起見,將模擬 AngularJS 語法。聲明控制器并將模板元素綁定到控制器屬性:
首先,定義一個帶有屬性的控制器,然后在模板中使用這個控制器。最后,使用 ng-bind 屬性啟用與元素值的雙向綁定。
解析模板并實(shí)例化控制器要使屬性綁定,需要獲得一個控制器來聲明這些屬性, 因此,有必要定義一個控制器并將其引入框架中。
在控制器聲明期間,框架將查找?guī)в?ng-controller 屬性的元素。
如果它符合其中一個已聲明的控制器,它將創(chuàng)建該控制器的新實(shí)例,這個控制器實(shí)例只負(fù)責(zé)這個特定的模板。
var controllers = {}; var addController = function (name, constructor) { // Store controller constructor controllers[name] = { factory: constructor, instances: [] }; // Look for elements using the controller var element = document.querySelector("[ng-controller=" + name + "]"); if (!element){ return; // No element uses this controller } // Create a new instance and save it var ctrl = new controllers[name].factory; controllers[name].instances.push(ctrl); // Look for bindings..... }; addController("InputController", InputController);
這是手動處理的控制器變量聲明。 controllers 對象包含通過調(diào)用 addController 在框架內(nèi)聲明的所有控制器。
對于每個控制器,保存一個 factory 函數(shù),以便在需要時實(shí)例化一個新控制器,該框架還存儲模板中使用的相同控制器的每個新實(shí)例。
查找 bind 屬性現(xiàn)在,已經(jīng)有了控制器的一個實(shí)例和使用這個實(shí)例的一個模板,下一步是查找具有使用控制器屬性的綁定的元素。
var bindings = {}; // Note: element is the dom element using the controller Array.prototype.slice.call(element.querySelectorAll("[ng-bind]")) .map(function (element) { var boundValue = element.getAttribute("ng-bind"); if(!bindings[boundValue]) { bindings[boundValue] = { boundValue: boundValue, elements: [] } } bindings[boundValue].elements.push(element); });
上述中,它存儲對象的所有綁的值定。該變量包含要與當(dāng)前值綁定的所有屬性和綁定該屬性的所有 DOM 元素。
雙向綁定在框架完成了初步工作之后,接下就是有趣的部分:雙向綁定。它涉及到將 controller 屬性綁定到 DOM 元素,以便在代碼更新屬性值時更新 DOM。
另外,不要忘記將 DOM 元素綁定到 controller 屬性。這樣,當(dāng)用戶更改輸入值時,它將更新 controller 屬性,接著,它還將更新綁定到此屬性的所有其他元素。
使用代理檢測代碼的更新如上所述,Vue3 組件中通過封裝 proxy 監(jiān)聽響應(yīng)屬性更改。 這里僅為控制器添加代理來做同樣的事情。
// Note: ctrl is the controller instance var proxy = new Proxy(ctrl, { set: function (target, prop, value) { var bind = bindings[prop]; if(bind) { // Update each DOM element bound to the property bind.elements.forEach(function (element) { element.value = value; element.setAttribute("value", value); }); } return Reflect.set(target, prop, value); } });
每當(dāng)設(shè)置綁定屬性時,代理將檢查綁定到該屬性的所有元素,然后用新值更新它們。
在本例中,我們只支持 input 元素綁定,因?yàn)橹辉O(shè)置了 value 屬性。
響應(yīng)事件最后要做的是響應(yīng)用戶交互,DOM 元素在檢測到值更改時觸發(fā)事件。
監(jiān)聽這些事件并使用事件的新值更新綁定屬性,由于代理,綁定到相同屬性的所有其他元素將自動更新。
Object.keys(bindings).forEach(function (boundValue) { var bind = bindings[boundValue]; // Listen elements event and update proxy property bind.elements.forEach(function (element) { element.addEventListener("input", function (event) { proxy[bind.boundValue] = event.target.value; // Also triggers the proxy setter }); }) });React && Virtual DOM
接著將學(xué)習(xí)了解決如何使用單 個HTML 文件運(yùn)行 React,解釋這些概念:functional component,函數(shù)組件, JSX 和 Virtual DOM。
React 提供了用組件構(gòu)建代碼的方法,收下,創(chuàng)建 watch 組 件。
忽略依賴項(xiàng)的 HTML 樣板和腳本,剩下的幾行就是 React 代碼。首先,定義 Watch 組件及其模板,然后掛載React 到 DOM中,來渲染 Watch 組件。
向組件中注入數(shù)據(jù)我們的 Wacth 組件很簡單 ,它只展示我們傳給它的時和分鐘。
你可以嘗試修改這些屬性的值(在 React中稱為 props )。它將最終顯示你傳給它的內(nèi)容,即使它不是數(shù)字。
const Watch = (props) =>{props.hours}:{props.minutes}; ReactDOM.render(, document.getElementById("app"));
props 只是通過周圍組件傳遞給組件的數(shù)據(jù),組件使用 props 進(jìn)行業(yè)務(wù)邏輯和呈現(xiàn)。
但是一旦 props 不屬于組件,它們就是不可變的(immutable)。因此,提供 props 的組件是能夠更新props 值的唯一代碼。
使用 props 非常簡單,使用組件名稱作為標(biāo)記名稱創(chuàng)建 DOM 節(jié)點(diǎn)。 然后給它以 props 名的屬性,接著通過組件中的 this.props 可以獲得傳入的值。
那些不帶引號的 HTML 呢?注意到 render 函數(shù)返回的不帶引號的 HTML, 這個使用是 JSX 語法,它是在 React 組件中定義 HTML 模板的簡寫語法。
// Equivalent to JSX:React.createElement(Watch, {"hours": "9", "minutes": "15"});
現(xiàn)在你可能希望避免使用 JSX 來定義組件的模板,實(shí)際上,JSX 看起來像 語法糖。
以下代碼片段,分別使用 JSX 和 React 語法以構(gòu)建相同結(jié)果。
// Using JS with React.createElement React.createElement("form", null, React.createElement("div", {"className": "form-group"}, React.createElement("label", {"htmlFor": "email"}, "Email address"), React.createElement("input", {"type": "email", "id": "email", "className": "form-control"}), ), React.createElement("button", {"type": "submit", "className": "btn btn-primary"}, "Submit") ) // Using JSX進(jìn)一步探索虛擬 DOM
最后一部分比較復(fù)雜,但是很有趣,這將幫助你了解 React 底層的原理。
更新頁面上的元素 (DOM樹中的節(jié)點(diǎn)) 涉及到使用 DOM API。它將重新繪制頁面,但可能很慢(請參閱本文了解原因)。
許多框架,如 React 和 Vue.js 繞過了這個問題,它們提出了一個名為虛擬 DOM 的解決方案。
{ "type":"div", "props":{ "className":"form-group" }, "children":[ { "type":"label", "props":{ "htmlFor":"email" }, "children":[ "Email address"] }, { "type":"input", "props":{ "type":"email", "id":"email", "className":"form-control"}, "children":[] } ] }
想法很簡單。讀取和更新 DOM 樹非常昂貴。因此,盡可能少地進(jìn)行更改并更新盡可能少的節(jié)點(diǎn)。
減少對 DOM API 的調(diào)用及將 DOM 樹結(jié)構(gòu)保存在內(nèi)存中, 由于討論的是 JavaScript 框架,因此選擇JSON 數(shù)據(jù)結(jié)構(gòu)比較合理。
這種處理方式會立即展示了虛擬 DOM 中的變化。
此外虛擬 DOM 會先緩存一些更新操作,以便稍后在真正 DOM 上渲染,這個樣是為了頻繁操作重新渲染造成一些性能問題。
你還記得 React.createElement 嗎? 實(shí)際上,這個函數(shù)作用是 (直接調(diào)用或通過 JSX 調(diào)用) 在 Virtual DOM 中 創(chuàng)建一個新節(jié)點(diǎn)。
要應(yīng)用更新,Virtual DOM核心功能將發(fā)揮作用,即 協(xié)調(diào)算法,它的工作是提供最優(yōu)的解決方案來解決以前和當(dāng)前虛擬DOM 狀態(tài)之間的差異。
原文:
https://medium.freecodecamp.o...
https://medium.freecodecamp.o...
代碼部署后可能存在的BUG沒法實(shí)時知道,事后為了解決這些BUG,花了大量的時間進(jìn)行l(wèi)og 調(diào)試,這邊順便給大家推薦一個好用的BUG監(jiān)控工具 Fundebug。
你的點(diǎn)贊是我持續(xù)分享好東西的動力,歡迎點(diǎn)贊!
交流干貨系列文章匯總?cè)缦拢X得不錯點(diǎn)個Star,歡迎 加群 互相學(xué)習(xí)。
https://github.com/qq44924588...
我是小智,公眾號「大遷世界」作者,對前端技術(shù)保持學(xué)習(xí)愛好者。我會經(jīng)常分享自己所學(xué)所看的干貨,在進(jìn)階的路上,共勉!
關(guān)注公眾號,后臺回復(fù)福利,即可看到福利,你懂的。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/108948.html
摘要:為了方便大家共同學(xué)習(xí),整理了之前博客系列的文章,目前已整理是如何工作這個系列,可以請猛戳博客查看。以下列出該系列目錄,歡迎點(diǎn)個星星,我將更友動力整理理優(yōu)質(zhì)的文章,一起學(xué)習(xí)。 為了方便大家共同學(xué)習(xí),整理了之前博客系列的文章,目前已整理 JavaScript 是如何工作這個系列,可以請猛戳GitHub博客查看。 以下列出該系列目錄,歡迎點(diǎn)個星星,我將更友動力整理理優(yōu)質(zhì)的文章,一起學(xué)習(xí)。 J...
摘要:本文轉(zhuǎn)載自眾成翻譯譯者鏈接原文今天,我們從一開始就開始。讓我們看看是什么,是什么讓運(yùn)轉(zhuǎn)起來。什么是是一個用于構(gòu)建用戶界面的庫。它是應(yīng)用程序的視圖層。所有應(yīng)用程序的核心是組件。組件是可組合的。虛擬完全存在于內(nèi)存中,并且是網(wǎng)絡(luò)瀏覽器的的表示。 本文轉(zhuǎn)載自:眾成翻譯譯者:iOSDevLog鏈接:http://www.zcfy.cc/article/3765原文:https://www.ful...
摘要:在他的重學(xué)前端課程中提到到現(xiàn)在為止,前端工程師已經(jīng)成為研發(fā)體系中的重要崗位之一。大部分前端工程師的知識,其實(shí)都是來自于實(shí)踐和工作中零散的學(xué)習(xí)。一基礎(chǔ)前端工程師吃飯的家伙,深度廣度一樣都不能差。 開篇 前端開發(fā)是一個非常特殊的行業(yè),它的歷史實(shí)際上不是很長,但是知識之繁雜,技術(shù)迭代速度之快是其他技術(shù)所不能比擬的。 winter在他的《重學(xué)前端》課程中提到: 到現(xiàn)在為止,前端工程師已經(jīng)成為研...
摘要:在他的重學(xué)前端課程中提到到現(xiàn)在為止,前端工程師已經(jīng)成為研發(fā)體系中的重要崗位之一。大部分前端工程師的知識,其實(shí)都是來自于實(shí)踐和工作中零散的學(xué)習(xí)。一基礎(chǔ)前端工程師吃飯的家伙,深度廣度一樣都不能差。開篇 前端開發(fā)是一個非常特殊的行業(yè),它的歷史實(shí)際上不是很長,但是知識之繁雜,技術(shù)迭代速度之快是其他技術(shù)所不能比擬的。 winter在他的《重學(xué)前端》課程中提到: 到現(xiàn)在為止,前端工程師已經(jīng)成為研發(fā)體系...
摘要:是流行的框架之一,在年及以后將會更加流行。于年首次發(fā)布,多年來廣受歡迎。下面是另一個名為的高階函數(shù)示例,該函數(shù)接受另外兩個函數(shù),分別是和。將所有較小的函數(shù)組合成更大的函數(shù),最終,得到一個應(yīng)用程序,這稱為組合。 React是流行的javascript框架之一,在2019年及以后將會更加流行。React于2013年首次發(fā)布,多年來廣受歡迎。它是一個聲明性的、基于組件的、用于構(gòu)建用戶界面的高...
閱讀 1845·2021-09-14 18:03
閱讀 2277·2019-08-30 15:48
閱讀 1133·2019-08-30 14:09
閱讀 518·2019-08-30 12:55
閱讀 2739·2019-08-29 11:29
閱讀 1498·2019-08-26 13:43
閱讀 2322·2019-08-26 13:30
閱讀 2380·2019-08-26 12:17