“React 检测到 Hooks 的顺序发生了变化”,但 Hooks 似乎是按顺序调用的
Posted
技术标签:
【中文标题】“React 检测到 Hooks 的顺序发生了变化”,但 Hooks 似乎是按顺序调用的【英文标题】:"React has detected a change in the order of Hooks" but Hooks seem to be invoked in order 【发布时间】:2019-12-15 06:25:22 【问题描述】:我正在尝试通过 React 的钩子使用 Context
和 Reducers
,但遇到了钩子顺序不一致的问题。我的理解是,只要useHook(…)
的顺序保持不变,就可以在任何类型的控制流中调用返回的状态/更新函数/reducer。否则,我将在 FunctionComponents 的开头调用钩子。
是我在循环中生成Days
吗?还是遗漏了什么?
Warning: React has detected a change in the order of Hooks
called by Container. This will lead to bugs and errors if not fixed. For
more information, read the Rules of Hooks:
https://reactjs.org/docs/hooks-rules.html
Previous render Next render
------------------------------------------------------
1. useContext useContext
2. undefined useRef
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
Container
的完整版如下。下面是Day
的摘录,并有来自react-dnd
的useDrop
的参考。
export const Container: FunctionComponent<Props> = () =>
let events = useContext(State.StateContext)
//let events: Array<Event.Event> = [] <- with this, no warning
const getDaysEvents = (day: Event.Time, events: Array<Event.Event>) =>
return events.map(e =>
const isTodays = e.startTime.hasSame(day, "day")
return isTodays && Event.Event( dayHeight, event: e )
)
let days = []
for (let i = 0; i < 7; i++)
const day = DateTime.today().plus( days: i )
days.push(
<Day key=day.toISO() height=dayHeight date=day>
getDaysEvents(day, events)
</Day>
)
return <div className="Container">days</div>
摘自Day
(Event
同样使用useDrag
钩子,也像这里一样在顶层调用)。
const Day: FunctionComponent<DayProps> = ( date, height, children ) =>
const dispatch = useContext(State.DispatchContext)
const [ isOver, offset , dropRef] = useDrop(
// …uses the dispatch function within…
// …
)
// …
【问题讨论】:
这里没有看到 useRef ... 它在Day
组件内,而不是在Container 内。
该消息表明挂钩是由Container
调用的,而不是由“Day”调用的。这似乎很奇怪。 Event.Event()
有没有使用钩子?
是的,它是useDrag
,在顶层。使用来自它的 ref,以及闭包中的监视器返回的值。很高兴包含它,不过它与 Day
和 Container
是一样的。
如果 Event.Event 是像 Day 这样的反应组件(我从您的评论中了解到),问题是您直接调用它 Event.Event()
而不是返回 jsx return isTodays && <Event.Event dayHeight=dayHeight event=e />
【参考方案1】:
由于使用短路逻辑,我在编写的组件中遇到了同样的错误消息。
这导致了一个错误:
const x = useSelector(state => state.foo);
if (!x) return ;
const y = useSelector(state => state.bar);
这是因为当x
为真时,钩子列表的长度为2,但当x
为假时,列表的长度为1。
为了解决这个错误,我必须在提前终止之前使用所有钩子。
const x = useSelector(state => state.foo);
const y = useSelector(state => state.bar);
if (!x) return ;
【讨论】:
这里也一样,我做了 const x = something ? 0 : useMemo(); 谢谢!同样,不可能有条件地使用选择器。例如:let client = client_id && useSelector(state => state.clients[client_id])
不好,但 let client = useSelector(state => client_id ? state.clients[client_id] : null)
工作正常。【参考方案2】:
写下我的评论作为答案:
问题是你直接调用Event.Event()
,即使它是一个反应组件。这会导致 react 将函数内部的钩子调用视为 Container
的一部分,即使您打算让它们成为 Event 的一部分。
解决办法是使用JSX:
return isTodays && <Event.Event dayHeight=dayHeight event=e />
当你用生成的 JS 代码替换 JSX 时,为什么会更清楚:
return isTodays && React.createElement(Event.Event, dayHeight, event: e )
见https://reactjs.org/docs/react-api.html#createelement。您永远不想直接调用函数组件,react 的工作原理是您始终将引用组件传递给 react 并让它在正确的时间调用函数。
【讨论】:
它发生在我身上,当一个被破坏的道具对象在组件中未定义时,我通过删除未使用的被破坏的道具对象来解决这个问题。【参考方案3】:当你收到这个error
时,不是这个问题,而是其他原因
这实际上是由于钩子实现的任何不良做法而发生的
1 - 仅在顶层调用 Hooks
不要在循环、条件或嵌套函数中调用 Hooks。相反,请始终在 React 函数的顶层使用 Hooks
注意:首先在函数顶部实现
useState
钩子
2 - 仅来自 React 函数的调用挂钩
不要从常规 javascript 函数调用 Hooks
3 - 测试期间出错
如果您在测试组件时遇到此错误,请注意设置自定义挂钩的位置(替换到函数顶部)
最佳实践
使用 eslint 对代码进行 lint 以避免出现 React Hooks Rules 错误
使用 npm 或 yarn 安装包
npm install eslint-plugin-react-hooks --save-dev
【讨论】:
【参考方案4】:这不是问题场景,而是错误本身,希望它能帮助某人:)
const chatSession, userBelongsToSession = useChatSession(session_id)
const activeSession, setActiveSession = useActiveChatSession()
const isCurrentActiveSession = useMemo(() => activeSession != null && activeSession.session_id === session_id, [activeSession, session_id])
if (chatSession == null || activeSession == null)
return (<div></div>)
const Container = styled(BorderedContainer)`height: 72px; cursor: pointer;`
我在这段代码中也发生了同样的错误,这与 useRef 被 styled-components 调用而之前由于条件渲染而没有被调用有关
if (chatSession == null || activeSession == null)
return (<div></div>)
默认情况下,hook 将返回 null,并且我的组件将在没有 useRef 的情况下呈现,尽管当实际填充 hooks 时,styled-components 将生成一个带有 useRef 的组件。
if (chatSession == null || activeSession == null)
return (<div></div>)
const Container = styled(BorderedContainer)`height: 72px; cursor: pointer;
【讨论】:
【参考方案5】:我在编写的测试中调用不同的钩子方法时随机收到此错误。
对我的修复是在我实施的useRef
的间谍中:
const useRefSpy = jest
.spyOn(React, 'useRef')
.mockReturnValueOnce( whatever )
将mockReturnValueOnce
更改为mockReturnValue
修复了错误。
【讨论】:
【参考方案6】:从 useEffect 调用多个 api 调用
在我的情况下,我正在执行多个 api 调用并将它们中的每一个保存到不同的状态,这导致了我这个错误。但在将其更改为单一状态后,我能够克服这一点。
const [resp, setGitData] = useState( data: null, repos: null );
useEffect(() =>
const fetchData = async () =>
const respGlobal = await axios(
`https://api.github.com/users/$username`
);
const respRepos = await axios(
`https://api.github.com/users/$username/repos`
);
setGitData( data: respGlobal.data, repos: respGlobal.data );
;
fetchData();
, []);
【讨论】:
【参考方案7】:我在 react-native 应用程序中也遇到了同样的问题。这是由于不必要的元素加载过程而发生的。
旧代码
return (......
<NotificationDialog
show=notificationPayload !== null
title=notificationPayload?.title
.......
/>
);
这将重新渲染 NotificationDialog 甚至 notificationPayload !== null。但这不是必需的。只是我添加了这个空检查并避免在它为空时呈现这个 NotificationDialog
修复
return (......
notificationPayload !== null? <NotificationDialog
show=notificationPayload !== null
title=notificationPayload?.title
.......
/> : null
);
【讨论】:
以上是关于“React 检测到 Hooks 的顺序发生了变化”,但 Hooks 似乎是按顺序调用的的主要内容,如果未能解决你的问题,请参考以下文章