成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

React Hooks 解析(下):進(jìn)階

APICloud / 690人閱讀

摘要:第一次了解這項特性的時候,真的有一種豁然開朗,發(fā)現(xiàn)新大陸的感覺。在絕大多數(shù)情況下,是更好的選擇。唯一例外的就是需要根據(jù)新的來進(jìn)行操作的場景。會保證在頁面渲染前執(zhí)行,也就是說頁面渲染出來的是最終的效果。上面條規(guī)則都是為了保證調(diào)用順序的穩(wěn)定性。

歡迎關(guān)注我的公眾號睿Talk,獲取我最新的文章:

一、前言

React Hooks 是從 v16.8 引入的又一開創(chuàng)性的新特性。第一次了解這項特性的時候,真的有一種豁然開朗,發(fā)現(xiàn)新大陸的感覺。我深深的為 React 團(tuán)隊天馬行空的創(chuàng)造力和精益求精的鉆研精神所折服。本文除了介紹具體的用法外,還會分析背后的邏輯和使用時候的注意事項,力求做到知其然也知其所以然。

這個系列分上下兩篇,這里是上篇的傳送門:
React Hooks 解析(上):基礎(chǔ)

二、useLayoutEffect

useLayoutEffect的用法跟useEffect的用法是完全一樣的,都可以執(zhí)行副作用和清理操作。它們之間唯一的區(qū)別就是執(zhí)行的時機(jī)。

useEffect不會阻塞瀏覽器的繪制任務(wù),它在頁面更新后才會執(zhí)行。

useLayoutEffectcomponentDidMountcomponentDidUpdate的執(zhí)行時機(jī)一樣,會阻塞頁面的渲染。如果在里面執(zhí)行耗時任務(wù)的話,頁面就會卡頓。

在絕大多數(shù)情況下,useEffectHook 是更好的選擇。唯一例外的就是需要根據(jù)新的 UI 來進(jìn)行 DOM 操作的場景。useLayoutEffect會保證在頁面渲染前執(zhí)行,也就是說頁面渲染出來的是最終的效果。如果使用useEffect,頁面很可能因為渲染了 2 次而出現(xiàn)抖動。

三、useContext

useContext可以很方便的去訂閱 context 的改變,并在合適的時候重新渲染組件。我們先來熟悉下標(biāo)準(zhǔn)的 context API 用法:

const ThemeContext = React.createContext("light");

class App extends React.Component {
  render() {
    return (
      
        
      
    );
  }
}

// 中間層組件
function Toolbar(props) {
  return (
    
); } class ThemedButton extends React.Component { // 通過定義靜態(tài)屬性 contextType 來訂閱 static contextType = ThemeContext; render() { return

除了定義靜態(tài)屬性的方式,還有另外一種針對Function Component的訂閱方式:

function ThemedButton() {
    // 通過定義 Consumer 來訂閱
    return (
        
          {value => 

使用useContext來訂閱,代碼會是這個樣子,沒有額外的層級和奇怪的模式:

function ThemedButton() {
  const value = useContext(NumberContext);
  return 

在需要訂閱多個 context 的時候,就更能體現(xiàn)出useContext的優(yōu)勢。傳統(tǒng)的實現(xiàn)方式:

function HeaderBar() {
  return (
    
      {user =>
        
          {notifications =>
            
Welcome back, {user.name}! You have {notifications.length} notifications.
} }
); }

useContext的實現(xiàn)方式更加簡潔直觀:

function HeaderBar() {
  const user = useContext(CurrentUser);
  const notifications = useContext(Notifications);

  return (
    
Welcome back, {user.name}! You have {notifications.length} notifications.
); }
四、useReducer

useReducer的用法跟 Redux 非常相似,當(dāng) state 的計算邏輯比較復(fù)雜又或者需要根據(jù)以前的值來計算時,使用這個 Hook 比useState會更好。下面是一個例子:

function init(initialCount) {
  return {count: initialCount};
}

function reducer(state, action) {
  switch (action.type) {
    case "increment":
      return {count: state.count + 1};
    case "decrement":
      return {count: state.count - 1};
    case "reset":
      return init(action.payload);
    default:
      throw new Error();
  }
}

function Counter({initialCount}) {
  const [state, dispatch] = useReducer(reducer, initialCount, init);
  return (
    <>
      Count: {state.count}
      
      
      
    
  );
}

結(jié)合 context API,我們可以模擬 Redux 的操作了,這對組件層級很深的場景特別有用,不需要一層一層的把 state 和 callback 往下傳:

const TodosDispatch = React.createContext(null);
const TodosState = React.createContext(null);

function TodosApp() {
  const [todos, dispatch] = useReducer(todosReducer);

  return (
    
      
        
      
    
  );
}

function DeepChild(props) {
  const dispatch = useContext(TodosDispatch);
  const todos = useContext(TodosState);

  function handleClick() {
    dispatch({ type: "add", text: "hello" });
  }

  return (
    <>
      {todos}
      
    
  );
}
五、useCallback / useMemo / React.memo

useCallbackuseMemo設(shè)計的初衷是用來做性能優(yōu)化的。在Class Component中考慮以下的場景:

class Foo extends Component {
  handleClick() {
    console.log("Click happened");
  }
  render() {
    return ;
  }
}

傳給 Button 的 onClick 方法每次都是重新創(chuàng)建的,這會導(dǎo)致每次 Foo render 的時候,Button 也跟著 render。優(yōu)化方法有 2 種,箭頭函數(shù)和 bind。下面以 bind 為例子:

class Foo extends Component {
  constructor(props) {
    super(props);
    this.handleClick = this.handleClick.bind(this);
  }
  handleClick() {
    console.log("Click happened");
  }
  render() {
    return ;
  }
}

同樣的,Function Component也有這個問題:

function Foo() {
  const [count, setCount] = useState(0);

  const handleClick() {
    console.log(`Click happened with dependency: ${count}`)
  }
  return ;
}

而 React 給出的方案是useCallback Hook。在依賴不變的情況下 (在我們的例子中是 count ),它會返回相同的引用,避免子組件進(jìn)行無意義的重復(fù)渲染:

function Foo() {
  const [count, setCount] = useState(0);

  const memoizedHandleClick = useCallback(
    () => console.log(`Click happened with dependency: ${count}`), [count],
  ); 
  return ;
}

useCallback緩存的是方法的引用,而useMemo緩存的則是方法的返回值。使用場景是減少不必要的子組件渲染:

function Parent({ a, b }) {
  // 當(dāng) a 改變時才會重新渲染
  const child1 = useMemo(() => , [a]);
  // 當(dāng) b 改變時才會重新渲染
  const child2 = useMemo(() => , [b]);
  return (
    <>
      {child1}
      {child2}
    
  )
}

如果想實現(xiàn)Class ComponentshouldComponentUpdate方法,可以使用React.memo方法,區(qū)別是它只能比較 props,不會比較 state:

const Parent = React.memo(({ a, b }) => {
  // 當(dāng) a 改變時才會重新渲染
  const child1 = useMemo(() => , [a]);
  // 當(dāng) b 改變時才會重新渲染
  const child2 = useMemo(() => , [b]);
  return (
    <>
      {child1}
      {child2}
    
  )
});
六、useRef

Class Component獲取 ref 的方式如下:

class MyComponent extends React.Component {
  constructor(props) {
    super(props);
    this.myRef = React.createRef();
  }
  
  componentDidMount() {
    this.myRef.current.focus();
  }  

  render() {
    return ;
  }
}

Hooks 的實現(xiàn)方式如下:

function() {
  const myRef = useRef(null);

  useEffect(() => {
    myRef.current.focus();
  }, [])
  
  return ;
}

useRef返回一個普通 JS 對象,可以將任意數(shù)據(jù)存到current屬性里面,就像使用實例化對象的this一樣。另外一個使用場景是獲取 previous props 或 previous state:

function Counter() {
  const [count, setCount] = useState(0);

  const prevCountRef = useRef();

  useEffect(() => {
    prevCountRef.current = count;
  });
  const prevCount = prevCountRef.current;

  return 

Now: {count}, before: {prevCount}

; }
七、自定義 Hooks

還記得我們上一篇提到的 React 存在的問題嗎?其中一點是:

帶組件狀態(tài)的邏輯很難重用

通過自定義 Hooks 就能解決這一難題。

繼續(xù)以上一篇文章中訂閱朋友狀態(tài)的例子:

import React, { useState, useEffect } from "react";

function FriendStatus(props) {
  const [isOnline, setIsOnline] = useState(null);

  useEffect(() => {
    function handleStatusChange(status) {
      setIsOnline(status.isOnline);
    }

    ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange);
    return () => {
      ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange);
    };
  });

  if (isOnline === null) {
    return "Loading...";
  }
  return isOnline ? "Online" : "Offline";
}

假設(shè)現(xiàn)在我有另一個組件有類似的邏輯,當(dāng)朋友上線的時候展示為綠色。簡單的復(fù)制粘貼雖然可以實現(xiàn)需求,但太不優(yōu)雅:

import React, { useState, useEffect } from "react";

function FriendListItem(props) {
  const [isOnline, setIsOnline] = useState(null);

  useEffect(() => {
    function handleStatusChange(status) {
      setIsOnline(status.isOnline);
    }

    ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange);
    return () => {
      ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange);
    };
  });

  return (
    
  • {props.friend.name}
  • ); }

    這時我們就可以自定義一個 Hook 來封裝訂閱的邏輯:

    import React, { useState, useEffect } from "react";
    
    function useFriendStatus(friendID) {
      const [isOnline, setIsOnline] = useState(null);
    
      useEffect(() => {
        function handleStatusChange(status) {
          setIsOnline(status.isOnline);
        }
    
        ChatAPI.subscribeToFriendStatus(friendID, handleStatusChange);
        return () => {
          ChatAPI.unsubscribeFromFriendStatus(friendID, handleStatusChange);
        };
      });
    
      return isOnline;
    }

    自定義 Hook 的命名有講究,必須以use開頭,在里面可以調(diào)用其它的 Hook。入?yún)⒑头祷刂刀伎梢愿鶕?jù)需要自定義,沒有特殊的約定。使用也像普通的函數(shù)調(diào)用一樣,Hook 里面其它的 Hook(如useEffect)會自動在合適的時候調(diào)用:

    function FriendStatus(props) {
      const isOnline = useFriendStatus(props.friend.id);
    
      if (isOnline === null) {
        return "Loading...";
      }
      return isOnline ? "Online" : "Offline";
    }
    
    function FriendListItem(props) {
      const isOnline = useFriendStatus(props.friend.id);
    
      return (
        
  • {props.friend.name}
  • ); }

    自定義 Hook 其實就是一個普通的函數(shù)定義,以use開頭來命名也只是為了方便靜態(tài)代碼檢測,不以它開頭也完全不影響使用。在此不得不佩服 React 團(tuán)隊的巧妙設(shè)計。

    八、Hooks 使用規(guī)則

    使用 Hooks 的時候必須遵守 2 條規(guī)則:

    只能在代碼的第一層調(diào)用 Hooks,不能在循環(huán)、條件分支或者嵌套函數(shù)中調(diào)用 Hooks。

    只能在Function Component或者自定義 Hook 中調(diào)用 Hooks,不能在普通的 JS 函數(shù)中調(diào)用。

    Hooks 的設(shè)計極度依賴其定義時候的順序,如果在后序的 render 中 Hooks 的調(diào)用順序發(fā)生變化,就會出現(xiàn)不可預(yù)知的問題。上面 2 條規(guī)則都是為了保證 Hooks 調(diào)用順序的穩(wěn)定性。為了貫徹這 2 條規(guī)則,React 提供一個 ESLint plugin 來做靜態(tài)代碼檢測:eslint-plugin-react-hooks。

    九、總結(jié)

    本文深入介紹了 6 個 React 預(yù)定義 Hook 的使用方法和注意事項,并講解了如何自定義 Hook,以及使用 Hooks 要遵循的一些約定。到此為止,Hooks 相關(guān)的內(nèi)容已經(jīng)介紹完了,內(nèi)容比我剛開始計劃的要多不少,想要徹底理解 Hooks 的設(shè)計是需要投入相當(dāng)精力的,希望本文可以為你學(xué)習(xí)這一新特性提供一些幫助。

    文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

    轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/103769.html

    相關(guān)文章

    • 新上課程推薦:《React Hooks 案例詳解(React 進(jìn)階必備)》

      摘要:課程制作和案例制作都經(jīng)過精心編排。對于開發(fā)者意義重大,希望對有需要的開發(fā)者有所幫助。是從提案轉(zhuǎn)為正式加入的新特性。并不需要用繼承,而是推薦用嵌套。大型項目中模塊化與功能解耦困難。從而更加易于復(fù)用和獨立測試。但使用會減少這種幾率。 showImg(https://segmentfault.com/img/bVbpNRZ?w=1920&h=1080); 講師簡介 曾任職中軟軍隊事業(yè)部,參與...

      Lin_YT 評論0 收藏0
    • React Hooks 解析(上):基礎(chǔ)

      摘要:第一次了解這項特性的時候,真的有一種豁然開朗,發(fā)現(xiàn)新大陸的感覺。為了解決這一痛點,才會有剪頭函數(shù)的綁定特性。它同時具備和三個生命周期函數(shù)的執(zhí)行時機(jī)。 歡迎關(guān)注我的公眾號睿Talk,獲取我最新的文章:showImg(https://segmentfault.com/img/bVbmYjo); 一、前言 React Hooks 是從 v16.8 引入的又一開創(chuàng)性的新特性。第一次了解這項特性...

      yy736044583 評論0 收藏0
    • React系列 --- 從Mixin到HOC再到HOOKS(四)

      摘要:返回元素的是將新的與原始元素的淺層合并后的結(jié)果。生命周期方法要如何對應(yīng)到函數(shù)組件不需要構(gòu)造函數(shù)。除此之外,可以認(rèn)為的設(shè)計在某些方面更加高效避免了需要的額外開支,像是創(chuàng)建類實例和在構(gòu)造函數(shù)中綁定事件處理器的成本。 React系列 React系列 --- 簡單模擬語法(一)React系列 --- Jsx, 合成事件與Refs(二)React系列 --- virtualdom diff算法實...

      Lionad-Morotar 評論0 收藏0
    • React 新特性 Hooks 講解及實例(四)

      摘要:粟例說明一下獲取子組件或者節(jié)點的句柄指向已掛載到上的文本輸入元素本質(zhì)上,就像是可以在其屬性中保存一個可變值的盒子。粟例說明一下渲染周期之間的共享數(shù)據(jù)的存儲上述使用聲明兩個副作用,第一個每隔一秒對加,因為只需執(zhí)行一次,所以每二個參為空數(shù)組。 想閱讀更多優(yōu)質(zhì)文章請猛戳GitHub博客,一年百來篇優(yōu)質(zhì)文章等著你! React 新特性講解及實例(一) React 新特性 Hooks 講解及實...

      aboutU 評論0 收藏0
    • Webpack Loader 高手進(jìn)階(一)

      摘要:在一個構(gòu)建過程中,首先根據(jù)的依賴類型例如調(diào)用對應(yīng)的構(gòu)造函數(shù)來創(chuàng)建對應(yīng)的模塊。 文章首發(fā)于個人github blog: Biu-blog,歡迎大家關(guān)注~ Webpack 系列文章: Webpack Loader 高手進(jìn)階(一)Webpack Loader 高手進(jìn)階(二)Webpack Loader 高手進(jìn)階(三) Webpack loader 詳解 loader 的配置 Webpack...

      MAX_zuo 評論0 收藏0

    發(fā)表評論

    0條評論

    最新活動
    閱讀需要支付1元查看
    <