Maison  >  Article  >  interface Web  >  Une brève analyse du nouveau concept Transition dans React18

Une brève analyse du nouveau concept Transition dans React18

青灯夜游
青灯夜游avant
2022-03-25 10:52:462384parcourir

Cet article vous fera comprendre le nouveau concept de Transition dans React18, et présentera brièvement l'utilisation de la nouvelle API : startTransition et les nouveaux hooks : useTransition et usedeferredValue. J'espère que cela sera utile à tout le monde !

Une brève analyse du nouveau concept Transition dans React18

React 18 introduit un nouveau concept - transition, qui apporte une nouvelle API - startTransition et deux nouveaux hooks ——useTransition et usedeferredValue, cet article commence par une introduction aux premiers utilisateurs. [Recommandations associées : Tutoriel vidéo Redis]transition,由此带来了一个新的API——startTransition和两个新的hooks——useTransitionusedeferredValue,本文由此展开使用尝鲜介绍。【相关推荐:Redis视频教程

1. 总览

本文分为4部分进行:

  • tansition 产生初衷
  • startTransition 使用和介绍
  • useTransition 使用和介绍
  • useDeferredValue 使用和介绍

2. transition产生初衷

transtion 直接翻译为 过渡。tansition本质上是为了解决渲染并发问题所提出。在React中一旦组件状态改变并触发了重新渲染,则无法停止渲染。直到组件重新渲染完毕,页面才能继续响应用户的交互。

为此react 18中更新都可以划分为以下两类:

  • 紧急更新(urgent update):用户期望马上响应的更新操作,例如鼠标单击或键盘输入。
  • 过渡更新(transition update):一些延迟可以接受的更新操作,如查询时,搜索推荐、搜索结果的展示等。
// 被startTransiton标记后为过渡更新
startTransition(()=> {
    // 非紧急更新,会被降低优先级,延迟执行
    setQueryValue(inputValue)
})

// 未被标记则马上执行
setInputValue(inputValue)

在react 18中被startTrionstion标记的更新,即为过渡更新(执行的优先级被降低),此时react会根据内部的调度机制延迟执行内部的state更新。

开发中开发者可以通过transition hook决定哪些更新被标记为transition事件。一旦被标记则代表为低优先级执行,即react知道该state可以延迟更新,通过区分更新优先级,让高优先级的事件保持响应,提高用户交互体验,保持页面响应

3. startTransiton

startTransiton使用介绍

const handleClick = () => {
    // startTransition包裹标记为低优先级更新
    startTransition(()=> {
        setQueryValue(inputValue)
    })
    
    // 未被标记则马上执行
    setInputValue(inputValue)
}

首先我们来介绍下最简单的startTransition

  • startTransiton 是一个接受回调的函数,用于告知React需要延迟更新的state。
  • 如果某个state的更新会导致组件挂起,则应该包裹在startTransition中

通过演示对比

这是一个对输入字符后展示搜索结果的场景模拟,通过伪造大量搜索结果,模拟容易卡顿的情况。

我们试着连续输入123,监听搜索框值value变化(urgent update)和搜索值searchVal变化(transition update)并输出到控制栏。

import React, { useEffect, useState, startTransition } from 'react';
import './App.css'

const SearchResult = (props) => {
    const resultList = props.query
        ? Array.from({ length: 10000 }, (_, index) => ({
            id: index,
            keyword: `${props.query} -- 搜索结果${index}`,
        })) : [];
    return resultList.map(({ id, keyword }) => (
        <li key={id}>{keyword}</li>
    ))
}

const App = () => {
    const [type, setTpye] = useState(1)
    const [value, setValue] = useState(&#39;&#39;);
    const [searchVal, setSearchVal] = useState(&#39;-&#39;);

    useEffect(() => {
        // 监听搜索值改变
        console.log(&#39;对搜索值更新的响应++++++&#39; + searchVal + &#39;+++++++++++&#39;)
    }, [searchVal])

    useEffect(() => {
        console.log(&#39;对输入框值更新的响应-----&#39; + value + &#39;-------------&#39;)
        if (type === 1) {
            setSearchVal(value || &#39;-&#39;)
        }
        if (type === 2) {
            startTransition(() => {
                setSearchVal(value || &#39;-&#39;)
            })
       }
    }, [value, type]);

    return (
        <div className=&#39;App&#39;>
            <input value={value} onChange={e => setValue(e.target.value)} />
            <div className={`type_button ${type === 1 ? &#39;type_button_checked&#39; : &#39;&#39;}`} onClick={() => setTpye(1)}>normal</div>
            <div className={`type_button ${type === 2 ? &#39;type_button_checked&#39; : &#39;&#39;}`} onClick={() => setTpye(2)}>transiton</div>
            <ul>
                <SearchResult query={searchVal}></SearchResult>
            </ul>
        </div>
    );
};

普通模式下

Une brève analyse du nouveau concept Transition dans React18

如图所示:连续输入字符123,当第一个字符输入后,搜索值马上响应,列表渲染立刻开始,造成卡顿输入框停止了对用户输入的响应,直到渲染结束,输入框才继续响应。

使用startTransition后

Une brève analyse du nouveau concept Transition dans React18

如图所示:连续输入字符123,输入框不断响应,搜索值的响应被延后,保证页面反馈,直到输入结束,才开始响应搜索值,渲染搜索结果,保持页面响应。

4. useTransiton

useTransiton使用介绍

import { useTransiton } from &#39;react&#39;

const [isPending, startTransition] = useTransiton({timeoutMs: 2000})
// 例如, 在pending状态下,您可以展示一个Spinner
{ isPending ? < Spinner /> : null }
  • startTransition 是一个接受回调的函数,用于告知React需要延迟更新的state。
  • isPending 是一个布尔值,这是react告知我们是否等待过渡完成的方式。
  • useTransition 接受带有 timeoutMs 的延迟响应的值,如果给定的timeoutMs内未完成,它将会强制执行startTransition回调函数内state的更新。

useTransiton简单分析

我们通过伪代码理解下useTransition

1. Présentation🎜🎜Cet article est divisé en 4 parties : 🎜
  • transition Intention originale
  • Utilisation et introduction de
  • startTransition
  • useTransition utilisation et introduction
  • useDeferredValue utilisation et introduction
🎜2. L'intention initiale de la transition🎜🎜transtion Traduction directe pour transition. Tansition est essentiellement proposé pour résoudre le problème de la concurrence de rendu. Dans React, une fois que l'état du composant change et qu'un nouveau rendu est déclenché, le rendu ne peut pas être arrêté. La page ne peut pas continuer à répondre à l'interaction de l'utilisateur jusqu'à ce que le composant soit restitué. 🎜🎜Pour cette raison, les mises à jour de React 18 peuvent être divisées dans les deux catégories suivantes : 🎜
  • Mise à jour d'urgence (mise à jour urgente) : opérations de mise à jour auxquelles les utilisateurs s'attendent à répondre immédiatement, telles que sous forme de clics de souris ou de saisie au clavier.
  • Mise à jour de transition (mise à jour de transition) : certaines opérations de mise à jour avec un délai acceptable, telles que le temps de requête, les recommandations de recherche, l'affichage des résultats de recherche, etc.
function useTransition(){
    const [isPending, setPending] = mountState(false);
    const start = (callback)=>{
        setPending(true);
        // Scheduler.unstable_next 通过 transiton 模式,低优先级调度执行回调函数
        // 可以降低更新的优先级。如果回调中触发的更新优先级会比较低,
        // 它会让位为高优先级的更新,或者当前事务繁忙时,调度到下一空闲期再应用。
        Scheduler.unstable_next(() => {
            const prevTransition = ReactCurrentBatchConfig.transition;
            ReactCurrentBatchConfig.transition = 1;
            try {
                setPending(false);
                //实行回调函数
                callback();
            } finally {
                ReactCurrentBatchConfig.transition = prevTransition;
            }
        })
    }
    return [isPending, start];
}
🎜La mise à jour marquée par startTrionstion dans React 18 est une mise à jour transitoire (la priorité d'exécution est réduite à ce moment, React retardera en fonction de la planification interne). mécanisme Effectuer des mises à jour d’état internes. 🎜🎜Les développeurs en cours de développement peuvent décider quelles mises à jour sont marquées comme événements de transition via des hooks de transition. Une fois marqué, il représente une exécution de faible priorité, c'est-à-dire que React sait que l'état peut être retardé lors de la mise à jour. En distinguant la priorité de mise à jour, les événements de haute priorité peuvent rester réactifs, améliorant la priorité. expérience d'interaction utilisateur. Gardez les pages réactives. 🎜🎜3. startTransiton🎜🎜Introduction à l'utilisation de startTransiton🎜
const [value, setValue] = useState(&#39;&#39;)
// defferedValue值延后于state更新
const deferredValue = useDeferredValue(value, {timeoutMs: 2000})
🎜Tout d'abord , présentons le plus simplestartTransition🎜
  • startTransiton est une fonction qui accepte un rappel et est utilisée pour informer React de l'état qui doit être retardé.
  • Si une certaine mise à jour d'état entraîne le blocage du composant, elle doit être enveloppée dans startTransition
🎜Comparaison via démonstration 🎜🎜Il s'agit d'un scénario de simulation d'affichage des résultats de recherche après la saisie de caractères. En forgeant un grand nombre de résultats de recherche, il simule une situation où il est facile de se figer. 🎜🎜Nous essayons de saisir 123 en continu, de surveiller les modifications de la valeur du champ de recherche value (mise à jour urgente) et de rechercher les modifications de la valeur searchVal (mise à jour de transition) et de les afficher dans la barre de contrôle. . 🎜
import React, { useEffect, useState, useTransition, useDeferredValue } from &#39;react&#39;;
import &#39;./App.css&#39;

const SearchResult = (props) => {
    const resultList = props.query
        ? Array.from({ length: 10000 }, (_, index) => ({
            id: index,
            keyword: `${props.query} -- 搜索结果${index}`,
        })) : [];
    return resultList.map(({ id, keyword }) => (
        <li key={id}>{keyword}</li>
    ))
}

const App = () => {
    const [value, setValue] = useState(&#39;&#39;);
    const searchValue = useDeferredValue(value, { timeoutMs: 2000 });

    useEffect(() => {
        console.log(&#39;对输入框值的响应--------&#39; + value + &#39;---------------&#39;)
    }, [value])

    useEffect(() => {
        // 监听搜索值改变
        console.log(&#39;对搜索值的更新响应++++++&#39; + searchValue + &#39;+++++++++++&#39;)
    }, [searchValue])

    return (
        <div className=&#39;App&#39;>
            <input value={value} onChange={e => setValue(e.target.value)} />
        <div className={`type_button type_button_checked`}>useDeferredValue</div>
        <ul>
            <SearchResult query={searchValue}></SearchResult>
        </ul>
    </div>
    );
};
🎜Mode normal🎜🎜Une brève analyse du nouveau concept Transition dans React18🎜🎜Comme le montre l'image : Saisissez les caractères 123 en continu Lorsque le premier caractère est saisi, la valeur de recherche répond immédiatement et le rendu de la liste démarre immédiatement, provoquant. lag. La zone de saisie cesse de répondre aux entrées de l'utilisateur et ne continue pas à répondre jusqu'à la fin du rendu. 🎜🎜Après avoir utilisé startTransition🎜🎜Une brève analyse du nouveau concept Transition dans React18🎜🎜Comme le montre l'image : Saisissez les caractères 123 en continu, la zone de saisie continue de répondre et la réponse à la valeur de recherche est retardée pour garantir un retour de page jusqu'à la saisie est terminée. Commencez à répondre aux valeurs de recherche, à afficher les résultats de la recherche et à garder la page réactive. 🎜🎜4. useTransiton🎜🎜Introduction à useTransiton🎜
function useDeferredValue(value){
    const [prevValue, setValue] = updateState(value);
    updateEffect(() => {
        // 在 useEffect 中通过 transition 模式来更新 value 。
        Scheduler.unstable_next(() => {
            const prevTransition = ReactCurrentBatchConfig.transition;
            ReactCurrentBatchConfig.transition = 1;
            try {
                setValue(value);
            } finally {
                ReactCurrentBatchConfig.transition = prevTransition;
            }
         })
    }, [value]);
    return prevValue;
}
  • startTransition est une fonction qui accepte un rappel et est utilisée pour informer React de l'état qui doit être retardé.
  • isPending est une valeur booléenne, qui permet à React de nous dire s'il faut attendre la fin de la transition.
  • useTransition accepte une valeur de réponse retardée avec timeoutMs Si elle n'est pas terminée dans le timeoutMs donné, cela forcera l'exécution de startTransition. Mise à jour de l'état dans la fonction de rappel.
🎜Analyse simple de useTransiton🎜🎜Nous comprenons useTransition à travers un pseudo-code. 🎜
function useTransition(){
    const [isPending, setPending] = mountState(false);
    const start = (callback)=>{
        setPending(true);
        // Scheduler.unstable_next 通过 transiton 模式,低优先级调度执行回调函数
        // 可以降低更新的优先级。如果回调中触发的更新优先级会比较低,
        // 它会让位为高优先级的更新,或者当前事务繁忙时,调度到下一空闲期再应用。
        Scheduler.unstable_next(() => {
            const prevTransition = ReactCurrentBatchConfig.transition;
            ReactCurrentBatchConfig.transition = 1;
            try {
                setPending(false);
                //实行回调函数
                callback();
            } finally {
                ReactCurrentBatchConfig.transition = prevTransition;
            }
        })
    }
    return [isPending, start];
}

startTransition执行过程中,会触发两次setPending ,一次在transition=1之前,一次在之后。startTransition被调用时setPending(true),当startTransition内部的回调函数执行时transiton过渡任务更新setPending(false)。react内部可以根据pending值的变化准确把握等待的过渡时间,并依此判断是否超过了timeoutMs(如果有传入)强制执行更新。

5. useDeferredValue

useDeferredValue使用介绍

const [value, setValue] = useState(&#39;&#39;)
// defferedValue值延后于state更新
const deferredValue = useDeferredValue(value, {timeoutMs: 2000})
  • useDeferredValue 返回一个延迟响应的状态,可以设置最长延迟时间timeoutMs
  • 可以传入可选的timeoutMs,如果给定的timeoutMs内未完成,它将会强制更新。
  • 与useTransition的不同: useTransition是处理一段逻辑,而useDeferred是产生一个新状态

useDeferredValue的使用

import React, { useEffect, useState, useTransition, useDeferredValue } from &#39;react&#39;;
import &#39;./App.css&#39;

const SearchResult = (props) => {
    const resultList = props.query
        ? Array.from({ length: 10000 }, (_, index) => ({
            id: index,
            keyword: `${props.query} -- 搜索结果${index}`,
        })) : [];
    return resultList.map(({ id, keyword }) => (
        <li key={id}>{keyword}</li>
    ))
}

const App = () => {
    const [value, setValue] = useState(&#39;&#39;);
    const searchValue = useDeferredValue(value, { timeoutMs: 2000 });

    useEffect(() => {
        console.log(&#39;对输入框值的响应--------&#39; + value + &#39;---------------&#39;)
    }, [value])

    useEffect(() => {
        // 监听搜索值改变
        console.log(&#39;对搜索值的更新响应++++++&#39; + searchValue + &#39;+++++++++++&#39;)
    }, [searchValue])

    return (
        <div className=&#39;App&#39;>
            <input value={value} onChange={e => setValue(e.target.value)} />
        <div className={`type_button type_button_checked`}>useDeferredValue</div>
        <ul>
            <SearchResult query={searchValue}></SearchResult>
        </ul>
    </div>
    );
};

Une brève analyse du nouveau concept Transition dans React18

useDeferredValue简单分析

我们通过伪代码理解下useDeferredValue

function useDeferredValue(value){
    const [prevValue, setValue] = updateState(value);
    updateEffect(() => {
        // 在 useEffect 中通过 transition 模式来更新 value 。
        Scheduler.unstable_next(() => {
            const prevTransition = ReactCurrentBatchConfig.transition;
            ReactCurrentBatchConfig.transition = 1;
            try {
                setValue(value);
            } finally {
                ReactCurrentBatchConfig.transition = prevTransition;
            }
         })
    }, [value]);
    return prevValue;
}

useDeferredValue通过useEffect监听传入值的变化,然后通过过渡任务执行值的改变。这样保证defrredValue的更新滞后于setState,同时符合过渡更新的原则,因为是通过transition 调度机制执行的。

更多编程相关知识,请访问:编程视频!!

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Cet article est reproduit dans:. en cas de violation, veuillez contacter admin@php.cn Supprimer