React獲取DOM系列

在前端開發(fā)過程中,必然有些情況會(huì)需要dom元素進(jìn)行一些節(jié)點(diǎn)操作,拋開原生的獲取方法竹捉,本文將介紹React提供的解決方案态贤,本文所有示例運(yùn)行版本為react@16.11.0

1. ref

React16.3版本之后React.createRef() API付呕,使用方式如下

class Home extends Component {

    constructor(props){
        super(props)
        this.state = {

        }
        // ref回調(diào)方式獲取
        this.pRef = null
        this.setCallBackRef = element => {
            this.pRef = element
        }
        this.myNode = React.createRef()
        this.myPara = React.createRef()
    }

    componentDidMount(){
        console.log(this.myNode.current)
        console.log(this.myPara.current)
        // 無current
        console.log(this.pRef)
    }

    render() {
        return <div>
            <Para ref={this.myPara} />
            <p ref={this.setCallBackRef}>123</p>
            <p ref={this.myNode}>{this.props.number}</p>
        </div>
    }
}

當(dāng) ref 被傳遞給 render 中的元素時(shí),對(duì)該節(jié)點(diǎn)的引用可以在 ref 的 current 屬性中被訪問廓八。
ref 的值根據(jù)節(jié)點(diǎn)的類型而有所不同:

  • 當(dāng) ref 屬性用于 HTML 元素時(shí)锅很,構(gòu)造函數(shù)中使用 React.createRef() 創(chuàng)建的 ref 接收底層 DOM 元素作為其 current 屬性。即:this.myNode.current獲取到的元素為dom
  • 當(dāng) ref 屬性用于自定義 class組件時(shí)多望,ref 對(duì)象接收組件的掛載實(shí)例作為其 current 屬性嫩舟。即:this.myPara.current獲取到的是組件的實(shí)例
  • 函數(shù)組件不能使用ref屬性,因?yàn)樗麄儧]有實(shí)例

備注:ref在componentDidMountcomponentDidUpdate 生命周期鉤子觸發(fā)前更新怀偷。

上述代碼運(yùn)行結(jié)果如下
ref-運(yùn)行結(jié)果.jpg

可以看出家厌,我們?cè)趻煸诘阶远x組件時(shí),獲取到了這個(gè)組件的實(shí)例枢纠,可以獲取到它的props像街,context以及自定義函數(shù),那么有意思的事情就來了晋渺,我們這時(shí)可以再父組件中調(diào)用子組件的函數(shù)來完成某些業(yè)務(wù)場(chǎng)景

class Para extends React.Component {

    close = () => {
        alert(1)
    }

    render(){
        return <div>
            <p>1</p>
        </div>
    }
}
class Home extends Component {

    constructor(props){
        super(props)
        this.myPara = React.createRef()
    }

    render() {
        return <div>
            <Para ref={this.myPara} />
            <button onClick={()=>this.myPara.current.close()}>close</button>
        </div>
    }
}

運(yùn)行結(jié)果如下
父組件調(diào)用子組件實(shí)例函數(shù).jpg

這個(gè)操作可以實(shí)現(xiàn)有趣的功能镰绎,可以好好玩一下,特別是在業(yè)務(wù)組件里木西。

2. findDOMNode()

reactDom提供了一個(gè)api畴栖,可以讓我們獲取dom節(jié)點(diǎn),不過大多數(shù)情況下不推薦使用八千,且在嚴(yán)格模式下已經(jīng)被棄用吗讶。
棄用原因:findDOMNode 只返回第一個(gè)子節(jié)點(diǎn),但是使用 Fragments恋捆,組件可以渲染多個(gè) DOM 節(jié)點(diǎn)照皆。findDOMNode 是一個(gè)只讀一次的 API。調(diào)用該方法只會(huì)返回第一次查詢的結(jié)果沸停。如果子組件渲染了不同的節(jié)點(diǎn)膜毁,則無法跟蹤此更改。因此愤钾,findDOMNode 僅在組件返回單個(gè)且不可變的 DOM 節(jié)點(diǎn)時(shí)才有效瘟滨。

class Home extends Component {

    constructor(props){
        super(props)
        this.state = {

        }
        // ref回調(diào)方式獲取
        this.pRef = null
        this.setCallBackRef = element => {
            this.pRef = element
        }
        this.myNode = React.createRef()
        this.myPara = React.createRef()
    }

    // 獲取實(shí)例化組件dom
    getComponent = () => {
        const node = findDOMNode(this.myPara.current)
        console.log(node)
    }

    componentDidMount(){
        console.log(this.myNode.current)
        console.log(this.myPara.current)
        // 無current
        console.log(this.pRef)
    }

    render() {
        return <div>
            <Para ref={this.myPara} />
            <p ref={this.setCallBackRef}>123</p>
            <p ref={this.myNode}>{this.props.number}</p>
            <button onClick={()=>this.myPara.current.close()}>close</button>
            <button onClick={this.getComponent}>getNode</button>
        </div>
    }
}

ReactDOM下的幾個(gè)api都很有趣,回頭分析一下能颁,埋個(gè)坑杂瘸。

3.useRef

react@16.8版本之后,使用hook伙菊,hook提供了useRef來獲取dom败玉,注意:返回的 ref 對(duì)象在組件的整個(gè)生命周期內(nèi)保持不變(current是變化的)敌土,相對(duì)React.createRef()在生命周期過程中是一直變化的。代碼如下

import React, { useRef, useEffect } from 'react';
import Para from './Para'

const UseRef = () => {
    const myNode = useRef(null);
    const myPara = useRef(null);

    useEffect(() => {
        console.log(myNode.current)
        console.log(myPara.current)
    }, []);

    return <div>
        <p ref={myNode}>123</p>
        <Para ref={myPara} />
    </div>
}

export default UseRef;
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末绒怨,一起剝皮案震驚了整個(gè)濱河市纯赎,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌南蹂,老刑警劉巖犬金,帶你破解...
    沈念sama閱讀 207,248評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異六剥,居然都是意外死亡晚顷,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門疗疟,熙熙樓的掌柜王于貴愁眉苦臉地迎上來该默,“玉大人,你說我怎么就攤上這事策彤∷ㄐ洌” “怎么了?”我有些...
    開封第一講書人閱讀 153,443評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵店诗,是天一觀的道長(zhǎng)裹刮。 經(jīng)常有香客問我,道長(zhǎng)庞瘸,這世上最難降的妖魔是什么捧弃? 我笑而不...
    開封第一講書人閱讀 55,475評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮擦囊,結(jié)果婚禮上违霞,老公的妹妹穿的比我還像新娘。我一直安慰自己瞬场,他們只是感情好买鸽,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,458評(píng)論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著贯被,像睡著了一般眼五。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上刃榨,一...
    開封第一講書人閱讀 49,185評(píng)論 1 284
  • 那天,我揣著相機(jī)與錄音双仍,去河邊找鬼枢希。 笑死,一個(gè)胖子當(dāng)著我的面吹牛朱沃,可吹牛的內(nèi)容都是我干的苞轿。 我是一名探鬼主播茅诱,決...
    沈念sama閱讀 38,451評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼搬卒!你這毒婦竟也來了瑟俭?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,112評(píng)論 0 261
  • 序言:老撾萬榮一對(duì)情侶失蹤契邀,失蹤者是張志新(化名)和其女友劉穎摆寄,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體坯门,經(jīng)...
    沈念sama閱讀 43,609評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡微饥,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,083評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了古戴。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片欠橘。...
    茶點(diǎn)故事閱讀 38,163評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖现恼,靈堂內(nèi)的尸體忽然破棺而出肃续,到底是詐尸還是另有隱情,我是刑警寧澤叉袍,帶...
    沈念sama閱讀 33,803評(píng)論 4 323
  • 正文 年R本政府宣布始锚,位于F島的核電站,受9級(jí)特大地震影響畦韭,放射性物質(zhì)發(fā)生泄漏疼蛾。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,357評(píng)論 3 307
  • 文/蒙蒙 一艺配、第九天 我趴在偏房一處隱蔽的房頂上張望察郁。 院中可真熱鬧,春花似錦转唉、人聲如沸皮钠。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽麦轰。三九已至,卻和暖如春砖织,著一層夾襖步出監(jiān)牢的瞬間款侵,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評(píng)論 1 261
  • 我被黑心中介騙來泰國(guó)打工侧纯, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留新锈,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,636評(píng)論 2 355
  • 正文 我出身青樓眶熬,卻偏偏與公主長(zhǎng)得像妹笆,于是被迫代替她去往敵國(guó)和親块请。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,925評(píng)論 2 344