React v16.7 之 Hooks

Hooks are an upcoming feature that lets you use state and other React features without writing a class. They’re currently in React v16.7.0-alpha.

What

Hooks 是 React 函数组件内的一类特殊函数,使开发者能在 function component 里继续使用 state 和 lifecycle。通过 Custom Hooks 可以复用业务逻辑,从而避免添加额外的components。

Why

过去我们使用React时,component 基本分为两种:function component 和 class component 。其中function component就是一个 pure render component,不存在 state 和 lifecycle 。function component 使组件之间耦合度降低,但一旦需要 state 或 lifecycle ,就需要变成 class component 。而 class component 也会带来一些缺点:

  • React 组件树过于臃肿

单向数据流使组件间的通信必须一层一层往下传,当有些状态不适合使用 Redux 这种 global store 的情况下,此时组件之间的逻辑复用和沟通就会变得十分困难。为此,过去我们会使用各种 HOC(高阶组件)来传递状态。这就导致了当应用规模越来越庞大的时候,会多了很多无关 UI 的 wrapper 组件,也就使得 React 组件树变得越来越臃肿,开发和调试效率随之变低。

Write

了解了Hooks的基本知识,接下来就是如何去使用 Hooks API 了。Hooks 主要分为三种:

  • State Hook (在 function component 使用 state)
  • Effect Hook (在 function component 使用 lifecycle 和 side effect )
  • Custom Hook (通过自定义Hook来复用逻辑)
WX20181224-184244@2x.png

State Hook

官方 Example:

import { useState } from 'react';

function Example() {
  // Declare a new state variable, which we'll call "count"
  const [count, setCount] = useState(0);

  return (
    <div>
      <p>You clicked {count} times</p>
      <button onClick={() => setCount(count + 1)}>
        Click me
      </button>
    </div>
  );
}

Hook 的本质实际上就是一个特殊的函数,通常以"use"开头。这里的 useState 就是一个Hook,通过它可以实现在组件内使用state。useState 会返回一个pair:分别是当前 state 的值和修改这个 state 的函数。用法有点类似 class component 里的 this.setState,只是这里不会合并 state 对象,而且注意到没,这里的 useState 的初始值是0,跟 this.state 不同在于它不需要是一个 Object。

官方有个例子对比 useState 与 this.state 的。(传送门:https://reactjs.org/docs/hooks-state.html

多个 state 变量

function ExampleWithManyStates() {
  // Declare multiple state variables!
  const [age, setAge] = useState(42);
  const [fruit, setFruit] = useState('banana');
  const [todos, setTodos] = useState([{ text: 'Learn Hooks' }]);
  // ...
}

你完全可以声明多个 useState,一点问题都木有!这种做法带来的好处是:我们的 state 将不会变得非常臃肿,每个 state 都非常直观,独立。

Effect Hook

我们经常在 React 组件中进行拉取数据、订阅或操作DOM,这种行为被称为 "side effects"(副作用),这种行为通常只能在生命周期中执行,而不能在 render 里。

React 通过 useEffect 来解决这样的问题,具体怎么用我们看看例子:

import { useState, useEffect } from 'react';

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

  // Similar to componentDidMount and componentDidUpdate:
  useEffect(() => {
    // Update the document title using the browser API
    document.title = `You clicked ${count} times`;
  });

  return (
    <div>
      <p>You clicked {count} times</p>
      <button onClick={() => setCount(count + 1)}>
        Click me
      </button>
    </div>
  );
}

这种做法相当于把 React classes 里的 componentDidMountcomponentDidUpdatecomponentWillUnmount 合并成了一个。

由于 Hooks 是定义在最外层函数的,所以这里是能访问到 props 和 state 的,它默认会在每一次 render 后都会被调用(包括第一次)。

同样的,官方有个例子对比了 useEffect 和 class lifecycle 的。(传送门:https://reactjs.org/docs/hooks-effect.html

cleanup

通常我们的大部分 effect 行为都是不需要清理,比如网络请求、DOM操作或者日志记录等。但如果我们在 effects 进行了类似外部数据订阅这样的操作,那么我们就需要在 Unmount 前取消订阅。针对这种场景,可以通过在 useEffect 中返回一个函数的方式进行清理。此处应该有代码:

import { useState, useEffect } from 'react';

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

  function handleStatusChange(status) {
    setIsOnline(status.isOnline);
  }

  useEffect(() => {
    ChatAPI.subscribeToFriendStatus(props.friend.id, handleStatusChange);
    // Specify how to clean up after this effect:
    return function cleanup() {
      ChatAPI.unsubscribeFromFriendStatus(props.friend.id, handleStatusChange);
    };
  });

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

乍一看,使用 Hook 的方式相对于 class component 只是代码写少点而已,但实际上它带来的好处可不止这些。如果用 class component 的话,我们需要在 componentDidMount 订阅 props.friend.id 的状态,然后在 componentWillUnmount 中取消订阅。

  componentDidMount() {
    ChatAPI.subscribeToFriendStatus(
      this.props.friend.id,
      this.handleStatusChange
    );
  }

  componentWillUnmount() {
    ChatAPI.unsubscribeFromFriendStatus(
      this.props.friend.id,
      this.handleStatusChange
    );
  }

但这种做法可能会引发bug(试想一下,如果 friend prop 变了会怎样?)。这种情况下,页面上显示的状态就不是当前这个 friend 的啦~ 所以这是一个bug,解决的方式就是在 componentDidUpdate 中先进行 unsubscribe,再重新 subscribe

componentDidUpdate(prevProps) {
    // Unsubscribe from the previous friend.id
    ChatAPI.unsubscribeFromFriendStatus(
      prevProps.friend.id,
      this.handleStatusChange
    );
    // Subscribe to the next friend.id
    ChatAPI.subscribeToFriendStatus(
      this.props.friend.id,
      this.handleStatusChange
    );
  }

在实际开发中,我们可能经常会没考虑到这种情况。那么现在有了 Hook,你可以不用担心了!

性能优化之 Skipping effects

每次 render 都会 cleanup 或执行 effect,这可能会导致性能问题。在 class component 中我们通常会在 componentDidUpdate 里进行对比判断。而在 Hook 里,我们可以通过给 useEffect 传递第二个参数(数组形式)来选择 Hook 的触发时机,

useEffect(() => {
  document.title = `You clicked ${count} times`;
}, [count]); // Only re-run the effect if count changes

同步的 useLayoutEffect

必须说明的是,useEffect 是异步的,即它不会阻塞浏览器渲染页面,因为大多数情况下,这些 effects 都不需要同步执行。在极少数的场景下,可以选择用 useLayoutEffect。它会在 re-render 后同步执行,阻塞浏览器进行渲染。

Custom Hook

过去我们在组件中复用逻辑的通常做法是使用高阶函数 ( high-order component ) 和 render props。如今有了 Hooks,我们可以避免添加更多的组件到我们的组件树中了。

我们把上面 FriendStatus 稍加改动。

import { useState, useEffect } from 'react';

function useFriendStatus(friendID) {
  const [isOnline, setIsOnline] = useState(null);

  function handleStatusChange(status) {
    setIsOnline(status.isOnline);
  }

  useEffect(() => {
    ChatAPI.subscribeToFriendStatus(friendID, handleStatusChange);
    return () => {
      ChatAPI.unsubscribeFromFriendStatus(friendID, handleStatusChange);
    };
  });

  return isOnline;
}

它对外接受一个 friendId 作为参数,然后返回具体状态。而在其他组件里引用也非常简单:

function FriendStatus(props) {
  const isOnline = useFriendStatus(props.friend.id);

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

这样,复用了逻辑而又不需要引入新的 state,简直完美!

其他 built-in Hooks

还好很多内置的 Hooks,如:

  • useContext(替代<Context.Consumer> 使用render props 的写法)
function Example() {
  const locale = useContext(LocaleContext);
  const theme = useContext(ThemeContext);
  // ...
}
  • useReducer (相当于组件自带 redux reducer,负责 dispatch action 并更新 state)
function Todos() {
  const [todos, dispatch] = useReducer(todosReducer);
  // ...

注意事项

  • 在最外层函数使用 Hooks ,不要在循环块、条件块或函数内调用 Hooks;
  • 只在 React function component 里使用 Hooks(除非是Custom Hooks,否则不要在普通函数中使用 Hooks)

最后总结一下

在 React 里,Hooks 就是一系列的特殊函数,在 function component 内部“勾住” 组件的 state 和 lifecycle。Hook 是向后兼容的,但官方不推荐大家将旧代码的 class component 都改成 Hook,大家可以在新代码中体验一下这种写法。针对 Hooks 新特性的官方文档很详细,这里限于篇幅,就不过多讲了,推荐大家去看官方文档。

掰掰~

参考文档:https://reactjs.org/docs/hooks-intro.html

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342

推荐阅读更多精彩内容

  • It's a common pattern in React to wrap a component in an ...
    jplyue阅读 3,248评论 0 2
  • 原教程内容详见精益 React 学习指南,这只是我在学习过程中的一些阅读笔记,个人觉得该教程讲解深入浅出,比目前大...
    leonaxiong阅读 2,809评论 1 18
  • Learn from React 官方文档 一、Rendering Elements 1. Rendering a...
    恰皮阅读 2,659评论 2 3
  • 挂单碧水源涨停板,后取消;战胜心理因素,坚决持有,等待市场发出卖出信号; 北新建材卖出后上涨,有后悔心理,因为之前...
    W大先生阅读 99评论 0 0
  • 图片发自简书App 浣衣娘 人言苏杭是天堂 玉景美人遮眼忙 我说...
    c45736e529f0阅读 473评论 0 6