React 的 HOC 以及的 Render Props

来源:https://www.cnblogs.com/cnyball/archive/2018/07/15/9313465.html
-Advertisement-
Play Games

重覆是不可能的,這輩子都不可能寫重覆的代碼 當然,這句話分分鐘都要被產品打臉,天天喊著改需求,老哥,這裡改下可好? 所以,我們需要抽象,封裝重覆的功能或者邏輯,這樣改需求,也不用到處改 那麼這次我們來講講 React 里的高級組件 React 高級組件有兩種方式: 1. 使用高階組件( Higher ...


重覆是不可能的,這輩子都不可能寫重覆的代碼

當然,這句話分分鐘都要被產品打臉,天天喊著改需求,老哥,這裡改下可好?
所以,我們需要抽象,封裝重覆的功能或者邏輯,這樣改需求,也不用到處改

那麼這次我們來講講 React 里的高級組件

React 高級組件有兩種方式:

  1. 使用高階組件( Higher Order Component ==> HOC )
  2. 子組件作為函數的模式( Render Props )

高階組件

首先來說說高階組件,它不是 React 的提供的 API,它是模式,一種增強組件的模式,簡單來說其實就是高階函數,高階函數大家應該不陌生

高階函數 : 把函數作為參數傳入,返回一個新的函數,這樣的函數稱為高階函數

而所謂的高階組件也就是傳入一個現有組件作為參數,返回一個新的組件,高階租價也分為代理方式和繼承方式
我們先來看看一個簡單的 HOC :

import React from 'react';

function addNameProp(WrapperComponent) {
  return class extends React.Component {
    render() {
      const { name, ...props } = this.props;
      return <WrapperComponent {...props} name={name || 'cnyballk'} />;
    }
  };
}

export default addNameProp;

在上面的代碼里我們定義了一個 addNameProp。 函數,它的作用就是為沒有傳入 name 的 prop 傳入的組件添加一個 name 的 prop ,然後渲染出來

這個高階組件就完成了對傳入組件的增強,這也是一個代理方式的高階組件

那麼繼承方式是怎麼樣的呢

import React from 'react';

function addNameProp(WrapperComponent) {
  return class extends WrapperComponent {
    render() {
      this.props = {
        ...this.props,
        name: this.props.name || 'canyballk',
      };
      return super.render();
    }
  };
}

export default addNameProp;

和上面的高階組件 一樣的效果,而繼承模式和代理模式最大的區別就是一個是返回傳入的組件,一個是調用繼承的父組件的 render 方法

⚠️ 在代理模式下 WrapperComponent 經歷了完整的生命周期,返回的組件也有一次完整的生命周期,而繼承模式 則是調用了 WrapperComponent 的 render ,只有返回的組件的一個生命周期

如果我們用過 react-redux 的 connect 也就知道 connect 也是一個代理模式的高階組件

⚠️ 高階組件可以這麼使用

@Wrapper
class Index extends Component {
   ...略
}

emmmm...這個涉及到 js 的裝飾器,各位自己去搜來學習哈

Render Props

那麼我們已經認識了高階組件的重用方法,也認識到了高階組件的優點。

但是高階組件的缺點是什麼呢?

相信你們也能看到了,我們傳遞一個 name 的 Prop ,那是不是得傳入的組件能夠處理才可以,而且還有命名的衝突危險,畢竟有些組件的 props 命名各有不同,或者說子組件需要同一份數據處理方式卻不一樣,所以說這個時候就不適用 HOC 了

HOC 對於使用者來說是一個黑盒,還需要去看他們的實現

而 Render Props 則是 數據給你,其他的你自己來
我們來看一個簡單的例子

import React from 'react';

class AddNameProp extends React.Component {
  render() {
    const name = 'cnyballk';
    return this.props.children(name);
  }
}

export default AddNameProp;

簡單的使用

<AddNameProp>{name => <div>我叫 {name}</div>}</AddNameProp>

想傳遞給組件

<AddNameProp>
  {name =>
    <Hello name={name}/>;
  }
</AddNameProp>

或者是其他的 prop 名

<AddNameProp>
  {name =>
    <Hello myName={name}/>;
  }
</AddNameProp>

我們可以看到,這個方式很靈活,因為子組件是一個函數,一切皆有可能,所以現在有很多 Render Props 黨,但是 Render Props 也有缺點就是難以做性能優化,高階組件可以利用 SCU 來避免重覆渲染。雖然這樣, Render Props 卻是一個非常好的一個模式,我也非常喜歡

程式員可以懶,但不能是複製粘貼的懶

固然有很多人喜歡偷懶複製粘貼,但俗話說的好,複製粘貼一時爽,代碼重構火葬場,一旦需要修改需求或者出現 bug,到處改顯然是不行的,我們應該做好封裝

小結

本文介紹了 React 的高級組件的兩個方式,各有優缺點,但它們都是為了重用代碼,我們可以自己選擇喜歡的模式去做

高階組件代理模式可以更好的控制和實現,繼承模式則可以控制特定組件的生命周期,與高階組件相比, Render Props 模式更加靈活,有函數,我們可以更自由


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.前言 Vue 是一款友好的、多用途且高性能的javascript框架,與其它大型框架不同的是,Vue 被設計為可以自底向上逐層應用,它能夠幫你創建可維護性和可測試性更強的代碼庫,Vue是漸進式的javascript框架庫。 2.安裝簡介 Vue集成環境三大核心組件:node.js,npm,vue ...
  • 網路載入類 減少 HTTP 資源請求次數 合併靜態資源圖片、JavaScript 或 CSS 代碼,減少頁面請求數和資源請求消耗 避免重覆的資源,防止增加多餘請求 減小 HTTP 請求大小 減少沒必要的圖片、JavaScript、CSS 及 HTML 代碼 對文件進行壓縮優化 使用 gzip 等方式 ...
  • 表單選擇器 :input(匹配所有input、textarea、select和button元素) :text(匹配所有單行文本框) :password(匹配所有密碼框) :radio(匹配所有單項按鈕) :checkbox(匹配所有覆選框) :submit(匹配所有提交按鈕) :image(匹配所有 ...
  • 在《JavaScript高級程式設計》第三版 4.1.3,講到傳遞參數: ECMAscript中所有函數的參數都是按值傳遞 按值傳遞 也就是,把函數外部的值複製給函數內部的參數,就和把值從一個變數複製到另一個變數一樣 當傳遞value給函數foo的時候,相當於拷貝一份value給foo假設拷貝的那份 ...
  • 介紹 什麼是Browsh? Browsh是一個純文本瀏覽器,可以運行在大多數的TTY終端環境和任何瀏覽器。目前1 ,終端客戶端比瀏覽器客戶端更先進2。 TTY 客戶端 終端客戶端即時更新和交付,以便於體驗新的功能,例如,你可以觀看視頻。它使用UTF 8半塊技巧(& 9600;)3從每個字元單元中獲取 ...
  • 參考: Best Practices for Speeding Up Your Web Site ...
  • JS在if中的強制類型轉換 眾所周知,JS在很多情況下會進行強制類型轉換,其中,最常見兩種是: 1.使用非嚴格相等進行比較,對 左邊的值進行類型轉換 2.在if判斷時,括弧內的值進行類型轉換,轉化為布爾值 今天,我就來聊一聊JS在if中的強制類型轉換。 其實,如果詳細要討論哪些值在if中強制轉換為 ...
  • 版權聲明:未經博主允許不得轉載 SQL DDL數據定義語言 TPL事務處理語言 DCL數據控制語言 DML數據操作語言 DML SELECT INSERT UPDATE DELETE Join從句 Join: 內連接INNER 全外連接FULL OUTER 左外連接LEFT OUTER 右外連接RI ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...