成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

來自 React 19 的背刺:ForwardRef 被無情拋棄

開發(fā) 前端
在面向?qū)ο缶幊讨校琁OC (Inversion of Control) 控制反轉(zhuǎn)是一個(gè)非常高級(jí)的概念。它是一種設(shè)計(jì)理念,在減少對(duì)象之間的耦合關(guān)系上有非常重要的作用。

在之前的版本中,forwardRef 一直是我最愛用的 ref 之一。它在封裝組件時(shí)非常有用。可是萬萬沒想到,由于使用方式稍微麻煩了一點(diǎn),在新的版本中,直接被 React 19 背刺一刀,實(shí)現(xiàn)同樣的功能,以后可以不用它了.... forwardRef 被無情拋棄。

本文主要內(nèi)容包括如下幾個(gè)部分。

  • React 中的 控制反轉(zhuǎn) IOC。
  • forwardRef 基礎(chǔ)知識(shí)。
  • React 19 中 ref 機(jī)制更改,forwardRef 被無情拋棄。
  • useImperativeHandle 與 ref 的新配合。

一、React 中的控制反轉(zhuǎn)

在面向?qū)ο缶幊讨校琁OC (Inversion of Control) 控制反轉(zhuǎn)是一個(gè)非常高級(jí)的概念。它是一種設(shè)計(jì)理念,在減少對(duì)象之間的耦合關(guān)系上有非常重要的作用。

?

許多前端雖然對(duì)其有所耳聞,但說實(shí)話,能理解的并不多。甚至很對(duì)前端對(duì)解耦這個(gè)概念都是一頭霧水。

IOC 的設(shè)計(jì)理念里,有三個(gè)角色,一個(gè)角色是容器 C,一個(gè)角色是被控制者 B,一個(gè)角色是控制者 A,許多時(shí)候,在代碼開發(fā)中,控制者 A 直接去控制對(duì)象 B,會(huì)導(dǎo)致 B 被多次實(shí)例化而從讓代碼邏輯變得更加復(fù)雜。因此 IOC 的理念是讓 控制者 A 失去對(duì) B 的直接控制權(quán),它只能與容器交互。從而將 A 與 B 的直接聯(lián)系隔離開。

這樣說可能會(huì)有點(diǎn)繞,但是呢,我們使用一個(gè)大家經(jīng)常使用的代碼來說明一下,你一下就能明白。

const ref = useRef()
<InputNumber ref={ref} />
ref.current.focus()

在這個(gè)案例里,我們的目標(biāo)是直接獲取到 input 對(duì)象,并且調(diào)用它的 focus 方法,讓 input 獲得焦點(diǎn)。

var input = document.getElementById('input')
input.focus()

但是在模塊的劃分上,input 元素本身并不屬于當(dāng)前模塊/組件,因此,調(diào)用 input 方法的行為,其實(shí)是屬于一種混亂。除非我們不做解耦和封裝。

因此,在 React 的組件封裝中,并不支持直接獲取到 input 的引用,而是以一種傳入控制器的方式來調(diào)用它。在這個(gè)場(chǎng)景里:

input 對(duì)象本身是被調(diào)用者
InputNumber 組件是容器
ref 是控制器

當(dāng)前組件利用 ref 來調(diào)用 input。從而讓代碼的解耦變得非常合理。可擴(kuò)展性也很強(qiáng)。

?

注意一些概念上的區(qū)分:控制反轉(zhuǎn)是一種設(shè)計(jì)思維,依賴注入是控制反轉(zhuǎn)的一種具體實(shí)現(xiàn),在 React 中,ref 也是一種控制反轉(zhuǎn)的具體實(shí)現(xiàn)。

所以不要聽著別人吹控制反轉(zhuǎn)就覺得牛,你可能也天天在用。

二、forwardRef 基礎(chǔ)知識(shí)

forwardRef 能夠幫助 React 組件傳遞 ref。或者說是內(nèi)部對(duì)象控制權(quán)的轉(zhuǎn)移與轉(zhuǎn)發(fā)。它接收一個(gè)組件作為參數(shù)。

import { forwardRef } from 'react';

function MyInput(props, ref) {
  // ...
}

const InputNumber = forwardRef(MyInput);

這里需要注意的是,我們需要把 ref 放在自定義組件的參數(shù)中。

function MyInput(props, ref) {
  // ...
}

forwardRef 返回一個(gè)可渲染的函數(shù)組件。因此,我們可以通過一個(gè)簡(jiǎn)單的案例完善上面的代碼。

function MyInput(props, ref) {
  return (
    <label>
      {props.label}
      <input ref={ref} />
    </label>
  );
}

const InputNumber = forwardRef(MyInput)

所以呢,在 React 的開發(fā)中,forwardRef 能夠幫助我們實(shí)現(xiàn)更良好的解耦,這也是我一直非常喜歡使用 forwardRef 的原因。

三、ref 機(jī)制更改,forwardRef 被無情拋棄

但是,在 React 19 中,forwardRef 被直接背刺,由于 ref 傳遞機(jī)制的更改,我們可以不用 forwardRef 也能做到同樣的事情了。

首先,在聲明組件時(shí),ref 不再獨(dú)立成為一個(gè)參數(shù),而是作為 props 屬性中的一個(gè)屬性。

function MyInput(props) {
  return (
    <label>
      {props.label}
      <input ref={props.ref} />
    </label>
  );
}

其次,代碼這樣寫了之后,就可以直接在父組件中,通過 ref 拿到 input 的控制權(quán)。

function Index() {
  const input = useRef(null)

  function __clickHandler() {
    input.current.focus()
  }

  return (
    <div>
      <button onClick={__clickHandler}>
        點(diǎn)擊獲取焦點(diǎn)
      </button>
      <MyInput ref={input} />
    </div>
  )
}

在父組件中的使用與以前一樣,但是子組件由于不再需要 forwardRef,變得更加簡(jiǎn)單了。

四、useImperativeHandle 與 ref 的新配合

除了直接拿到元素對(duì)象本身就已經(jīng)存在的 ref,我們還可以通過 useImperativeHandle 來自定義 ref 控制器能執(zhí)行哪些方法

useImperativeHandle 接收三個(gè)參數(shù),分別是:

  • ref: 組件聲明時(shí)傳入的 ref。
  • createHandle: 回調(diào)函數(shù),需要返回 ref 引用的對(duì)象,我們也是在這里重寫 ref 引用。
  • deps: 依賴項(xiàng)數(shù)組,可選。state,props 以及內(nèi)部定義的其他變量都可以作為依賴項(xiàng),React 內(nèi)部會(huì)使用 Object.is 來對(duì)比依賴項(xiàng)是否發(fā)生了變化。依賴項(xiàng)發(fā)生變化時(shí),createHandle 會(huì)重新執(zhí)行,ref 引用會(huì)更新。如果不傳入依賴項(xiàng),那么每次更新 createHandle 都會(huì)重新執(zhí)行。

在官方文檔中,有這樣一個(gè)案例,演示效果如圖所示,當(dāng)我點(diǎn)擊按鈕時(shí),下方的 input 自動(dòng)獲取焦點(diǎn),并且中間的滾動(dòng)條滾動(dòng)到最底部。

我們結(jié)合新的 ref 傳遞機(jī)制和 useImperativeHandle 一起來分析一下這個(gè)案例應(yīng)該怎么實(shí)現(xiàn)。

!

思考時(shí),請(qǐng)一定要把封裝的思維帶入進(jìn)來,否則可能很難感受到這樣做在解耦上的具體好處

首先我們先進(jìn)行組件拆分,將整個(gè)內(nèi)容拆分為按鈕部分與信息部分,信息部分主要負(fù)責(zé)信息的展示與輸入,因此頁面組件大概長(zhǎng)這樣

<>
  <button>Write a comment</button>
  <Post />
</>

我們期望點(diǎn)擊按鈕時(shí),信息部分的輸入框自動(dòng)獲取焦點(diǎn),信息部分的信息展示區(qū)域能滾動(dòng)到最底部,因此整個(gè)頁面組件的代碼可以表示為如下:

import { useRef } from 'react';
import Post from './Post.js';

export default function Page() {
  const postRef = useRef(null);

  function handleClick() {
    postRef.current.scrollAndFocusAddComment();
  }

  return (
    <>
      <button onClick={handleClick}>
        Write a comment
      </button>
      <Post ref={postRef} />
    </>
  );
}

再來思考信息部分。

信息部分 Post 又分為兩個(gè)部分,分別是信息展示部分與信息輸入部分。此時(shí)這兩個(gè)部分的 ref 要透?jìng)鹘o Post,并最終再次透?jìng)鹘o頁面組件。因此他們的組件結(jié)構(gòu)應(yīng)該長(zhǎng)這樣。

<>
  <article>
    <p>Welcome to my blog!</p>
  </article>
  <CommentList ref={commentsRef} />
  <AddComment ref={addCommentRef} />
</>

這個(gè)時(shí)候,有一個(gè)需要考慮的點(diǎn)就是,有兩個(gè)對(duì)象需要被控制,因此我們需要借助 useImperativeHandle 來自定義控制器,并在控制的方法中,整合他們。

useImperativeHandle(ref, () => {
  return {
    scrollAndFocusAddComment() {
      commentsRef.current.scrollToBottom();
      addCommentRef.current.focus();
    }
  };
}, []);

ref 的傳遞,只需要把它看成是一個(gè)普通的 props 即可,因此,Post 組件完整代碼如下:

const Post = ({ref}) => {
  const commentsRef = useRef(null);
  const addCommentRef = useRef(null);

  useImperativeHandle(ref, () => {
    return {
      scrollAndFocusAddComment() {
        commentsRef.current.scrollToBottom();
        addCommentRef.current.focus();
      }
    };
  }, []);

  return (
    <>
      <article>
        <p>Welcome to my blog!</p>
      </article>
      <CommentList ref={commentsRef} />
      <AddComment ref={addCommentRef} />
    </>
  );
}

同樣的道理,我們只需要把 CommentList 與 AddComment 的 ref 傳遞出來就可以了。

所以信息展示部分 CommentList 組件的代碼為。

import { useRef, useImperativeHandle } from 'react';

const CommentList = ({ref}) => {
  const divRef = useRef(null);

  useImperativeHandle(ref, () => {
    return {
      scrollToBottom() {
        const node = divRef.current;
        node.scrollTop = node.scrollHeight;
      }
    };
  }, []);

  let comments = [];
  for (let i = 0; i < 50; i++) {
    comments.push(<p key={i}>Comment #{i}</p>);
  }

  return (
    <div className="CommentList" ref={divRef}>
      {comments}
    </div>
  );
};

export default CommentList;

信息輸入部分 AddComment 的代碼為。

function AddComment(props) {
  return (
    <input 
      placeholder="Add comment..." 
      ref={props.ref} 
    />
  )
};

export default AddComment;

與之前相比,確實(shí)要簡(jiǎn)單了許多。

五、總結(jié)

如果你對(duì)封裝解耦比較重視,那么你一定能明顯感受到,ref 與 useImperativeHandle 的結(jié)合能發(fā)揮的想象空間遠(yuǎn)不止于此,這種方式給 React 提供了一種擴(kuò)展 React 能力的重要手段,因此,當(dāng)你成為 React 高手之后,你一定會(huì)非常喜歡使用它們,他們的組合能讓 React 項(xiàng)目變得更加多樣化。

責(zé)任編輯:姜華 來源: 這波能反殺
相關(guān)推薦

2019-11-13 10:35:27

React Conf前端總結(jié)

2018-04-27 13:20:12

Java 8Java 9Java 10

2021-02-18 23:31:42

Starlink通信擴(kuò)張

2023-03-28 15:36:43

智能汽車汽車智能化

2014-05-21 09:47:23

2024-09-06 17:26:49

2025-04-09 10:41:43

2009-09-22 08:56:54

MySQLOracle

2020-04-24 09:14:21

術(shù)語網(wǎng)絡(luò)安全黑客

2023-09-14 19:15:21

2010-10-29 11:01:11

簡(jiǎn)歷

2015-10-10 16:02:36

React NativAndroid

2011-09-08 14:03:36

2023-06-25 07:45:52

2022-07-11 08:20:49

DDoS攻擊網(wǎng)絡(luò)攻擊

2022-06-15 14:48:39

谷歌TensorFlowMeta

2014-01-09 10:19:43

大數(shù)據(jù)

2021-02-26 07:46:09

WebDAV文件管理

2022-05-05 08:25:22

模型OpenAI代碼

2023-06-13 14:05:56

ChatGPT數(shù)據(jù)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 亚洲成人毛片 | 亚洲色在线视频 | a看片| 国产精品 亚洲一区 | 精品99爱视频在线观看 | 激情欧美日韩一区二区 | 97超碰在线免费 | 欧美日韩国产高清 | 91久久精品国产91久久 | 8x国产精品视频一区二区 | 欧美日韩精品免费观看 | 不卡在线视频 | 天天干夜夜操视频 | 极品电影院 | 成人亚洲片 | 91精品久久久久久久久久 | 男人的天堂在线视频 | 精品国产乱码久久久久久牛牛 | 国产成人久久av免费高清密臂 | 日本三级网站在线观看 | 91九色在线观看 | 色综合av | 国产目拍亚洲精品99久久精品 | 久久免费观看一级毛片 | 91超碰caoporn97人人 | 精品国产伦一区二区三区观看体验 | 99久久久国产精品 | 久久精品小视频 | 四虎影视在线 | 草久久| 久久免费电影 | 日韩av免费在线电影 | 精品伊人 | 亚洲精品乱码 | 91国在线高清视频 | 黄色在线| av片免费| 中文字幕韩在线第一页 | 先锋资源在线 | 欧美 日韩 视频 | 久久亚洲春色中文字幕久久久 |