Разделение ответственности в React. Как использовать контейнерные и презентационные компоненты.
2 года назад·6 мин. на чтение
Многие новички в React объединяют логику и представление в одном и том же компоненте. И они могут не знать, почему важно разделять эти две вещи.
В таком случае может обнаружиться, что нужно внести большие изменения в файл. Затем придется вносить много переделок, чтобы разделить логику и представление.
Это происходит из-за того, что разработчик может не знать о разделении ответственности и таком шаблоне как презентационные и контейнерные компоненты (presentational and container components). В этой статье рассмотрим этот паттерн, чтобы смягчить эту проблему на ранних этапах жизненного цикла разработки проекта.
Чтобы решить эту проблему и придерживаться разделения ответственности, мы должны разделить эти две части — то есть запрос данных и их представление в пользовательском интерфейсе — на два разных компонента.
Шаблон контейнеры и презентационные компоненты (smart/dummy components) поможет нам решить эту проблему.
Вот что делает этот компонент:
Ниже приведены некоторые причины, по которым нам требуются контейнерные и презентационные компоненты:
Компонент
Файл с типами.
Компонент
Компонент
Теперь мы можем просто удалить компонент-контейнер
Что такое разделение ответственности?
Разделение ответственности — это концепция, которая широко используется в программировании. В нем говорится, что логика, выполняющая разные действия, не должна группироваться или объединяться вместе. Например, то, что мы обсуждали во вводной части, нарушает разделение задач, потому что мы поместили логику выборки данных и представления данных в один и тот же компонент.Что такое контейнерные и презентационные компоненты?
Контейнерные компоненты
Это компоненты, которые предоставляют, создают или хранят данные для дочерних компонентов. Единственная работа компонента-контейнера — обработка данных. Он не состоит из собственного пользовательского интерфейса. Скорее, он состоит из презентационных компонентов в качестве своих дочерних элементов, которые используют эти данные. Простым примером может быть компонент с именемFetchUserContainer
, который состоит из некоторой логики, которая извлекает данные всех пользователей.
Презентационные компоненты
Это компоненты, основной обязанностью которых является представление данных в пользовательском интерфейсе. Они принимают данные из компонентов контейнера. Эти компоненты не имеют состояния, если только им не требуется собственное состояние для отображения пользовательского интерфейса. Они не изменяют данные, которые они получают. Примером этого может быть компонентUserList
, который просто отображает всех пользователей.
Зачем нам нужны эти компоненты?
Чтобы понять это, возьмем простой пример. Мы хотим отобразить список сообщений, которые мы получаем из JSON placeholder API.// DisplayPosts.tsx import { useEffect, useState } from "react"; interface ISinglePost { userId: number; id: number; title: string; body: string; } /* Пример того как НЕ нужно объединять логику и отображение */ export default function DisplayPosts() { const [posts, setPosts] = useState<ISinglePost[] | null>(null); const [isLoading, setIsLoading] = useState<Boolean>(false); const [error, setError] = useState<unknown>(); // Логика useEffect(() => { (async () => { try { setIsLoading(true); const resp = await fetch("https://jsonplaceholder.typicode.com/posts"); const data = await resp.json(); setPosts(data); setIsLoading(false); } catch (err) { setError(err); setIsLoading(false); } })(); }, []); // Представление return isLoading ? ( <span>Loading... </span> ) : posts ? ( <ul> {posts.map((post: ISinglePost) => ( <li key={`item-${post.id}`}> <span>{post.title}</span> </li> ))} </ul> ) : ( <span>{JSON.stringify(error)}</span> ); }
- Он имеет 3 переменные состояния:
posts
,isLoading
иerror
. - У нас есть хук
useEffect
, который состоит из бизнес-логики. Здесь мы извлекаем данные из jsonplaceholder API с помощьюfetch
. - Когда данные извлекаются, мы сохраняем их в переменной состояния
posts
, используяsetPosts
. - Мы также гарантируем, что переключаем значения
isLoading
иerror
во время соответствующих сценариев. - Мы поместили всю эту логику в асинхронную функцию.
- Возвращаем посты в виде списка и отображаем их.
- Они помогают нам создавать слабосвязанные компоненты.
- Они помогают нам поддерживать разделение ответственности.
- Рефакторинг кода становится намного проще.
- Код становится более организованным и удобным в сопровождении
- Это значительно упрощает тестирование.
Пример компонента-представления и контейнера
Для примера будем использовать тот же пример, что и выше — получение данных из JSON placeholder API. Разберемся со структурой файлов. Нашим контейнерным компонентом будетPostContainer
. У нас будет два презентационных компонента:
Posts
: компонент с неупорядоченным списком.SinglePost
: компонент, отображающий элемент списка.
components
. Теперь, когда мы знаем, что куда помещать, давайте начнем с компонента-контейнера: PostContainer
.
Компонент PostContainer
// components/PostContainer.tsx import { useEffect, useState } from "react"; import { ISinglePost } from "../Definitions"; import Posts from "./Posts"; export default function PostContainer() { const [posts, setPosts] = useState<ISinglePost[] | null>(null); const [isLoading, setIsLoading] = useState<Boolean>(false); const [error, setError] = useState<unknown>(); useEffect(() => { (async () => { try { setIsLoading(true); const resp = await fetch("https://jsonplaceholder.typicode.com/posts"); const data = await resp.json(); setPosts(data.filter((post: ISinglePost) => post.userId === 1)); setIsLoading(false); } catch (err) { setError(err); setIsLoading(false); } })(); }, []); return isLoading ? ( <span>Loading... </span> ) : posts ? ( <Posts posts={posts} /> ) : ( <span>{JSON.stringify(error)}</span> ); }
Приведенный выше код просто содержит логику выборки данных. Эта логика присутствует в хукеcomponents/Definitions.ts export interface SinglePost { userId: number; id: number; title: string; body: string; }
useEffect
. Этот компонент-контейнер передает эти данные презентационному компоненту Posts
.
Давайте взглянем на презентационный компонент Posts
.
Компонент Posts
Как видите, это простой файл, состоящий из тега// components/Posts.tsx import { ISinglePost } from "../Definitions"; import SinglePost from "./SinglePost"; export default function Posts(props: { posts: ISinglePost[] }) { return ( <ul style={{ display: "flex", flexDirection: "column", alignItems: "center" }} > {props.posts.map((post: ISinglePost) => ( <SinglePost {...post} /> ))} </ul> ); }
ul
— неупорядоченного списка. Затем этот компонент отображает посты, которые передаются в качестве пропса. Мы передаем каждый объект поста в компонент SinglePost
.
Существует еще один презентационный компонент, который отображает элемент списка, это тег li
. Он отображает заголовок и тело сообщения.
Компонент SinglePost
Эти презентационные компоненты просто отображают данные на экране. Вот и все. Они не делают ничего другого. Поскольку здесь они просто отображают данные, они также будут иметь собственные стили. Теперь, когда мы настроили компоненты, давайте посмотрим, что удалось достичь:// components/SinglePost.tsx import { ISinglePost } from "../Definitions"; export default function SinglePost(props: ISinglePost) { const { userId, id, title, body } = props; return ( <li key={`item-${userId}-${id}`} style={{ width: 400 }}> <h4> <strong>{title}</strong> </h4> <span>{body}</span> </li> ); }
- Концепция разделения ответственности в этом примере не нарушается.
- Написание модульных тестов для каждого компонента становится проще.
- Сопровождаемость и читабельность кода намного лучше. Таким образом, наша кодовая база стала намного более организованной.
Как заменить контейнерные компоненты на React хуки
Начиная с React 16.8 стало намного проще создавать и разрабатывать компоненты с помощью функциональных компонентов и хуков. Здесь мы воспользуемся этими возможностями и заменим компонент-контейнер хуком.Что дает это улучшение:// hooks/usePosts.ts import { useEffect, useState } from "react"; import { ISinglePost } from "../Definitions"; export default function usePosts() { const [posts, setPosts] = useState<ISinglePost[] | null>(null); const [isLoading, setIsLoading] = useState<Boolean>(false); const [error, setError] = useState<unknown>(); useEffect(() => { (async () => { try { setIsLoading(true); const resp = await fetch("https://jsonplaceholder.typicode.com/posts"); const data = await resp.json(); setPosts(data.filter((post: ISinglePost) => post.userId === 1)); setIsLoading(false); } catch (err) { setError(err); setIsLoading(false); } })(); }, []); return { isLoading, posts, error }; }
- Извлечена логика, которая присутствовала в компоненте
PostContainer
, в хук. - Этот хук вернет объект, содержащий значения
isLoading
,posts
иerror
.
PostContainer
. Затем, вместо того, чтобы передавать данные контейнера презентационным компонентам в качестве пропса, мы можем напрямую использовать этот хук внутри презентационного компонента Posts
.
Внесем следующие изменения в компонент Posts
.
Используя хуки, мы устранили дополнительный слой компонента, который присутствовал поверх этих презентационных компонентов. С хуками мы достигли тех же результатов, что и с шаблоном контейнерные/презентационные компоненты.// components/Posts.tsx import { ISinglePost } from "../Definitions"; import usePosts from "../hooks/usePosts"; import SinglePost from "./SinglePost"; export default function Posts(props: { posts: ISinglePost[] }) { const { isLoading, posts, error } = usePosts(); return ( <ul style={{ display: "flex", flexDirection: "column", alignItems: "center" }} > {isLoading ? ( <span>Loading...</span> ) : posts ? ( posts.map((post: ISinglePost) => <SinglePost {...post} />) ) : ( <span>{JSON.stringify(error)}</span> )} </ul> ); }
Итоги
Итак, в этой статье мы рассмотрели:- Разделение ответственности.
- Контейнерные и презентационные компоненты
- Зачем нам нужны эти компоненты
- Как хуки могут заменить компоненты-контейнеры
Как вызвать метод дочернего компонента из родительского компонента с помощью useImperativeHandle
2 года назад·3 мин. на чтение
Быстрый старт с useImperativeHandle
В этой статье будет показано, как вызвать метод дочернего компонента с помощью ссылки. Чтобы решить эту проблему, мы будем использовать хуки
Хук
Теперь давайте сосредоточимся на нашей задаче. Мы хотим вызвать метод (
На этом этапе мы можем создать ссылку в родительском компоненте с помощью хука
В родительский компонент нам нужно импортировать этот
useRef
и useImperativeHandle
.
Дочерний компонент
Начнем с простого дочернего компонента, в котором содержится кнопка. Нажатие на кнопку вызывает внутренний методdoSomething
.
// Child.jsx function Child(props, ref) { const doSomething = () => { console.log("do something"); }; return ( <div> <h1>Child Component</h1> <button onClick={doSomething}>Run</button> </div> ); } export default Child;
Родительский компонент
Далее рассмотрим родительский компонент. В нем используется дочерний компонент, описанный выше. Обратите внимание, что в родительском компоненте есть собственная кнопка сохранения.// App.jsx import Child from "./Child"; function App() { const save = () => {}; return ( <div> <Child /> <button onClick={save}>Save</button> </div> ); } export default App;
Хук useImperativeHandle
Теперь давайте сосредоточимся на нашей задаче. Мы хотим вызвать метод (doSomething
) дочернего компонента при нажатии кнопки (Save
) из родительского компонента.
Чтобы вызвать метод из дочернего компонента, нам нужно сначала выставить его наружу.
useImperativeHandle
определяет значение объекта, которое предоставляется родительскому компоненту при использовании ref
. Добавляя наш метод к этому объекту, мы делаем его доступным в родительских компонентах.
// Child.jsx import { useImperativeHandle } from "react"; function Child(props, ref) { const doSomething = () => { console.log("do something"); }; useImperativeHandle(ref, () => ({ doSomething })); return ( <div> <h1>Child Component</h1> <button onClick={doSomething}>Run</button> </div> ); } export default Child;
useImperativeHandle
следует использовать с forwardRef
.
forwardRef
позволяет родительскому компоненту передавать ссылки своим дочерним элементам. Чтобы прикрепить функции или поля к этой ссылке (к рефу), используется хук useImperativeHandle
.
// Child.jsx import { forwardRef, useImperativeHandle } from "react"; function Child(props, ref) { const doSomething = () => { console.log("do something"); }; useImperativeHandle(ref, () => ({ doSomething })); return ( <div> <h1>Child Component</h1> <button onClick={doSomething}>Run</button> </div> ); } export default forwardRef(Child); // Child обернут в forwardRef
useRef
и передать ее дочернему компоненту. Получив эту ссылку, мы можем вызвать метод doSomething
дочернего компонента.
// App.jsx import { useRef } from "react"; import Child from "./Child"; function App() { const childRef = useRef(null); const save = () => { if (childRef.current) { childRef.current.doSomething(); } }; return ( <div> <Child ref={childRef} /> <button onClick={save}>Save</button> </div> ); } export default App;
Добавим TypeScript
Далее посмотрим, какие изменения нужно сделать, чтобы вызвать тот же дочерний метод из родительского компонента при использовании TypeScript. Во-первых, нам нужно определить новый интерфейс, содержащий метод, который будет представлен.Затем новый тип (export interface RefType { doSomething: () => void; }
RefType
) используется при получении ссылки в дочернем компоненте.
Ниже приведен полный код дочернего компонента.function Child(props: PropsType, ref: Ref<RefType>)
// Child.jsx import { forwardRef, useImperativeHandle, Ref } from "react"; export interface PropsType {} export interface RefType { doSomething: () => void; } function Child(props: PropsType, ref: Ref<RefType>) { const doSomething = () => { console.log("do something"); }; useImperativeHandle(ref, () => ({ doSomething })); return ( <div> <h1>Child Component</h1> <button onClick={doSomething}>Run</button> </div> ); } export default forwardRef(Child);
RefType
, содержащий все публичные дочерние методы, и использовать его при создании ref
.
Полный код родительского компонента.// App.jsx import Child, { RefType } from "./Child"; //... const childRef = useRef<RefType>(null);
import { useRef } from "react"; import Child, { RefType } from "./Child"; function App() { const childRef = useRef<RefType>(null); const save = () => { if (childRef.current) { childRef.current.doSomething(); } }; return ( <div> <Child ref={childRef} /> <button onClick={save}>Save</button> </div> ); } export default App;