کامپوننت (مولفه یا جز) های React اغلب دارای یک state یا همان حالت هستند که این state میتواند هر چیزی باشد؛ این هر چیز نیز معمولا شامل مواردی مثل لاگاین بودن یا نبودن کاربر، فعال بودن یا غیرفعال بودن او و یا حتی آرایهای از پستها میباشد.
کامپوننتهای React که دارای state هستند، موارد رابط کاربری (UI) را بر اساس مقدار فعلی state، رندر میکنند؛ در نتیجه هر گاه state یک کامپوننت نیز تغییر کند، ظاهر آن کامپوننت نیز تغییر خواهد کرد.
پس ما باید بدانیم که دقیقا در چه هنگام و به چه شکلی به تغییر مقدار state یک کامپوننت بپردازیم. در پایان این مقاله شما باید نحوهی کارکرد فانکشن setState را بیاموزید و علاوه بر آن، با برخی اشکالات رایج که در هنگام کار با stateهای ریاکتی بروز میکند، آشنا شوید.
کارکرد setState()
این فانکشن تنها راه درست برای تغییر دادن و بروز (update) کردن مقدار state در فریمورک React پس از مقدار دهی اولیه به آن است. بیایید فرض کنیم که یک کامپوننت جستوجو داریم و میخواهیم عبارت جستوجو شده توسط کاربر را به نمایش در آوریم:
import React, { Component } from 'react'
class Search extends Component {
constructor(props) {
super(props)
state = {
searchTerm: ''
}
}
}
ما در حال پاس دادن مقدار یک رشته (string) خالی به state searchTerm این کامپوننت هستیم. حال میتوان با فراخوانی فانکشن setState() به تغییر مقدار اولیهی آن پرداخت:
setState({ searchTerm: event.target.value })
همانطور که در قطعه کد بالا میبینید، در حال پاس دادن یک آبجکت به فانکشن ست استیت هستیم. این آبجکت شامل قسمتی از state است که قصد بروز کردن آن را داریم که در این مثال، searchTerm است. ریاکت این مقدار را گرفته و آن را با آبجکت state ادغام (merge) میکند. در واقع این اتفاق به مانند این است که کامپوننت search از فانکشن state انتظار دارد که مقدار searchTerm را مشخص کند.
در واقع فانکشن ست استیت شروع کنندهی پروسهای به نام reconciliation (اصلاح) است. این فرآیند باعث آپدیت شدن DOM توسط React بهوسیلهی ایجاد تغییراتی بر اساس مقدار جدید state در کامپوننتهاست. وقتی درخواست setState اعلام میشود، React یک درخت تازه که شامل المنتهای واکنشی در کامپوننت میشود را میسازد. این درخت برای تعیین تغییرات جدید لازم در کامپوننت search به کار خواهد آمد. (با مقایسهی درخت جدید با درخت قبلی)
نتیجتا React خواهد دانست که با توجه به state جدید، چه تغییراتی در کامپوننت موجود ایجاد کند تا به کامپوننت نهایی متناسب با state جدید برسد و آن را برای کاربر رندر کند؛ پس ما احتمالا در React با قابلیت setState، به هدف ایدهآلمان که ایجاد حداقل تغییرات لازم برای رسیدن به کامپوننت نهایی است، خواهیم رسید و یک وب اپ سریع و بهینه خواهیم داشت.
حال بیایید برای درک بهتر، این جریان رخ داده شده تا به اینجا را خلاصه کنیم:
- یک کامپوننت جست و جو داریم که عبارت جست و جو شده را به نمایش در میآورد.
- عبارت جست و جویی فعلا خالی است.
- کاربر یک عبارت را برای جست و جو کردن وارد میکند.
- این عبارت توسط فانکشن setState React گرفته (کپچر) و ذخیره میشود.
- فرآیند Reconciliation (اصلاح) آغاز میشود و ریاکت متوجه تغییر رخ داده شده در مقدار state کامپوننت خود میشود.
- React به کامپوننت جستوجو، دستور بروزرسانی مقدار جدید را میدهد و نهایتا عبارت جست و جو شده در کامپوننت ادغام میشود.
فرآیند Reconciliation، لزوما تمام درخت را تغییر نمیدهد؛ مگر در شرایطی که ریشهی درخت به صورت زیر تغییر کند:
// old
<div>
<Search />
</div>
// new
<span>
<Search />
</span>
در مثال بالا میبینیم که تگ div به تگ span تغییر کرده است؛ در نتیجه کل درخت کامپوننت ما آپدیت خواهد شد.
هیچگاه نباید مقدار state را به طور مستقیم تغییر داد. همیشه برای ایجاد تغییر در state از setState استفاده کنید چرا که تغییر مستقیم state، باعث رندر مجدد (re-render) کامپوننتها در ری اکت نمیشود:
// do not do this
this.state = {
searchTerm: event.target.value
}
پاس دادن یک فانکشن به setState()
برای بیان این مورد نیز اجازه دهید یک مثال بیاوریم؛ در این مثال در حال ساخت یک شمارندهی ساده هستیم که با کلیک، میتوان مقدار آن را کم یا زیاد کرد:
class App extends React.Component {
state = { count: 0 }
handleIncrement = () => {
this.setState({ count: this.state.count + 1 })
}
handleDecrement = () => {
this.setState({ count: this.state.count - 1 })
}
render() {
return (
<div>
<div>
{this.state.count}
</div>
<button onClick={this.handleIncrement}>Increment by 1</button>
<button onClick={this.handleDecrement}>Decrement by 1</button>
</div>
)
}
}
حال میتوانیم با کلیک بر روی دکمههای ساخته شده، مقدار شمارنده را، یک واحد، کم یا زیاد کنیم.
اما اگر بخواهیم با هر بار کلیک، مقدار ۳ واحد را کم یا زیاد کنیم چه؟ برای انجام این عمل میتوانیم با هربار کلیک، فانکشن setState را سه بار فراخوانی کنیم:
handleIncrement = () => {
this.setState({ count: this.state.count + 1 })
this.setState({ count: this.state.count + 1 })
this.setState({ count: this.state.count + 1 })
}
handleDecrement = () => {
this.setState({ count: this.state.count - 1 })
this.setState({ count: this.state.count - 1 })
this.setState({ count: this.state.count - 1 })
}
اما به شکلی غافلگیرانه متوجه میشویم که کد بالا مطابق انتظار ما کار نخواهد کرد. این کد معادل قطعه کد زیر است:
Object.assign(
{},
{ count: this.state.count + 1 },
{ count: this.state.count + 1 },
{ count: this.state.count + 1 },
)
Object.assign() برای کپی کردن دادهها از یک آبجکت منبع به یک آبجکت مقصد و هدف است. اگر تمام دادهها با استفاده از یک کلید مشترک، قصد کپی شدن از آبجکت مبدا به آبجکت مقصد را داشته باشند، دچار ایراد خواهیم شد. (همانند اتفاقی که در کد ما افتاد)
در زیر میتوانید یک نسخهی سادهتر از کد را ببینید تا نحوهی عملکرد object.assign() را بهتر متوجه شوید:
let count = 3
const object = Object.assign({},
{count: count + 1},
{count: count + 2},
{count: count + 3}
);
console.log(object);
// output: Object { count: 6 }
همانطور که در کد بالا میبینید، تنها آبجکت آخر که count را به علاوهی ۳ کرده اعمال شده و دو آبجک قبلی آن ندیده گرفته شدهاند.
پس در مثال اولیهی ما نیز به جای فراخوانی سهباره، تنها یک بار setState فراخوانی میشود. این مورد را میتوان با پاس دادن یک فانکشن به setState() تصحیح کرد. این کار را میتوان به همان ترتیب پاس دادن آبجکت به setState انجام داد تا در نهایت از مشکل یاد شده در بالا راحت شویم.
فانکشن handleIncrement را به شکل زیر تغییر دهید:
handleIncrement = () => {
this.setState((prevState) => ({ count: prevState.count + 1 }))
this.setState((prevState) => ({ count: prevState.count + 1 }))
this.setState((prevState) => ({ count: prevState.count + 1 }))
}
حال با این کد جدید میتوانیم با هر بار کلیک، سه واحد شمارنده را اضافه کنیم.
در این حالت جدید، React به جای ادغام، فراخوانیهای فانکشن را در صف قرار داده تا به ترتیب، یکی پس از دیگری اعمال شوند و همچنین state نیز تنها یک بار (در آخر) تغییر کند.
دسترسی به state قبلی با استفاده از یک Updater
گاهی اوقات در حین ساخت اپلیکیشن ریاکتی ممکن است نیاز داشته باشید که به state قبلی کامپوننت خود دسترسی داشته باشید و state جدید آن را بر اساس state قبلی محاسبه کنید. همیشه نمیتوان به this.state اعتماد داشت و توقع داشت بلافاصله پس از فراخوانی setState() مقدار آن نیز تصحیح شود؛ چرا که همیشه مقدار state کنونی همواره تنها مطابق با مقداری است که باعث آخرین رندر کامپوننت شده است.
بیایید مجدد به سراغ مثال شمارنده خود برویم و بررسی دقیقتری از کارکرد آن بکنیم. حال فرض کنیم که یک فانکشن داریم که مقدار count را یک واحد کم میکند. این فانکشن را در زیر مشاهده میکنید:
changeCount = () => {
this.setState({ count: this.state.count - 1})
}
حال میخواهیم قابلیت تغییر سه واحدی شمارندهی خود را داشته باشیم. فانکشن changeCount را میتوان سه بار در فانکشنی که رویداد کلیک را مدیریت و هندل میکند فراخوانی کرد:
handleDecrement = () => {
this.changeCount()
this.changeCount()
this.changeCount()
}
پس از انجام این تغییرات نیز باز هم با هر بار کلیک، تنها یک واحد تغییر را در شمارنده مشاهده میکنیم. این اتفاق به دلیل این است که مقدار this.state.count تا زمانی که کامپوننت رندر مجدد نشود، بروز نخواهد شد. راهحل این مشکل نیز استفاده از یک updater یا بروز کننده است. یک بروزکننده به شما اجازه میدهد که به state کنونی دسترسی داشته باشید و از آن بلافاصله پس از تغییر state استفاده کنید تا موارد دیگر دلخواهتان را نیز آپدیت کنید. پس فانکشن changeCount() باید به شکل زیر در آید:
changeCount = () => {
this.setState((prevState) => {
return { count: prevState.count - 1}
})
}
حال ما دیگر وابسته به نتیجهی مقدار this.state نیستیم و مقادیر count به دنبال یکدیگر در حال تغییراند. از این پس قادر به دسترسی به مقدار صحیح state که توسط setState تغییر کرده است، خواهیم بود.
به خاطر داشته باشید که باید با setState به شکل async یا ناهمزمان رفتار کنید و انتظار نداشته باشید که مقدار state شما بلافاصله پس از ست کردن آن، تغییر کند.
جمعبندی
در هنگام کار با stateها در React نیاز است که به چند نکته توجه داشته باشید:
- بروز یا آپدیت کردن state یک کامپوننت باید تنها با استفاده از setState() صورت گیرد.
- شما میتوانید به setState() یک آبجکت یا فانکشن پاس دهید.
- برای تغییر چندبارهی state به setState، فانکشن پاس دهید.
- به مقدار this.state بلافاصله پس از فراخوانی setState() اطمینان نکنید و حتما در این موارد از یک فانکشن بروز کننده استفاده کنید.
دیدگاه و پرسش
در حال دریافت نظرات از سرور، لطفا منتظر بمانید
در حال دریافت نظرات از سرور، لطفا منتظر بمانید