اندازهی قلم متن
تخمین مدت زمان مطالعهی مطلب:
چهار دقیقه
فرض کنید میخواهیم دادههایی را از دیتابیس بخوانیم و در نهایت درون UI نمایش دهیم. شاید فکر کنید که متد render مکان مناسبی برای اینکار باشد:
render() { fetch('https://jsonplaceholder.typicode.com/users') .then(response => response.json()) .then(data => this.setState({ users: data })) return ( <ul> {this.state.users.map(user => <li key={user.id}>{user.name}</li>)} </ul> ); }
اما قرار دادن فراخوانی AJAX درون متد render ایدهی خوبی نیست؛ زیرا متد render نباید side effectی ایجاد کند. هدف متد render همانطور از نامش پیداست، رندر کردن یک کامپوننت است. در نتیجه اینچنین فراخوانیهایی باید درون lifecycle events تعریف شوند که در واقع یکسری رخداد هستند و این امکان را در اختیارمان قرار میدهند که بتوانیم کارهایی از این قبیل را در طول چرخهی حیات یک کامپوننت انجام دهیم. در React به فرآیند ایجاد یک کامپوننت و در نهایت render شدن محتوای آن برای اولین بار، mounting گفته میشود. در این فرآیند سه متد نقش دارند:
- constructor
- render
- componentDidMount
سازنده یک کامپوننت زمانی فراخوانی میشود که React نیاز به ایجاد یک وهله از کامپوننت داشته باشد. در اینجا کامپوننت فرصت این را خواهد داشت تا props مورد نیازش را از والدینش دریافت کند، state dataیش را تعریف کند و یا یکسری کارهای آمادهسازی را انجام دهد. در مرحلهی بعدی، متد render فراخوانی خواهد شد. در اینجا کامپوننت، محتوایی را که قرار است به DOM اضافه شود، در اختیار React قرار میدهد. در نهایت متد componentDidMount فراخوانی خواهد شد و به این معنا است که محتوا، به DOM اضافه شدهاست. در نتیجه متد componentDidMount محل مناسبی برای قرار دادن مثال ابتدای مطلب میباشد. به عنوان مثال در کد زیر هر بار که checkbox را فعال میکنیم، یک وهله از کامپوننت ایجاد خواهد شد. یعنی در واقع هر بار از فاز mounting عبور خواهد کرد.
//App.js import React, { Component } from 'react'; import { Users } from "./Users"; export default class App extends Component { constructor(props) { super(props); this.state = { toggle: false }; } handleChange = () => { this.setState({ toggle: !this.state.toggle }); } render() { return <div className="container text-center"> <div className="row p-2"> <div className="form-check"> <input type="checkbox" className="form-check-input" checked={this.state.toggle} onChange={this.handleChange} /> <label className="form-check-label">Toggle</label> </div> <div className="row p-2"> {this.state.toggle && <Users />} </div> </div> </div> } } //Users import React, { Component } from 'react'; export class Users extends Component { constructor(props) { super(props); this.state = { users: [] } } componentDidMount() { fetch('https://jsonplaceholder.typicode.com/users') .then(response => response.json()) .then(data => { this.setState({ users: data }); console.log(data) }) } render() { return ( <div className="bg-info text-white p-2"> {this.state.users && <p>{JSON.stringify(this.state.users, null, 2)}</p> } </div> ); } }
Update phase
فرآیند پاسخگویی به تغییرات و طی شدن reconciliation، به عنوان update phase شناخته میشود. در این فاز متد render جهت دریافت محتوا از کامپوننت، فراخوانی خواهد شد و سپس متد componentDidUpdate بعد از آن فراخوانی میشود:
import React, { Component } from 'react'; export class Users extends Component { constructor(props) { // as before } componentDidMount() { // as before } componentDidUpdate() { console.log("componentDidUpdate Users Component"); } render() { // as before } }
Unmounting phase
در این فاز متد componentWillUnmount زمانی فراخوانی خواهد شد که کامپوننت در حال حذف از DOM است. طبیعتاً این متد میتواند محل خوبی برای آزاد کردن منابع، بستن اتصالات شبکه و یا متوقف کردن درخواستهای asynchronous باشد؛ برای نمونه در مثال قبل در حین غیرفعال کردن checkbox فراخوانی خواهد شد:
import React, { Component } from 'react'; export class Users extends Component { constructor(props) { // as before } componentDidMount() { // as before } componentDidUpdate() { console.log("componentDidUpdate Users Component"); } componentWillUnmount() { console.log("componentWillUnmount Users Component"); } render() { // as before } }
Effect Hook
چرخهی فوق فقط مختص به کامپوننتهایی است که به صورت کلاس تعریف شدهاند. کامپوننتهایی که به صورت تابع هستند، نمیتوانند به روش فوق باشند. با استفاده از effect hook میتوانیم مشابه متدهای componentDidMount, componentDidUpdate, componentWillUnmount را پیادهسازی کنیم:
import React, { useState, useEffect } from 'react'; export const Users = () => { const [users, setUsers] = useState([]); useEffect(() => { console.log("Same as componentDidMount") fetch('https://jsonplaceholder.typicode.com/users') .then(response => response.json()) .then(data => { setUsers({ users: data }); }) }, []); useEffect(() => { console.log("Same as componentDidUpdate") }); useEffect(() => { return () => { console.log("Same as componentWillUnmount") } }, []); return ( <div className="bg-info text-white p-2"> {users && <p>{JSON.stringify(users, null, 2)}</p> } </div> ); }