您好,登錄后才能下訂單哦!
本篇文章給大家分享的是有關(guān)React中怎么實現(xiàn)受控組件與非受控組件,小編覺得挺實用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。
受控組件與非受控組件是React處理表單的入口。從React的思路來講,作者肯定讓數(shù)據(jù)控制一切,或者簡單的理解為,頁面的生成與更新得忠實地執(zhí)行JSX的指令。
但是表單元素有其特殊之處,用戶可以通過鍵盤輸入與鼠標(biāo)選擇,改變界面的顯示。界面的改變也意味著有一些數(shù)據(jù)被改動,比較明顯的是input的 value ,textarea的 innerHTML ,radio/checkbox的 checked ,不太明顯的是option的 selected 與 selectedIndex ,這兩個是被動修改的。
<input value="{this.state.value}"/>
當(dāng)input.value是由組件的state.value拍出來的,當(dāng)用戶進(jìn)行輸入修改后,然后JSX再次重刷視圖,這時input.value是采取用戶的新值還是state的新值?基于這個分歧,React給出一個折衷的方案,兩者都支持,于是就產(chǎn)生了今天的主題了。
React認(rèn)為value/checked不能單獨存在,需要與onInput/onChange/disabed/readOnly等控制value/checked的屬性或事件一起使用。 它們共同構(gòu)成 受控組件 ,受控是受JSX的控制。如果用戶沒有寫這些額外的屬性與事件,那么框架內(nèi)部會給它添加一些事件,如onClick, onInput, onChange,阻止你進(jìn)行輸入或選擇,讓你無法修改它的值。在框架內(nèi)部,有一個頑固的變量,我稱之為 persistValue,它一直保持JSX上次賦給它的值,只能讓內(nèi)部事件修改它。
因此我們可以斷言,受控組件是可通過 事件 完成的對value的控制。
在受控組件中,persistValue總能被刷新。
我們再看非受控組件,既然value/checked已經(jīng)被占用了,React啟用了HTML中另一組被忽略的屬性defaultValue/defaultChecked。一般認(rèn)為它們是與value/checked相通的,即,value不存在的情況下,defaultValue的值就當(dāng)作是value。
上面我們已經(jīng)說過,表單元素的顯示情況是由內(nèi)部的 persistValue 控制的,因此defaultXXX也會同步persistValue,然后再由persistValue同步DOM。但非受控組件的出發(fā)點是忠實于用戶操作,如果用戶在代碼中
input.value = "xxxx"
以后
<input defaultvalue="{this.state.value}"/>
就再不生效,一直是xxxx。
它怎么做到這一點,怎么辨識這個修改是來自框架內(nèi)部或外部呢?我翻看了一下React的源碼,原來它有一個叫valueTracker的東西跟蹤用戶的輸入
var tracker = { getValue: function () { return currentValue; }, setValue: function (value) { currentValue = '' + value; }, stopTracking: function () { detachTracker(node); delete node[valueField]; } }; return tracker; }
這個東西又是通過Object.defineProperty打進(jìn)元素的value/checked的內(nèi)部,因此就知曉用戶對它的取值賦值操作。
但value/checked還是兩個很核心的屬性,涉及到太多內(nèi)部機(jī)制(比如說value與oninput, onchange, 輸入法事件oncompositionstart,
compositionchange, oncompositionend, onpaste, oncut),為了平緩地修改value/checked,
還要用到 Object.getOwnPropertyDescriptor 。如果我要兼容IE8,沒有這么高級的玩藝兒。我采取另一種更安全的方式,
只用Object.defineProperty修改 defaultValue/defaultChecked 。
首先我為元素添加一個 _uncontrolled 的屬性,用來表示我已經(jīng)劫持過defaultXXX。 然后描述對象 ( Object.defineProperty的第三個參數(shù) )的set方法里面再添加一個開關(guān), _observing 。在框架內(nèi)部更新視圖,此值為false,更新完,它置為true。
這樣就知曉 input.defaultValue = “xxx”時,這是由用戶還是框架修改的。
if (!dom._uncontrolled) { dom._uncontrolled = true; inputMonitor.observe(dom, name); //重寫defaultXXX的setter/getter } dom._observing = false;//此時是框架在修改視圖,因此需要關(guān)閉開關(guān) dom[name] = val; dom._observing = true;//打開開關(guān),來監(jiān)聽用戶的修改行為
inputMonitor的實現(xiàn)如下
export var inputMonitor = {}; var rcheck = /checked|radio/; var describe = { set: function(value) { var controllProp = rcheck.test(this.type) ? "checked" : "value"; if (this.type === "textarea") { this.innerHTML = value; } if (!this._observing) { if (!this._setValue) { //defaultXXX只會同步一次_persistValue var parsedValue = (this[controllProp] = value); this._persistValue = Array.isArray(value) ? value : parsedValue; this._setValue = true; } } else { //如果用戶私下改變defaultValue,那么_setValue會被抺掉 this._setValue = value == null ? false : true; } this._defaultValue = value; }, get: function() { return this._defaultValue; }, configurable: true }; inputMonitor.observe = function(dom, name) { try { if ("_persistValue" in dom) { dom._setValue = true; } Object.defineProperty(dom, name, describe); } catch (e) {} };
又不小心貼了這么燒腦的代碼,這是碼農(nóng)的壞毛病。不過,到這步,大家都明白,無論是官方react還是anu/qreact都是通過Object.defineProperty來控制用戶的輸入的。
于是我們可以理解以下的代碼的行為了
var a = ReactDOM.render(<textarea defaultValue="foo" />, container); ReactDOM.render(<textarea defaultValue="bar" />, container); ReactDOM.render(<textarea defaultValue="noise" />, container); expect(a.defaultValue).toBe("noise"); expect(a.value).toBe("foo"); expect(a.textContent).toBe("noise"); expect(a.innerHTML).toBe("noise");
由于用戶一直沒有手動修改 defaultValue, dom._setValue 一直為 false/undefined ,因此 _persistValue 一直能修改。
另一個例子:
var renderTextarea = function(component, container) { if (!container) { container = document.createElement("div"); } const node = ReactDOM.render(component, container); node.defaultValue = node.innerHTML.replace(/^\n/, ""); return node; }; const container = document.createElement("div"); //注意這個方法,用戶在renderTextarea中手動改變了defaultValue,_setValue就變成true const node = renderTextarea(<textarea defaultValue="giraffe" />, container); expect(node.value).toBe("giraffe"); // _setValue后,gorilla就不能同步到_persistValue,因此還是giraffe renderTextarea(<textarea defaultValue="gorilla" />, container); // expect(node.value).toEqual("giraffe"); node.value = "cat"; // 這個又是什么回事了呢,因此非監(jiān)控屬性是在diffProps中批量處理的,在監(jiān)控屬性,則是在更后的方法中處理 // 檢測到node.value !== _persistValue,于是重寫 _persistValue = node.value,于是輸出cat renderTextarea(<textarea defaultValue="monkey" />, container); expect(node.value).toEqual("cat");
純文本類:text, textarea, JSX的值,總是往字符串轉(zhuǎn)換
type=”number”的控制,值總是為數(shù)字,不填或為“”則轉(zhuǎn)換為“0”
radio有聯(lián)動效果,同一父節(jié)點下的相同name的radio控制只能選擇一個。
select的value/defaultValue支持?jǐn)?shù)組,不做轉(zhuǎn)換,但用戶對底下的option元素做增刪操作,selected會跟著變動。
此外select還有模糊匹配與精確匹配之分。
//精確匹配 var dom = ReactDOM.render( <select value={222}> <option value={111}>aaa</option> <option value={"222"}>xxx</option> <option value={222}>bbb</option> <option value={333}>ccc</option> </select>, container ); expect(dom.options[2].selected).toBe(true);//選中第三個
//模糊匹配 var dom = ReactDOM.render( <select value={222}> <option value={111}>aaa</option> <option value={"222"}>xxx</option> <option value={333}>ccc</option> </select>, container ); expect(dom.options[2].selected).toBe(true);//選中第二個
以上就是React中怎么實現(xiàn)受控組件與非受控組件,小編相信有部分知識點可能是我們?nèi)粘9ぷ鲿姷交蛴玫降?。希望你能通過這篇文章學(xué)到更多知識。更多詳情敬請關(guān)注億速云行業(yè)資訊頻道。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。