React Hook丨真正的邏輯復用

說起邏輯復用颁井,熟悉 react 小伙伴們一口道出了 HOC [高階組件] 厅贪。沒錯,高階組件可以實現(xiàn)邏輯復用雅宾,在 hook 之前 react 還有挺多不錯的方案卦溢。那么,讓我們來淺談 HOC 與 自定義 hook秀又。

HOC邏輯復用

說起HOC单寂,我想到了兩個標簽:1.【嵌套】 2.【一直嵌套】

讓我們來深入場景,舉個例子:

以封裝一個 input 雙向綁定為例

我們經(jīng)常會這樣去做一個雙向綁定

// ...
state = {
  value: 1
};
onChange = (e: any) => {
  this.setState({
    value: e.target.value
  });
};
// ...
<input value={this.state.value} onChange={this.onChange} />;

假設(shè)在一個組件內(nèi)有多個 input 我們希望可以更好的去復用「雙向綁定」的邏輯吐辙,于是我們對這塊邏輯用 HOC 進行抽象:

HOCInput.tsx

const HOCInput = (WrappedComponent: any) => {
  return class extends React.Component<
    {},
    {
      fields: {
        [key: string]: {
          value: string;
          onChange: (e: any) => void;
        };
      };
    }
  > {
    constructor(props: any) {
      super(props);
      this.state = {
        fields: {}
      };
    }
    setField = (name: string) => {
      if (!this.state.fields[name]) {
        this.state.fields[name] = {
          value: "",
          onChange: (event: any) => {
            this.state.fields[name].value = event.target.value;
            this.forceUpdate();
          }
        };
      }
      return {
        value: this.state.fields[name].value,
        onChange: this.state.fields[name].onChange
      };
    };
    getFieldValueTrim = (name: string) => {
      return this.state.fields[name]
        ? this.state.fields[name].value.trim()
        : "";
    };
    render() {
      const { setField, getFieldValueTrim } = this;
      const newProps = { setField, getFieldValueTrim };
      return <WrappedComponent {...this.props} {...newProps} />;
    }
  };
};

在 Vue 中有不錯的 v-model.trim 語法糖【自動去掉字符串頭尾空格】宣决,避免我們提交的時候有多余的空格。所以昏苏,這里我們實現(xiàn)這樣的功能并將 getFieldValueTrim 方法掛到 props 上尊沸。

調(diào)用

Demo1Component.tsx

class Demo1Component extends React.Component<{
  setField?: (name: string) => { value: string; onChange: (e: any) => {} };
  getFieldValueTrim?: (name: string) => string;
}> {
  render() {
    const { setField, getFieldValueTrim } = this.props;
    console.log("name :>> ", getFieldValueTrim!("name"));
    return (
      <div>
        <input {...setField!("name")} />
        <br />
        <input {...setField!("email")} />
      </div>
    );
  }
}
// 嵌套
const Demo1 = HOCInput(Demo1Component);
//...
<Demo1 />
// ...

這樣,我們就用 HOC 完成了一個邏輯復用贤惯。假設(shè)洼专,我們還有一個或多個「邏輯」需要抽象成一個高階組件呢?

如:我想要點擊按鈕隨機切換 input 框的背景顏色孵构。

那就讓我們繼續(xù)封裝 HOC

HOCInputBgColor.tsx

const HOCInputBgColor = (initialColor: string) => (WrappedComponent: any) => {
  return class extends React.Component<{}, { color: string }> {
    state = {
      color: initialColor
    };
    getRandomColor = () => {
      const randomNum = () => Math.floor(Math.random() * 100);
      return `rgb(${randomNum()},${randomNum()},${randomNum()})`;
    };
    handleChangeColor = () => this.setState({ color: this.getRandomColor() });
    render() {
      const newProps = {
        color: this.state.color,
        handleChangeColor: this.handleChangeColor
      };
      return <WrappedComponent {...this.props} {...newProps} />;
    }
  };
};

在原來的組件上進行調(diào)用

Demo1Component.tsx

class Demo1Component extends React.Component<{
  setField?: (name: string) => { value: string; onChange: (e: any) => {} };
  getFieldValueTrim?: (name: string) => string;
  color?: string;
  handleChangeColor?: () => void;
}> {
  render() {
    const {
      setField,
      getFieldValueTrim,
      color,
      handleChangeColor
    } = this.props;
    return (
      <div>
        <input style={{ background: color! }} {...setField!("name")} />
        <br />
        <button onClick={handleChangeColor!}>change bg-color</button>
      </div>
    );
  }
}
/
const Demo1 = HOCInput(HOCInputBgColor("rgb(158,158,158)")(Demo1Component));

當我們有更多的 HOC 時屁商,那么就會一直嵌套下去,好在有ts裝飾器的支持颈墅,讓我們看這個「嵌套」看著更加舒適蜡镶,如:

@HOCInput
@HOCInputBgColor("rgb(158,158,158)")
class Demo1Component extends React.Component { } 

我們也不再需要重新把組件賦值給一個變量雾袱,在調(diào)用組件的時候,直接 <Demo1Component />官还。

HOC缺點

當組件在調(diào)用多個HOC時芹橡,會調(diào)用 props 上 HOC 傳遞下來的 值/方法,如上面的例子:

const {
  setField,
  getFieldValueTrim,
  color,
  handleChangeColor
} = this.props;

要是 HOC 一多命名就要形成規(guī)范望伦,否則將有可能導致重命名發(fā)生覆蓋林说。這算是 HOC 的一個缺點吧。

自定義 hook 邏輯復用

官網(wǎng):自定義 hook 解決了以前在 React 組件無法靈活共享邏輯的問題屯伞。

我們直接把上面的例子改成 hook 版看看腿箩。

useInput.ts【自定義 Hook 名稱需要以 “use” 開頭】

const useInput = (
  initialValue = ""
): [{ value: string; onChange: (e: any) => void }, string] => {
  const [value, setValue] = useState(initialValue);
  const onChange = (e: any) => {
    setValue(e.target.value);
  };
  return [
    {
      value,
      onChange
    },
    `${value}`.trim()
  ];
};

使用

Demo2.tsx

const Demo2: React.FC = () => {
  const [nameIpt, name] = useInput();
  const [emailIpt, email] = useInput();
  console.log("Hook-name :>> ", name);
  console.log("Hook-email :>> ", email);
  return (
    <div>
      <input {...nameIpt} />
      <br />
      <input {...emailIpt} />
    </div>
  );
};

可以明顯的看到幾個優(yōu)點:

  1. 代碼更簡潔

  2. 不存在重命名覆蓋
    解釋:現(xiàn)在的可復用狀態(tài)沒有像 HOC 掛到被包裝組件的 this.props 上了,我們都知道 hook 的寫法可以暴露出一個數(shù)組:[ 值 , 方法 ]愕掏。在使用的時候度秘,可以用解構(gòu)的手法來實現(xiàn)對數(shù)組內(nèi)變量名的自定義顶伞,保證命名不重復饵撑。

  3. 沒有嵌套

如果還有更多的邏輯需要被抽象,我們只管繼續(xù)封裝 useXxx唆貌,然后在組件中進行使用滑潘。
如上面講的 HOCInputBgColor 高階組件,我們也可以用 hook版進行封裝锨咙,如 useInputBgColor语卤,小伙伴們,動手試試看吧~

總結(jié)

在數(shù)據(jù)的處理中酪刀,我們知道在處理“平級”的數(shù)據(jù)粹舵,往往比嵌套的、樹形的數(shù)據(jù)來得簡單骂倘。

就如:

const arr = [1, [2, 3, [4, 5]]];
arr.flat('Infinity'); 
// [1, 2, 3, 4, 5]

個人覺得 自定義hook 就類似這樣一個“拉平”眼滤,讓我們對于數(shù)據(jù)的處理更直觀,更不容易犯錯历涝。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末诅需,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子荧库,更是在濱河造成了極大的恐慌堰塌,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,640評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件分衫,死亡現(xiàn)場離奇詭異场刑,居然都是意外死亡,警方通過查閱死者的電腦和手機蚪战,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,254評論 3 395
  • 文/潘曉璐 我一進店門摇邦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來恤煞,“玉大人,你說我怎么就攤上這事施籍【影牵” “怎么了?”我有些...
    開封第一講書人閱讀 165,011評論 0 355
  • 文/不壞的土叔 我叫張陵丑慎,是天一觀的道長喜喂。 經(jīng)常有香客問我,道長竿裂,這世上最難降的妖魔是什么玉吁? 我笑而不...
    開封第一講書人閱讀 58,755評論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮腻异,結(jié)果婚禮上进副,老公的妹妹穿的比我還像新娘。我一直安慰自己悔常,他們只是感情好影斑,可當我...
    茶點故事閱讀 67,774評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著机打,像睡著了一般矫户。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上残邀,一...
    開封第一講書人閱讀 51,610評論 1 305
  • 那天皆辽,我揣著相機與錄音,去河邊找鬼芥挣。 笑死驱闷,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的空免。 我是一名探鬼主播空另,決...
    沈念sama閱讀 40,352評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼鼓蜒!你這毒婦竟也來了痹换?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,257評論 0 276
  • 序言:老撾萬榮一對情侶失蹤都弹,失蹤者是張志新(化名)和其女友劉穎娇豫,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體畅厢,經(jīng)...
    沈念sama閱讀 45,717評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡冯痢,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,894評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片浦楣。...
    茶點故事閱讀 40,021評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡袖肥,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出振劳,到底是詐尸還是另有隱情椎组,我是刑警寧澤,帶...
    沈念sama閱讀 35,735評論 5 346
  • 正文 年R本政府宣布历恐,位于F島的核電站寸癌,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏弱贼。R本人自食惡果不足惜蒸苇,卻給世界環(huán)境...
    茶點故事閱讀 41,354評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望吮旅。 院中可真熱鬧溪烤,春花似錦、人聲如沸庇勃。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,936評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽匪凉。三九已至枪眉,卻和暖如春捺檬,著一層夾襖步出監(jiān)牢的瞬間再层,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,054評論 1 270
  • 我被黑心中介騙來泰國打工堡纬, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留聂受,地道東北人。 一個月前我還...
    沈念sama閱讀 48,224評論 3 371
  • 正文 我出身青樓烤镐,卻偏偏與公主長得像蛋济,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子炮叶,可洞房花燭夜當晚...
    茶點故事閱讀 44,974評論 2 355

推薦閱讀更多精彩內(nèi)容