021-88881776

آموزش موضوعات خاص (Special Topics) React

آموزش React یکی از مهم‌ترین نیازهای توسعه‌دهندگانی است که به دنبال ایجاد تجربه‌های کاربری پیشرفته و کاربردی هستند. React به عنوان یکی از محبوب‌ترین کتابخانه‌های جاوااسکریپت، امکانات متعددی را برای ساخت رابط‌های کاربری پویا و کارآمد در اختیار ما قرار می‌دهد. در این مقاله، با موضوعات خاص (Special Topics) React آشنا می‌شویم و به بررسی مواردی از سطح مبتدی تا پیشرفته می‌پردازیم. این موضوعات شامل آشنایی با TypeScript در React، استفاده از React بدون JSX، تست‌ها در React، کتابخانه React Router، ردیابی پروفایل و تحلیل عملکرد، و استفاده از ابزارهای DevTools است. این مقاله با زبانی ساده و قابل فهم برای همه سطوح دانش، از مبتدیان تا توسعه‌دهندگان پیشرفته تهیه شده و برای درک بهتر مطالب، از مثال‌های عملی و توضیحات واضح استفاده شده است.

آشنایی با TypeScript در React

TypeScript یک زبان بالادستی برای جاوااسکریپت است که به برنامه‌نویسان اجازه می‌دهد تا از تایپ‌های استاتیک استفاده کنند. این ویژگی به ویژه در پروژه‌های بزرگ مفید است و کمک می‌کند تا خطاهای مربوط به نوع داده‌ها در زمان کامپایل کشف شوند. در پروژه‌های React، استفاده از TypeScript می‌تواند باعث افزایش دقت و خوانایی کد، خودکارسازی کدنویسی، و بهبود مستندسازی شود. این امر به توسعه‌دهندگان کمک می‌کند تا با اطمینان بیشتری کامپوننت‌های React را بسازند و از مشکلات ناشی از عدم سازگاری نوع داده‌ها جلوگیری کنند.

نصب و راه‌اندازی TypeScript در پروژه‌های React

برای استفاده از TypeScript در پروژه‌های React، اگر پروژه جدیدی را آغاز می‌کنید، می‌توانید آن را مستقیماً با تنظیمات TypeScript ایجاد کنید:

npx create-react-app my-app --template typescript

اما اگر در حال حاضر یک پروژه React با جاوااسکریپت دارید و می‌خواهید TypeScript را به آن اضافه کنید، کافی است TypeScript و کتابخانه‌های مرتبط را نصب کنید:

npm install typescript @types/react @types/react-dom

بعد از نصب، فایل‌های .js خود را به .tsx تغییر دهید و در صورت نیاز، فایل tsconfig.json را نیز برای تنظیمات بیشتر ایجاد کنید.

تعریف انواع داده‌ها (Types) و اینترفیس‌ها (Interfaces) در کامپوننت‌ها

یکی از کاربردهای اصلی TypeScript در React، امکان تعریف انواع داده‌ها و اینترفیس‌ها برای مشخص کردن ساختار داده‌های ورودی به کامپوننت‌هاست. این کار باعث می‌شود تا داده‌های ورودی و خروجی دقیق‌تر کنترل شوند. از اینترفیس‌ها برای تعریف نوع props کامپوننت‌ها استفاده می‌شود و از انواع (Types) نیز برای متغیرها، حالت‌ها (state)، و تابع‌ها بهره گرفته می‌شود.

مثال عملی با توضیحات
فرض کنید قصد داریم یک کامپوننت با نام User بسازیم که اطلاعات یک کاربر شامل نام و سن او را نمایش دهد. برای این کار، ابتدا یک اینترفیس به نام UserProps تعریف می‌کنیم تا ساختار داده‌های props مشخص شود:

interface UserProps {
  name: string;
  age: number;
}

در اینجا، UserProps نوع props مورد انتظار کامپوننت User را تعیین می‌کند و مشخص می‌سازد که name از نوع string و age از نوع number است. سپس این اینترفیس را به عنوان props در کامپوننت User به کار می‌بریم:

const User: React.FC<UserProps> = ({ name, age }) => {
  return (
    <div>
      <h1>{name}</h1>
      <p>{age} years old</p>
    </div>
  );
};

در این مثال:

از React.FC<UserProps> برای تعریف کامپوننت به عنوان یک کامپوننت تابعی با تایپ مشخص استفاده شده است. این روش به TypeScript می‌گوید که props این کامپوننت باید با ساختار UserProps سازگار باشد.
در صورت تلاش برای ارسال propsی متفاوت از UserProps، TypeScript خطا می‌دهد و از بروز خطاهای ناگهانی جلوگیری می‌کند.

استفاده از تایپ‌ها در State و Hook‌ها

TypeScript به ما این امکان را می‌دهد که نوع state و داده‌های مرتبط با hookها را نیز مشخص کنیم. فرض کنید می‌خواهیم یک کامپوننت بسازیم که لیستی از اسامی را نمایش دهد و بتوانیم نام‌ها را به لیست اضافه کنیم:

const NameList: React.FC = () => {
  const [names, setNames] = useState<string[]>([]);

  const addName = (name: string) => {
    setNames([...names, name]);
  };

  return (
    <div>
      {names.map((name, index) => (
        <p key={index}>{name}</p>
      ))}
      <button onClick={() => addName("New Name")}>Add Name</button>
    </div>
  );
};

در این مثال:

useState<string[]> به TypeScript می‌گوید که names یک آرایه از string است و setNames فقط باید داده‌های از نوع string[] دریافت کند.
تابع addName تنها نام‌های جدید از نوع string را می‌پذیرد.
این کار تایپ دقیق و ایمنی بیشتری را فراهم می‌کند، به طوری که اگر به طور تصادفی داده‌ای از نوع دیگری مانند number را به names اضافه کنیم، TypeScript خطا می‌دهد.

مزایای استفاده از TypeScript در React

کاهش خطاها: با کمک تایپ‌های استاتیک، خطاهای مربوط به نوع داده‌ها سریع‌تر شناسایی می‌شوند.
افزایش خوانایی کد: استفاده از تایپ‌ها و اینترفیس‌ها باعث می‌شود که ساختار props و state کامپوننت‌ها واضح‌تر و قابل درک‌تر باشند.
بهبود مستندسازی: با TypeScript، مشخصات کامپوننت‌ها و نوع داده‌ها به وضوح مستند می‌شود که توسعه‌دهندگان دیگر نیز به راحتی می‌توانند آن را درک کنند.
پشتیبانی بهتر از ابزارها: ابزارهای کدنویسی مانند Visual Studio Code با پشتیبانی از TypeScript، پیشنهادات کد (autocomplete) و بررسی خطاهای زودهنگام را بهبود می‌بخشند.
استفاده از TypeScript در پروژه‌های React به شما کمک می‌کند تا کامپوننت‌های دقیق‌تر، خواناتر و امن‌تری بسازید و پروژه‌های پیچیده را با اطمینان بیشتری مدیریت کنید.

استفاده از React بدون JSX

JSX (JavaScript XML) یکی از ویژگی‌های شاخص و پرطرفدار React است که به شما امکان می‌دهد تا با ترکیب جاوااسکریپت و HTML به روشی ساده و خوانا، کامپوننت‌ها را بنویسید. استفاده از JSX کد React را بسیار شبیه به HTML می‌کند و درک و توسعه رابط‌های کاربری را برای توسعه‌دهندگان ساده‌تر می‌سازد. با این حال، استفاده از React بدون JSX نیز امکان‌پذیر است، که برای موارد خاص مانند پیکربندی‌های سروری یا پروژه‌هایی که نیازمند استفاده از جاوااسکریپت خالص هستند، مفید است.

زمانی که از React بدون JSX استفاده می‌کنید، به جای اینکه از ساختارهای شبیه به HTML استفاده کنید، باید به صورت دستی عناصر را با استفاده از متد React.createElement ایجاد کنید. این متد، یک المان React را با تعریف نوع المان (مانند div یا h1)، خصوصیات آن، و محتوای داخلی آن می‌سازد.

مزایای استفاده از React بدون JSX

کاهش پیچیدگی‌های باندلینگ (Bundling): استفاده از JSX نیاز به کامپایلرهایی مانند Babel دارد تا JSX را به جاوااسکریپت خالص تبدیل کند. در برخی پروژه‌های کوچک یا شرایط خاص، کار کردن با جاوااسکریپت خالص و بدون JSX می‌تواند کار را ساده‌تر کند و نیاز به باندلینگ و کامپایل کمتر شود.

سازگاری بهتر با محیط‌های سروری و قدیمی: در پروژه‌هایی که برای اجرا در محیط‌های قدیمی‌تر تنظیم شده‌اند یا نیاز به پیکربندی ویژه سروری دارند، کدنویسی بدون JSX و با جاوااسکریپت خالص می‌تواند راحت‌تر باشد.

کنترل بیشتر در تولید عناصر: با استفاده از React.createElement، توسعه‌دهنده به طور مستقیم و دقیق‌تر می‌تواند نحوه ساخت عناصر را کنترل کند و حتی ساختارهای پیچیده‌تر را با منطق بیشتری ایجاد کند.

نحوه استفاده از React بدون JSX

برای تعریف کامپوننت‌ها و عناصر React بدون JSX، به جای استفاده از سینتکس شبیه به HTML، می‌توان از متد React.createElement بهره برد. این متد سه ورودی دریافت می‌کند:

نوع المان (به عنوان مثال div, h1, button)
خصوصیات و ویژگی‌های المان (مثل className، id، style)
فرزندان (children) یا محتوای داخلی المان

مثال‌های عملی

مقایسه بین JSX و جاوااسکریپت خالص

در حالت معمول با JSX، کد یک عنصر به این شکل است:

const element = <h1>Hello, world!</h1>;

در حالت بدون JSX، همین کد با استفاده از React.createElement به این صورت نوشته می‌شود:

const element = React.createElement('h1', null, 'Hello, world!');

در این مثال، ‘h1’ نوع المان است، null به معنای عدم وجود خصوصیات خاص برای المان است، و ‘Hello, world!’ محتوای داخل المان را مشخص می‌کند.

تعریف کامپوننت‌های پیچیده‌تر

مثلاً فرض کنید یک کامپوننت ساده به نام Card داریم که شامل یک عنوان و توضیحات است. کد این کامپوننت با JSX به شکل زیر خواهد بود:

const Card = () => (
  <div className="card">
    <h2>Card Title</h2>
    <p>This is a description.</p>
  </div>
);

حال، اگر بخواهیم همین کامپوننت را بدون JSX بنویسیم، از React.createElement به صورت زیر استفاده می‌کنیم:

const Card = () => {
  return React.createElement(
    'div',
    { className: 'card' },
    React.createElement('h2', null, 'Card Title'),
    React.createElement('p', null, 'This is a description.')
  );
};

در این مثال:

برای ایجاد المان div، از React.createElement استفاده کردیم و className: ‘card’ را به عنوان خصوصیت آن تنظیم کردیم.
سپس به صورت سلسله‌مراتبی، المان‌های h2 و p را به عنوان فرزندان div مشخص کردیم.

استفاده از آرایه‌ها برای مدیریت چندین عنصر

یکی از چالش‌های کار با React.createElement در کامپوننت‌های پیچیده، مدیریت چندین فرزند برای یک المان است. با استفاده از آرایه‌ها، می‌توان چندین المان را درون یک کامپوننت بدون نیاز به JSX تولید کرد.

مثال: فرض کنید قصد داریم لیستی از آیتم‌ها را نمایش دهیم:

const List = () => {
  return React.createElement(
    'ul',
    null,
    ['Item 1', 'Item 2', 'Item 3'].map((item, index) =>
      React.createElement('li', { key: index }, item)
    )
  );
};

در اینجا:

از map برای ایجاد چندین li استفاده کردیم و هر li را به عنوان فرزند ul ایجاد کردیم.
ویژگی key برای هر آیتم تعیین شده تا React بتواند آیتم‌ها را به درستی شناسایی و مدیریت کند.
استفاده از React بدون JSX اگرچه کدنویسی را طولانی‌تر می‌کند، اما کنترل بیشتری را بر ساختار و پیکربندی کامپوننت‌ها فراهم می‌سازد. این روش به ویژه در محیط‌های خاص، پروژه‌های ساده یا در مواردی که نیاز به استفاده از جاوااسکریپت خالص داریم، مفید است. با وجود آنکه JSX روش توصیه‌شده و محبوب برای کدنویسی در React است، آشنایی با روش جاوااسکریپت خالص و React.createElement می‌تواند شما را به درک عمیق‌تری از نحوه عملکرد React برساند و ابزار دیگری را برای موقعیت‌های خاص در اختیار شما قرار دهد.

تست‌ها در React

نوشتن تست‌ها یکی از مراحل حیاتی در توسعه نرم‌افزار است که به ما اطمینان می‌دهد کامپوننت‌ها همان‌طور که انتظار داریم عمل می‌کنند و تغییرات آینده باعث خراب شدن رفتار فعلی نمی‌شوند. در React، ابزارهای مختلفی برای تست کامپوننت‌ها وجود دارند، و دو ابزار محبوب برای این کار Jest و React Testing Library هستند. Jest، که توسط فیسبوک توسعه داده شده، یک فریمورک تست قدرتمند برای جاوااسکریپت است که شامل ابزارهایی برای نوشتن، اجرا و مدیریت تست‌ها می‌باشد. React Testing Library نیز به صورت خاص برای تست کامپوننت‌های React طراحی شده و به ما کمک می‌کند تا تست‌هایی نزدیک به تجربه کاربری واقعی بنویسیم.

آشنایی با Jest و راه‌اندازی آن

Jest یکی از فریمورک‌های تست قدرتمند برای جاوااسکریپت است که شامل ویژگی‌هایی مانند اجرای تست‌های همزمان، ایجاد mocks و پشتیبانی از snapshot testing می‌باشد. Jest معمولاً به صورت پیش‌فرض در پروژه‌هایی که با create-react-app ساخته می‌شوند، نصب می‌شود. با این حال، اگر از create-react-app استفاده نکرده‌اید، می‌توانید Jest را با استفاده از دستور زیر نصب کنید:

npm install --save-dev jest

برای اجرای تست‌ها، کافی است در خط فرمان دستور npm test را اجرا کنید. Jest به صورت خودکار تمام فایل‌های تست (فایل‌هایی که با .test.js یا .spec.js خاتمه می‌یابند) را شناسایی کرده و اجرا می‌کند.

آشنایی با React Testing Library

React Testing Library یک کتابخانه مکمل برای تست کامپوننت‌های React است که به صورت تخصصی روی تعامل کاربر با کامپوننت‌ها تمرکز دارد. این کتابخانه تلاش می‌کند تست‌هایی نزدیک به تجربه کاربر نهایی ایجاد کند، به گونه‌ای که توسعه‌دهنده به جای تمرکز بر جزئیات پیاده‌سازی داخلی، بیشتر بر نحوه عملکرد و نمایش کامپوننت‌ها در مرورگر تمرکز کند. برای نصب آن، می‌توانید از دستور زیر استفاده کنید:

npm install --save-dev @testing-library/react

نوشتن تست‌های ساده برای کامپوننت‌ها

در اینجا یک مثال از نحوه نوشتن تست برای یک کامپوننت ساده React را با استفاده از Jest و React Testing Library بررسی می‌کنیم.

مثال: تست یک دکمه با رویداد کلیک
فرض کنید یک کامپوننت Button داریم که با کلیک روی آن، یک تابع handleClick اجرا می‌شود:

import React from 'react';

const Button = ({ onClick, children }) => (
  <button onClick={onClick}>{children}</button>
);

export default Button;

در این کامپوننت، onClick به عنوان prop به دکمه ارسال شده است و children محتوای داخلی دکمه را تعیین می‌کند. حال می‌خواهیم اطمینان حاصل کنیم که با کلیک روی دکمه، تابع handleClick فراخوانی می‌شود. برای این منظور، از Jest و React Testing Library استفاده می‌کنیم:

import { render, fireEvent } from '@testing-library/react';
import Button from './Button';

test('button click', () => {
  const handleClick = jest.fn(); // یک mock برای تابع onClick ایجاد می‌کنیم
  const { getByText } = render(<Button onClick={handleClick}>Click me</Button>);

  // شبیه‌سازی کلیک روی دکمه
  fireEvent.click(getByText(/click me/i));

  // بررسی اینکه تابع handleClick یک بار فراخوانی شده است
  expect(handleClick).toHaveBeenCalledTimes(1);
});

 

توضیحات مثال:
ابتدا با jest.fn() یک mock function برای handleClick ایجاد کرده‌ایم. این تابع شبیه‌سازی‌شده به ما کمک می‌کند تا بررسی کنیم که آیا رویداد کلیک دکمه به درستی فراخوانی شده است یا خیر.
سپس با render، کامپوننت Button را رندر کرده و محتوای دکمه را به صورت Click me تنظیم کرده‌ایم.
برای شبیه‌سازی کلیک روی دکمه، از fireEvent.click استفاده می‌کنیم و با getByText(/click me/i)، دکمه‌ای را که حاوی متن “Click me” است، شناسایی می‌کنیم.
در نهایت، با expect(handleClick).toHaveBeenCalledTimes(1) بررسی می‌کنیم که handleClick تنها یک بار فراخوانی شده باشد.

تست کامپوننت‌ها با state و props متفاوت

تست کامپوننت‌هایی که دارای state یا props متغیر هستند، اهمیت زیادی دارد تا مطمئن شویم که با تغییر ورودی‌ها، خروجی‌ها نیز به درستی تغییر می‌کنند.

مثال: فرض کنید یک کامپوننت Counter داریم که دارای یک دکمه و یک شمارنده است که با هر بار کلیک مقدار شمارنده افزایش می‌یابد:

import React, { useState } from 'react';

const Counter = () => {
  const [count, setCount] = useState(0);

  return (
    <div>
      <p>Count: {count}</p>
      <button onClick={() => setCount(count + 1)}>Increment</button>
    </div>
  );
};

export default Counter;

برای تست این کامپوننت، ابتدا بررسی می‌کنیم که مقدار اولیه شمارنده 0 باشد و با کلیک بر روی دکمه، مقدار آن به درستی افزایش یابد.

import { render, fireEvent } from '@testing-library/react';
import Counter from './Counter';

test('increments counter', () => {
  const { getByText } = render(<Counter />);

  // بررسی مقدار اولیه
  expect(getByText(/count:/i)).toHaveTextContent('Count: 0');

  // کلیک روی دکمه
  fireEvent.click(getByText(/increment/i));

  // بررسی مقدار پس از کلیک
  expect(getByText(/count:/i)).toHaveTextContent('Count: 1');
});

توضیحات این تست:
ابتدا با getByText(/count:/i) بررسی می‌کنیم که مقدار شمارنده در ابتدا برابر با Count: 0 باشد.
سپس با fireEvent.click(getByText(/increment/i)) کلیک روی دکمه را شبیه‌سازی می‌کنیم.
پس از کلیک، مقدار شمارنده باید به Count: 1 تغییر کند و این تغییر را با expect(…).toHaveTextContent بررسی می‌کنیم.

مزایای تست در React

اطمینان از عملکرد صحیح کامپوننت‌ها: با نوشتن تست‌ها، اطمینان حاصل می‌کنیم که کامپوننت‌ها به درستی عمل می‌کنند و خطاهای احتمالی زودتر شناسایی می‌شوند.
توسعه امن‌تر: با وجود تست‌های جامع، می‌توانیم با خیال راحت تغییرات جدیدی اعمال کنیم و مطمئن باشیم که تغییرات باعث خراب شدن بخش‌های دیگر نمی‌شود.
مستندسازی کد: تست‌ها به نوعی مستندات زنده هستند که نحوه عملکرد کد را توضیح می‌دهند و به تیم توسعه در درک سریع‌تر کد کمک می‌کنند.
با استفاده از Jest و React Testing Library، می‌توانید به سادگی تست‌های دقیق و موثری برای کامپوننت‌های React بنویسید و از کیفیت و پایداری نرم‌افزار خود اطمینان حاصل کنید.

کتابخانه React Router

React Router یک کتابخانه قدرتمند و محبوب برای مدیریت مسیرها (Routing) در برنامه‌های React است. مسیریابی یا Routing به معنای تعریف صفحات و انتقال بین آن‌ها در یک برنامه تک‌صفحه‌ای (SPA) است. با استفاده از React Router، می‌توانیم صفحات مختلفی ایجاد کنیم و بین آن‌ها بدون بارگذاری مجدد صفحه، به شکلی روان و کاربرپسند انتقال یابیم. این قابلیت به ویژه برای توسعه‌دهندگانی که می‌خواهند تجربه کاربری بهتری را ایجاد کنند، بسیار مهم است.

نصب و راه‌اندازی React Router

برای شروع استفاده از React Router، باید آن را در پروژه خود نصب کنید. می‌توانید با استفاده از دستور زیر React Router را نصب کنید:

npm install react-router-dom

پس از نصب، می‌توانید از اجزای اصلی آن شامل BrowserRouter, Routes, و Route در پروژه خود استفاده کنید.

اجزای اصلی React Router

BrowserRouter: این کامپوننت اصلی‌ترین بخش React Router است که قابلیت مسیریابی را در برنامه فراهم می‌کند. باید کل برنامه درون آن قرار گیرد.

Routes: این کامپوننت شامل مجموعه‌ای از Routeها است و نقش محفظه‌ای برای تعریف مسیرهای مختلف را ایفا می‌کند.

Route: برای تعریف هر مسیر به صورت جداگانه استفاده می‌شود و دو ویژگی اصلی path (مسیر آدرس‌دهی) و element (کامپوننتی که باید در این مسیر رندر شود) را دارد.

پیکربندی اولیه با مثال

برای شروع، می‌توانیم یک برنامه ساده را با دو صفحه “خانه” و “درباره” تعریف کنیم.

import { BrowserRouter as Router, Route, Routes } from 'react-router-dom';
import Home from './Home';
import About from './About';

function App() {
  return (
    <Router>
      <Routes>
        <Route path="/" element={<Home />} />
        <Route path="/about" element={<About />} />
      </Routes>
    </Router>
  );
}

export default App;

در این مثال:

BrowserRouter به عنوان محفظه‌ای برای تمامی مسیرها استفاده شده است و این امکان را فراهم می‌کند که در برنامه به صورت تک‌صفحه‌ای جابه‌جا شویم.
Routes به عنوان ظرفی برای Routeها عمل می‌کند و مسیرها را مدیریت می‌کند.
Route با ویژگی path=”/” صفحه اصلی (Home) را نشان می‌دهد و مسیر path=”/about” نیز صفحه درباره (About) را مشخص می‌کند.

مسیریابی داینامیک (Dynamic Routing)

گاهی نیاز داریم که مسیرهای داینامیک، مثلاً برای نمایش پروفایل کاربر یا محتوای خاص، تعریف کنیم. برای این کار می‌توان از پارامترهای داینامیک استفاده کرد. به عنوان مثال، فرض کنید می‌خواهیم صفحه‌ای داشته باشیم که اطلاعات کاربر با id خاص را نمایش دهد.

import { BrowserRouter as Router, Route, Routes } from 'react-router-dom';
import Home from './Home';
import UserProfile from './UserProfile';

function App() {
  return (
    <Router>
      <Routes>
        <Route path="/" element={<Home />} />
        <Route path="/user/:id" element={<UserProfile />} />
      </Routes>
    </Router>
  );
}

export default App;

در اینجا:

:id یک پارامتر داینامیک است و می‌تواند هر مقداری بگیرد. مثلاً مسیر /user/1 یا /user/42.
در داخل کامپوننت UserProfile، می‌توانیم از useParams (یکی از hookهای React Router) برای دسترسی به id استفاده کنیم:

import { useParams } from 'react-router-dom';

function UserProfile() {
  const { id } = useParams();
  return <h2>User ID: {id}</h2>;
}

 

مسیریابی شرطی (Conditional Routing)

گاهی ممکن است بخواهیم به کاربرانی که وارد سیستم نشده‌اند، اجازه دسترسی به برخی صفحات را ندهیم. در این حالت می‌توانیم مسیریابی شرطی را پیاده‌سازی کنیم. فرض کنید یک صفحه پروفایل داریم که فقط در صورت ورود کاربر به سیستم نمایش داده شود.

import { BrowserRouter as Router, Route, Routes, Navigate } from 'react-router-dom';
import Home from './Home';
import Profile from './Profile';

function App() {
  const isAuthenticated = false; // به عنوان مثال فرض می‌کنیم کاربر وارد نشده است.

  return (
    <Router>
      <Routes>
        <Route path="/" element={<Home />} />
        <Route
          path="/profile"
          element={isAuthenticated ? <Profile /> : <Navigate to="/" />}
        />
      </Routes>
    </Router>
  );
}

export default App;

در اینجا:

اگر isAuthenticated برابر با false باشد، به کاربر اجازه ورود به مسیر /profile داده نمی‌شود و او را به صفحه اصلی (/) هدایت می‌کنیم.
Navigate یک کامپوننت خاص در React Router است که برای هدایت کاربران به مسیر دیگری استفاده می‌شود.

استفاده از لینک‌ها برای جابه‌جایی بین صفحات

برای جابه‌جایی بین صفحات و مسیرها به طور امن و بدون بارگذاری مجدد صفحه، از کامپوننت Link استفاده می‌کنیم:

import { Link } from 'react-router-dom';

function Navbar() {
  return (
    <nav>
      <Link to="/">Home</Link>
      <Link to="/about">About</Link>
      <Link to="/profile">Profile</Link>
    </nav>
  );
}

در اینجا:

Link به طور مستقیم مسیرها را به روزرسانی می‌کند و از بارگذاری مجدد صفحه جلوگیری می‌کند.
to ویژگی‌ای است که مقصد لینک را مشخص می‌کند.

مزایای استفاده از React Router

ایجاد تجربه کاربری بهتر: React Router به شما این امکان را می‌دهد که یک تجربه روان‌تر و سریع‌تر برای کاربران ایجاد کنید، چرا که نیازی به بارگذاری مجدد کل صفحه نیست.

مدیریت ساده‌تر صفحات مختلف: در برنامه‌های پیچیده و بزرگ، با React Router می‌توانیم به سادگی صفحات مختلف و انتقال بین آن‌ها را مدیریت کنیم.

پشتیبانی از مسیریابی شرطی و داینامیک: با استفاده از ویژگی‌هایی مانند Navigate و پارامترهای داینامیک، می‌توانیم مسیرها را با توجه به شرایط خاص به صورت داینامیک یا شرطی مدیریت کنیم.React Router ابزاری قدرتمند برای ایجاد مسیریابی در برنامه‌های React است و می‌تواند تجربه کاربری بهتری را در برنامه‌های تک‌صفحه‌ای فراهم کند. از ویژگی‌های مهم آن می‌توان به پشتیبانی از مسیریابی داینامیک و شرطی، قابلیت استفاده از لینک‌ها برای جابه‌جایی و هدایت کاربران به صورت برنامه‌ریزی‌شده اشاره کرد. این ابزار به توسعه‌دهندگان کمک می‌کند تا برنامه‌های پیچیده با چندین صفحه و مسیر را به روشی ساده‌تر و منظم‌تر پیاده‌سازی کنند.

ردیابی پروفایل و تحلیل عملکرد در React

در پروژه‌های بزرگ و پیچیده، بهبود عملکرد و اطمینان از سرعت مناسب برنامه یکی از مهم‌ترین اولویت‌ها برای توسعه‌دهندگان است. برنامه‌های React، به دلیل استفاده از کامپوننت‌ها و رندرهای مکرر، ممکن است با مشکلات عملکردی مواجه شوند، خصوصاً زمانی که حجم زیادی از داده‌ها پردازش و نمایش داده می‌شوند یا کامپوننت‌های پیچیده‌ای بارها رندر می‌شوند. برای شناسایی نقاط ضعف و بهینه‌سازی عملکرد، ردیابی پروفایل و تحلیل عملکرد بسیار اهمیت دارد. ابزارهای مختلفی مانند React Profiler در DevTools و Performance API می‌توانند به شناسایی بخش‌هایی از کد که باعث کندی برنامه می‌شوند کمک کنند.

Profiler در DevTools

Profiler یک ابزار داخلی React است که به شما امکان می‌دهد عملکرد کامپوننت‌ها را در زمان اجرا ردیابی کنید. Profiler مشخص می‌کند که کدام کامپوننت‌ها رندر شده‌اند، چه مدت طول کشیده تا رندر شوند و چه عواملی باعث رندر مجدد شده‌اند. این ابزار برای توسعه‌دهندگانی که می‌خواهند دقیقاً بدانند هر کامپوننت چقدر زمان برای رندر نیاز دارد، بسیار کاربردی است.

نحوه فعال‌سازی Profiler

برای استفاده از Profiler:

ابتدا اطمینان حاصل کنید که افزونه React DevTools برای مرورگر شما نصب شده است.
برنامه React خود را باز کنید و DevTools مرورگر را باز کنید.
به تب Profiler بروید.
با کلیک روی دکمه Record، می‌توانید پروفایل‌گیری را آغاز کنید. سپس می‌توانید در برنامه جا به جا شوید و کامپوننت‌های مختلف را بررسی کنید.
پس از اتمام، با کلیک روی دکمه Stop، ضبط را متوقف کنید و نتایج را مشاهده کنید.

تحلیل خروجی Profiler

زمانی که ضبط را متوقف می‌کنید، Profiler گزارشی از رندرها ارائه می‌دهد که شامل اطلاعات زیر است:

Duration: مدت زمانی که برای رندر شدن هر کامپوننت صرف شده است.
Commit: لحظه‌ای که تغییرات در DOM ثبت شده است.
Re-rendered Components: لیست کامپوننت‌هایی که به دلیل تغییرات مختلف، مجدداً رندر شده‌اند.
Render reason: دلیل اصلی رندر مجدد (مثلاً تغییر props یا state).
این اطلاعات به توسعه‌دهنده کمک می‌کند که بفهمد کدام کامپوننت‌ها نیاز به بهینه‌سازی دارند و چه عواملی باعث رندرهای مکرر و غیرضروری شده‌اند.

استفاده از Profiler API در کد

علاوه بر ابزار Profiler در DevTools، React یک API به نام Profiler ارائه می‌دهد که می‌توانید آن را مستقیماً در کد کامپوننت‌ها استفاده کنید تا رویدادهای رندر را ثبت کنید. این API به شما امکان می‌دهد که رندر کامپوننت‌های خاصی را به‌صورت برنامه‌ریزی‌شده ردیابی کنید.

مثال:

import React, { Profiler } from 'react';

function onRenderCallback(
  id, // شناسه کامپوننت
  phase, // 'mount' یا 'update'
  actualDuration, // زمان رندر شدن
  baseDuration, // زمان ایده‌آل برای رندر
  startTime, // زمان شروع رندر
  commitTime, // زمان ثبت تغییرات در DOM
  interactions // تعاملات مرتبط
) {
  console.log(`${id} [${phase}]: ${actualDuration}ms`);
}

function App() {
  return (
    <Profiler id="App" onRenderCallback={onRenderCallback}>
      <YourComponent />
    </Profiler>
  );
}

در این مثال:

onRenderCallback یک تابع است که هنگام هر بار رندر شدن YourComponent فراخوانی می‌شود و اطلاعاتی نظیر actualDuration (مدت واقعی رندر) و phase (نوع رندر، مثلاً mount یا update) را در اختیار ما قرار می‌دهد.
این اطلاعات به ما کمک می‌کند که تغییرات و بهبودهای عملکردی را بر اساس داده‌های دقیق پیاده‌سازی کنیم.

Performance API برای تحلیل عملکرد

Performance API مجموعه‌ای از ابزارهای مرورگر برای تحلیل عملکرد است که اطلاعات دقیق‌تری از زمان‌بندی عملیات‌ها فراهم می‌کند. این API برای اندازه‌گیری و تحلیل عمیق عملکرد برنامه کاربرد دارد و می‌تواند مکملی برای Profiler باشد. در React، می‌توان از Performance API برای اندازه‌گیری زمان انجام عملیات‌هایی مثل بارگذاری کامپوننت‌ها یا فراخوانی‌های شبکه‌ای استفاده کرد.

مثال:

function measureRenderTime() {
  performance.mark('start');

  // رندر کردن یا عملیات سنگین دیگر
  ReactDOM.render(<App />, document.getElementById('root'));

  performance.mark('end');
  performance.measure('Render time', 'start', 'end');

  const measure = performance.getEntriesByName('Render time')[0];
  console.log('Render time: ', measure.duration, 'ms');
}

measureRenderTime();

در این مثال:

از performance.mark برای ثبت نقاط شروع و پایان یک عملیات استفاده شده است.
با performance.measure می‌توانیم مدت زمان عملیات را اندازه‌گیری کنیم و آن را در کنسول مرورگر چاپ کنیم.
این روش برای شناسایی عملیات‌های سنگین یا کند بسیار مفید است.

تکنیک‌های بهینه‌سازی عملکرد در React

با استفاده از اطلاعات به دست آمده از Profiler و Performance API، می‌توان از تکنیک‌های مختلف برای بهینه‌سازی عملکرد استفاده کرد:

استفاده از React.memo: این تابع کامپوننت‌ها را به گونه‌ای بهینه می‌کند که فقط در صورت تغییر props، رندر شوند. این کار خصوصاً برای کامپوننت‌هایی که رندر آن‌ها سنگین است و نیازی به به‌روزرسانی مداوم ندارند، مفید است.

import React from 'react';

const MyComponent = React.memo(function MyComponent(props) {
  return <div>{props.content}</div>;
});

استفاده از useCallback و useMemo: این دو hook برای بهینه‌سازی توابع و مقادیر محاسبه‌شده استفاده می‌شوند. useCallback برای جلوگیری از ایجاد مجدد توابع و useMemo برای ذخیره محاسبات پرهزینه استفاده می‌شود.

import React, { useCallback, useMemo } from 'react';

function MyComponent({ items }) {
  const sortedItems = useMemo(() => {
    return items.sort();
  }, [items]);

  const handleClick = useCallback(() => {
    console.log('Clicked');
  }, []);

  return <button onClick={handleClick}>Click me</button>;
}

تقسیم کد (Code Splitting): با استفاده از Code Splitting می‌توان برنامه را به چند بخش کوچک‌تر تقسیم کرد و تنها زمانی که نیاز است، این بخش‌ها را بارگذاری کرد. React.lazy و Suspense ابزارهای اصلی در React برای این کار هستند.

import React, { Suspense, lazy } from 'react';

const OtherComponent = lazy(() => import('./OtherComponent'));

function App() {
  return (
    <Suspense fallback={<div>Loading...</div>}>
      <OtherComponent />
    </Suspense>
  );
}

کاهش رندرهای غیرضروری: با بررسی دقیق Profiler، می‌توانید متوجه شوید که چه تغییراتی باعث رندر مجدد کامپوننت‌ها می‌شوند و از تکنیک‌هایی مثل shouldComponentUpdate در کلاس‌ها یا React.memo در کامپوننت‌های تابعی استفاده کنید تا رندرهای غیرضروری را کاهش دهید. ردیابی پروفایل و تحلیل عملکرد در React به توسعه‌دهندگان کمک می‌کند که کامپوننت‌ها و ساختارهای برنامه را بهینه‌سازی کنند و از عملکرد سریع و روان اطمینان حاصل کنند. با استفاده از ابزارهای مانند React Profiler و Performance API، می‌توان بخش‌های کند برنامه را شناسایی و با تکنیک‌هایی مانند استفاده از React.memo، useCallback، و Code Splitting، تجربه کاربری بهتری ایجاد کرد.

استفاده از ابزارهای DevTools در React

React DevTools مجموعه‌ای از ابزارهای قدرتمند است که به توسعه‌دهندگان React امکان می‌دهد تا وضعیت، ساختار، و عملکرد کامپوننت‌ها را به‌طور دقیق بررسی کنند. این ابزارها به شما کمک می‌کنند تا به‌صورت بلادرنگ وضعیت (state) و ویژگی‌های (props) کامپوننت‌ها را مشاهده کنید، تغییرات را پیگیری کنید و عملکرد برنامه را بهبود بخشید. React DevTools به شما امکان می‌دهد ساختار درختی کامپوننت‌ها را ببینید، زمان‌بندی رندرها را بررسی کنید و درک کاملی از رفتار برنامه به‌دست آورید.

نصب و راه‌اندازی React DevTools

React DevTools به‌صورت یک افزونه مرورگر در دسترس است و برای مرورگرهای Chrome و Firefox قابل نصب می‌باشد. با نصب این افزونه:

پس از بارگذاری برنامه React، به بخش DevTools مرورگر خود بروید.
تب جدیدی به نام React به DevTools اضافه می‌شود که حاوی ابزارهای React DevTools است.

ویژگی‌های کلیدی React DevTools

React DevTools شامل ویژگی‌های متعددی است که فرآیند توسعه و دیباگ کردن را ساده‌تر و سریع‌تر می‌کند:

درخت کامپوننت‌ها (Component Tree): ساختار درختی کامپوننت‌ها را به شما نمایش می‌دهد و می‌توانید مشاهده کنید که هر کامپوننت چطور به دیگر کامپوننت‌ها متصل شده است. این بخش به شما اجازه می‌دهد به‌سادگی کامپوننت‌ها و زیرکامپوننت‌ها را شناسایی و وضعیت آن‌ها را مدیریت کنید.

مشاهده و تغییر Props و State: در React DevTools، شما می‌توانید مقادیر props و state کامپوننت‌ها را مشاهده کرده و حتی آن‌ها را به‌صورت بلادرنگ تغییر دهید. این ویژگی برای تست تغییرات و مشاهده تاثیر آن‌ها بر رفتار برنامه بدون نیاز به تغییر مستقیم کد، بسیار مفید است.

Profiler برای تحلیل عملکرد: در React DevTools، تب Profiler به شما اجازه می‌دهد عملکرد برنامه را ردیابی کنید. با استفاده از این ابزار، می‌توانید مدت زمان صرف شده برای رندر هر کامپوننت را مشاهده کرده و بفهمید کدام کامپوننت‌ها باعث کندی برنامه می‌شوند. این ابزار به شما کمک می‌کند تا رندرهای غیرضروری را شناسایی و بهینه‌سازی‌های لازم را انجام دهید.

ابزار جستجو (Search Tool): با استفاده از ابزار جستجو، می‌توانید درخت کامپوننت‌ها را بر اساس نام آن‌ها جستجو کنید. این ویژگی به خصوص در پروژه‌های بزرگ که شامل صدها کامپوننت مختلف هستند، بسیار مفید است و امکان دسترسی سریع‌تر به کامپوننت‌های خاص را فراهم می‌کند.

بررسی هوک‌ها (Hooks): React DevTools قابلیت نمایش هوک‌ها را نیز فراهم می‌کند. این ویژگی به شما امکان می‌دهد مقادیر و وضعیت هوک‌ها مانند useState و useEffect را مشاهده کنید. این ابزار برای بررسی و تغییر وضعیت کامپوننت‌های تابعی که از هوک‌ها استفاده می‌کنند بسیار مفید است.

نحوه استفاده از DevTools در توسعه و دیباگ کردن

مشاهده و تغییر Props و State

در تب React، می‌توانید روی هر کامپوننت کلیک کنید و مقادیر props و state آن را مشاهده کنید. برای تغییر state، می‌توانید مقادیر دلخواه را وارد کنید و تغییرات را بلافاصله در برنامه مشاهده کنید. این قابلیت برای تست و دیباگ وضعیت‌های مختلف برنامه بسیار کاربردی است.

تحلیل عملکرد با Profiler

برای شروع پروفایل‌گیری از عملکرد:

به تب Profiler در React DevTools بروید.
روی دکمه Record کلیک کنید تا پروفایل‌گیری آغاز شود.
برنامه را مرور کنید و هرگونه تعامل یا جابه‌جایی بین کامپوننت‌ها را انجام دهید.
با کلیک روی دکمه Stop، پروفایل‌گیری را متوقف کنید.
پس از اتمام، Profiler اطلاعات دقیقی از زمان صرف شده برای رندر هر کامپوننت و تغییراتی که باعث رندر مجدد شده‌اند ارائه می‌دهد. این داده‌ها به شما کمک می‌کنند تا رندرهای غیرضروری را شناسایی و آن‌ها را بهینه کنید.

بررسی هوک‌ها

در کامپوننت‌های تابعی که از هوک‌ها استفاده می‌کنند، React DevTools به شما امکان می‌دهد تا مقادیر هوک‌ها را مشاهده کنید. به عنوان مثال، اگر یک کامپوننت از useState استفاده کند، شما می‌توانید مقدار فعلی آن state و تغییرات آن را در طول زمان مشاهده کنید.

مثال: فرض کنید کامپوننت زیر را داریم که از useState برای مدیریت یک شمارنده استفاده می‌کند:

import React, { useState } from 'react';

function Counter() {
  const [count, setCount] = useState(0);

  return (
    <div>
      <p>Count: {count}</p>
      <button onClick={() => setCount(count + 1)}>Increment</button>
    </div>
  );
}

با استفاده از React DevTools، می‌توانید مقدار count را مشاهده کنید و با تغییر آن مستقیماً در DevTools، تاثیر آن بر UI را به صورت زنده ببینید.

نکات و توصیه‌ها برای استفاده بهینه از React DevTools

استفاده مداوم از Profiler: Profiler را به‌طور منظم اجرا کنید، مخصوصاً در طول توسعه کامپوننت‌های جدید. با این کار می‌توانید از همان ابتدا مشکلات عملکردی را شناسایی و رفع کنید.

مدیریت دقیق Props و State: هنگام تغییر props و state در DevTools، تاثیر آن‌ها را در دیگر کامپوننت‌ها و همچنین تاثیرات احتمالی بر عملکرد بررسی کنید.

جستجوی سریع در ساختار پروژه: از قابلیت جستجو برای پیدا کردن سریع کامپوننت‌ها در پروژه‌های بزرگ استفاده کنید. این ویژگی باعث می‌شود نیازی به جستجوی دستی در کل ساختار کامپوننت‌ها نداشته باشید.

پیکربندی برای محیط توسعه و تولید: در محیط تولید (Production)، معمولاً React DevTools غیرفعال است تا کارایی و امنیت برنامه افزایش یابد. به همین دلیل بهتر است در حین توسعه از DevTools استفاده کرده و بهینه‌سازی‌های لازم را انجام دهید. React DevTools یک ابزار ضروری برای هر توسعه‌دهنده React است. با استفاده از ویژگی‌های این ابزار مانند بررسی درخت کامپوننت‌ها، مشاهده props و state، پروفایل‌گیری از عملکرد و پشتیبانی از هوک‌ها، می‌توان به‌سرعت مشکلات را شناسایی و رفع کرد و عملکرد برنامه را بهبود بخشید. این ابزار به توسعه‌دهندگان کمک می‌کند تا به راحتی وضعیت کامپوننت‌ها و تعاملات آن‌ها را بررسی کرده و تجربه کاربری بهتری ایجاد کنند.

نتیجه‌گیری

در این مقاله، به بررسی موضوعات خاص (Special Topics) React پرداختیم و با ابزارها و تکنیک‌های مختلفی که به توسعه‌دهندگان کمک می‌کنند تا برنامه‌های قدرتمندتر و بهینه‌تری بسازند، آشنا شدیم. از TypeScript در React برای افزایش ایمنی و دقت در کدنویسی، تا استفاده از React بدون JSX برای سازگاری با محیط‌های خاص، و نوشتن تست‌ها در React برای اطمینان از عملکرد صحیح برنامه، این مباحث به توسعه‌دهندگان امکان می‌دهند تا پروژه‌های خود را با کیفیت بیشتری پیاده‌سازی کنند.

همچنین با استفاده از ابزارهایی نظیر React Router برای مدیریت مسیرها، Profiler و Performance API برای تحلیل عملکرد، و React DevTools برای بررسی وضعیت کامپوننت‌ها و دیباگ کردن، می‌توان تجربه کاربری برنامه را بهبود بخشید و از کارکرد درست و بهینه‌ی هر بخش اطمینان حاصل کرد.

توسعه‌دهندگان React با بهره‌گیری از این موضوعات خاص، می‌توانند سطح تخصص و کارایی خود را در ایجاد رابط‌های کاربری پیچیده و بهینه افزایش دهند. این ابزارها و تکنیک‌ها نه تنها فرآیند توسعه را سریع‌تر و دقیق‌تر می‌کنند، بلکه موجب بهبود قابل توجه عملکرد و تجربه کاربری نیز می‌شوند. اگر مایل به یادگیری بیشتر هستید، پیشنهاد می‌کنیم به مستندات رسمی React و منابع آنلاین معتبر رجوع کنید و با تمرین و پیاده‌سازی این مفاهیم، مهارت خود را در توسعه با React گسترش دهید.

آموزش موضوعات خاص (Special Topics) React

دیدگاه های شما

دیدگاهتان را بنویسید

نشانی ایمیل شما منتشر نخواهد شد. بخش‌های موردنیاز علامت‌گذاری شده‌اند *