بررسی فرمت تاریخ در جاوااسکریپت

فرمت تاریخ در جاوااسکریپت یکی از چالش‌های رایج در توسعه اپلیکیشن‌های وب است؛ چالشی که اگر به‌درستی مدیریت نشود، می‌تواند روی پایداری، دقت و تجربه کاربری تأثیر منفی بگذارد. در این مقاله، مدیریت و فرمت‌بندی تاریخ در جاوااسکریپت را با بررسی API داخلی Date و کتابخانه‌های تخصصی مقایسه می‌کنیم و با استفاده از مثال‌های عملی و معیارهای عملکردی، یاد می‌گیریم چگونه بهترین تصمیم را برای پیاده‌سازی بگیریم.

در ادامه بررسی می‌کنیم چه زمانی استفاده از متدهای داخلی جاوااسکریپت انتخاب مناسبی است و چه زمانی بهتر است سراغ کتابخانه‌های خارجی برویم. همچنین با نحوه مدیریت صحیح localization و منطقه زمانی (Time Zone) آشنا می‌شویم و یاد می‌گیریم چگونه از خطاهای رایج مرتبط با تاریخ در پروژه‌هایمان جلوگیری کنیم.

درک فرمت‌های تاریخ در جاوااسکریپت

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

در ادامه، سه فرمت رایج تاریخ را مشاهده می‌کنیم که همگی یک تاریخ و زمان یکسان را نمایش می‌دهند:

  • رشته‌های ISO 8601 (استاندارد و قابل استفاده مستقیم باnew Date() ) "2025-02-18T14:30:00.000Z"
  • تایم‌استمپ یونیکس (نمایش عددی و مستقل از Time Zone) 1732561800000
  • RFC 2822 Tue, 18 Feb 2025 14:30:00 +0000

هرکدام از این فرمت‌ها بسته به سناریو کاربرد خاص خودشان را دارند. فرمت ISO 8601 به دلیل استاندارد بودن و قابلیت parse آسان با new Date()، رایج‌ترین گزینه در APIها و دیتابیس‌ها محسوب می‌شود.

از آن‌جایی که Unix timestamps صرفاً اعداد خام هستند، برای محاسبات و مقایسه‌های زمانی گزینه بسیار مناسبی به شمار می‌آیند. فرمت RFC 2822 نیز بیشتر در سیستم‌های قدیمی‌تر یا ایمیل‌ها دیده می‌شود. فارغ از این‌که با کدام فرمت شروع می‌کنیم، آبجکت Date در جاوااسکریپت ابزار اصلی ما برای تفسیر و کار با این مقادیر است.

آبجکت Date در جاوااسکریپت

آبجکت Date روش داخلی جاوااسکریپت برای کار با تاریخ و زمان است. چند نکته‌ی مهم وجود دارد که باید درباره‌ی آن بدانیم:

// Creating a new Date object
const now = new Date(); // Current date and time
const isoDate = new Date('2025-02-18T14:30:00.000Z'); // From date string
const withComponents = new Date(2025, 1, 18); // Year, month (0-indexed!), day
const timeStampDate = new Date(1732561800000)

آبجکت Date تاریخ‌ها را به‌صورت تعداد میلی‌ثانیه از پنج‌شنبه ۱ ژانویه ۱۹۷۰ (Unix Epoch) ذخیره می‌کند، اما متدهایی در اختیار ما قرار می‌دهد که می‌توانیم با استفاده از آن‌ها این مقادیر را به فرمت‌های قابل خواندن برای انسان تبدیل کنیم.

متدهای رایج آبجکت  Date

متدهای داخلی این آبجکت به ما اجازه می‌دهند بخش‌های مختلف تاریخ، مانند سال، ماه، روز یا زمان را استخراج کنیم. برای مثال:

const date = new Date('2025-02-18T14:30:15Z');

// Getting components
date.getFullYear(); // 2025
date.getMonth(); // 1 (February, zero-indexed!!!!)
date.getDate(); // 18
date.getHours(); // 14
date.getMinutes(); // 30
date.getSeconds(); // 15
date.getDay(); // 2 (Tuesday, with 0 being Sunday)
date.getTime(); // Milliseconds since epoch

// Setting components
date.setFullYear(2026);
date.setMonth(5); // June (because zero-indexed!!!)

متدهای فرمت‌بندی داخلی Date

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

const date = new Date('2025-02-18T14:30:00Z');

// Basic string conversion
date.toString(); 
// "Tue Feb 18 2025 14:30:00 GMT+0000 (Coordinated Universal Time)"

// Date portion only
date.toDateString(); 
// "Tue Feb 18 2025"

// Time portion only
date.toTimeString(); 
// "14:30:00 GMT+0000 (Coordinated Universal Time)"

// UTC version (reliable across timezones)
date.toUTCString(); 
// "Tue, 18 Feb 2025 14:30:00 GMT"

// ISO 8601 format
date.toISOString(); 
// "2025-02-18T14:30:00.000Z"

این متدهای native یک راه سریع و بدون وابستگی اضافی برای فرمت‌بندی تاریخ در اختیار ما قرار می‌دهند. آن‌ها برای سناریوهای ساده‌ای مثل نمایش مقادیر UTC یا جدا کردن بخش تاریخ و زمان، گزینه‌ای ایده‌آل محسوب می‌شوند.

فرمت‌بندی مبتنی بر لوکال با toLocaleDateString()

const date = new Date('2025-02-18');

// Basic usage (uses browser's locale)
date.toLocaleDateString(); 
// In US: "2/18/2025"
// In UK: "18/02/2025"
// In Germany: "18.2.2025"

// With explicit locale
date.toLocaleDateString('fr-FR'); 
// "18/02/2025"

// With options
const options = { 
  weekday: 'long', 
  year: 'numeric', 
  month: 'long', 
  day: 'numeric' 
};
date.toLocaleDateString('de-DE', options); 
// "Dienstag, 18. Februar 2025"

بدون تعیین locale و option:

date.toLocaleDateString();
// 2/18/2025

فرمت‌بندی دستی تاریخ با جاوااسکریپت خام

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

function formatDate(date, format) {
  const day = String(date.getDate()).padStart(2, '0');
  const month = String(date.getMonth() + 1).padStart(2, '0');
  const year = date.getFullYear();
  const hours = String(date.getHours()).padStart(2, '0');
  const minutes = String(date.getMinutes()).padStart(2, '0');
  const seconds = String(date.getSeconds()).padStart(2, '0');

  // Replace tokens with actual values
  return format
    .replace('YYYY', year)
    .replace('MM', month)
    .replace('DD', day)
    .replace('HH', hours)
    .replace('mm', minutes)
    .replace('ss', seconds);
}

const date = new Date('2025-02-18T14:30:45Z');
console.log(formatDate(date, 'YYYY-MM-DD')); // "2025-02-18"
console.log(formatDate(date, 'DD/MM/YYYY HH:mm:ss')); // "18/02/2025 14:30:45"

زمانی که متدهای داخلی کافی نیستند

با این‌که روش‌های بالا در بسیاری از موارد کارآمد هستند، اما با در نظر گرفتن موارد زیر، پیچیدگی به‌سرعت افزایش پیدا می‌کند:

  • Localization (نام ماه‌ها و روزها)
  • سیستم‌های تقویمی مختلف
  • مدیریت مناطق زمانی
  • فرمت‌بندی AM / PM

برای هر سناریویی فراتر از فرمت‌های ساده، زمان آن رسیده است که سراغ ابزارها و کتابخانه‌های حرفه‌ای‌تر برویم.

رویکردهای مدرن با کتابخانه‌ها

متدهای داخلی جاوااسکریپت در بسیاری از سناریوهای ساده کاربردی هستند، اما در موارد پیچیده‌تر می‌توانند محدود کننده باشند؛ به‌ویژه زمانی که با Localization پیشرفته، فرمت‌های سفارشی یا مدیریت Time Zone سروکار داریم. در چنین شرایطی، استفاده از کتابخانه‌های تخصصی، فرایند فرمت تاریخ در جاوااسکریپت را به‌شکل قابل‌توجهی ساده‌تر، قابل‌اعتمادتر و خواناتر می‌کند. در ادامه، چند کتابخانه محبوب و رایج برای فرمت‌بندی تاریخ را بررسی می‌کنیم.

date-fns: رویکرد تابع‌محور و مدرن

date-fns یکی از بهترین انتخاب‌ها برای اپلیکیشن‌های مدرن محسوب می‌شود؛ به‌خصوص زمانی که خوانایی کد، عملکرد و کنترل دقیق روی فرمت تاریخ در جاوااسکریپت برایمان اهمیت دارد.

ویژگی‌های کلیدی date-fns:

  • Tree-shakable: فقط توابعی را import می‌کنیم که واقعاً به آن‌ها نیاز داریم
  • Immutable: هیچ‌کدام از توابع، مقدار تاریخ اصلی را تغییر نمی‌دهند
  • مستندسازی قوی و سازگار با تایپ اسکریپت: یک مزیت مهم برای پروژه‌های حرفه‌ای و مقیاس‌پذیر

با استفاده از date-fns می‌توان به‌سادگی کارهای رایجی مانند parse کردن رشته‌های ISO به آبجکت Date، فرمت کردن تاریخ به رشته‌های خوانا و انجام محاسبات زمانی (مثل اضافه کردن روز یا محاسبه اختلاف بین دو تاریخ) را انجام داد. طراحی تابع‌محور این کتابخانه باعث می‌شود کدها تمیز، قابل پیش‌بینی و به‌راحتی قابل ترکیب باشند؛ ویژگی‌ای که در پروژه‌های بزرگ اهمیت زیادی دارد.

import { format, parseISO, addDays, differenceInDays } from 'date-fns';

// Parsing
const date = parseISO('2025-02-18T14:30:00Z');

// Formatting
format(date, 'yyyy-MM-dd'); // "2025-02-18"
format(date, 'MMMM do, yyyy'); // "February 18th, 2025"
format(date, 'h:mm a'); // "2:30 PM"
format(date, 'EEEE, MMMM do, yyyy h:mm a'); // "Tuesday, February 18th, 2025 2:30 PM"

// Operations
const nextWeek = addDays(date, 7);
const daysBetween = differenceInDays(nextWeek, date); // 7

Localization با date-fns

کتابخانه date-fns از طریق import جداگانه localeها، پشتیبانی قدرتمندی از Localization ارائه می‌دهد. این رویکرد ماژولار کمک می‌کند حجم باندل نهایی حداقل بماند، زیرا فقط localeهایی را وارد پروژه می‌کنیم که واقعاً به آن‌ها نیاز داریم.

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

import { format, formatDistance, formatRelative, isDate } from 'date-fns'; 
import { es, de, fr, ja, zhCN } from 'date-fns/locale';

const date = new Date('2025-02-18T14:30:00Z'); 

// Basic locale formatting 
const localeExamples = { 
  english: format(date, 'MMMM d, yyyy', { locale: enUS }), 
  spanish: format(date, 'MMMM d, yyyy', { locale: es }), 
  german: format(date, 'MMMM d, yyyy', { locale: de }), 
  french: format(date, 'MMMM d, yyyy', { locale: fr }), 
  japanese: format(date, 'MMMM d, yyyy', { locale: ja }), 
  chinese: format(date, 'MMMM d, yyyy', { locale: zhCN }) 
};

console.log(localeExamples); 
 Output: 
  { 
   english: "February 18, 2025", 
   spanish: "febrero 18, 2025", 
   german: "Februar 18, 2025", 
   french: "février 18, 2025", 
   japanese: "2月 18, 2025", 
   chinese: "二月 18, 2025" 
 }

اگر به سفارشی‌سازی بیشتری نیاز داشته باشیم یا با سناریوهای خاص مواجه شویم، مراجعه به مستندات رسمی date-fns انتخاب مناسبی است؛ چرا که مثال‌ها و تکنیک‌های تکمیلی کاربردی‌ای در آن ارائه شده است.

مدیریت Time Zone با date-fns-tz

کتابخانه date-fns-tz قابلیت‌های date-fns را با پشتیبانی قدرتمند از Time Zone تکمیل می‌کند. این ابزار به ما اجازه می‌دهد تاریخ‌ها را بین مناطق زمانی مختلف تبدیل کرده و آن‌ها را به‌درستی فرمت کنیم. استفاده از این کتابخانه، پیاده‌سازی دقیق‌تر فرمت تاریخ در جاوااسکریپت را در اپلیکیشن‌هایی با کاربران بین‌المللی ممکن می‌سازد.

import { 
  format, 
  utcToZonedTime, 
  zonedTimeToUtc, 
  getTimezoneOffset 
} from 'date-fns-tz'; 

const date = new Date('2025-02-18T14:30:00Z');

// Basic timezone conversion 
const timezoneExamples = { 
  newYork: utcToZonedTime(date, 'America/New_York'), 
  tokyo: utcToZonedTime(date, 'Asia/Tokyo'), 
  london: utcToZonedTime(date, 'Europe/London'), 
  sydney: utcToZonedTime(date, 'Australia/Sydney') 
};

// console.log(timezoneExamples)

//{
//  newYork: Tue Feb 18 2025 09:30:00 GMT-0500 (Eastern Standard Time),
//  tokyo:   Tue Feb 18 2025 23:30:00 GMT+0900 (Japan Standard Time),
//  london:  Tue Feb 18 2025 14:30:00 GMT+0000 (Greenwich Mean Time),
//  sydney:  Wed Feb 19 2025 01:30:00 GMT+1100 (Australian Eastern Daylight Time)
//}

Day.js: جایگزین سبک و مدرن Moment.js

کتابخانه Day.js به‌عنوان یک جایگزین مدرن و مینیمال برای Moment.js، محبوبیت زیادی پیدا کرده است. این کتابخانه با هدف رفع محدودیت‌ها و مشکلات Moment طراحی شده، در حالی که API نسبتاً مشابهی ارائه می‌دهد؛ به همین دلیل، گزینه‌ای مناسب برای پروژه‌هایی است که قصد دارند بدون تغییرات گسترده در کدها، از Moment.js به یک راهکار مدرن‌تر انتقال پیدا کنند.

در مثال زیر می‌بینیم چگونه می‌توان با استفاده از پلاگین‌های مختلف، فرمت تاریخ در جاوااسکریپت، مدیریت Time Zone، فرمت‌های سفارشی و localeها را پیاده‌سازی کرد. همچنین امکان ایجاد تاریخ از ورودی‌های مختلف، تبدیل آن به رشته‌های خوانا، تغییر منطقه زمانی و انجام محاسبات زمانی (مثل اضافه یا کم کردن زمان) وجود دارد؛ آن هم بدون تغییر دادن مقدار تاریخ اصلی:

import dayjs from 'dayjs';
import utc from 'dayjs/plugin/utc';
import timezone from 'dayjs/plugin/timezone';
import localeData from 'dayjs/plugin/localeData';
import customParseFormat from 'dayjs/plugin/customParseFormat';

// Extend with plugins
dayjs.extend(utc);
dayjs.extend(timezone);
dayjs.extend(localeData);
dayjs.extend(customParseFormat);

// Creating Day.js objects
const today = dayjs();
const specificDate = dayjs('2025-02-18');
const fromFormat = dayjs('18/02/2025', 'DD/MM/YYYY');

// Formatting
specificDate.format('YYYY-MM-DD'); // "2025-02-18"
specificDate.format('dddd, MMMM D, YYYY'); // "Tuesday, February 18, 2025"

// Timezone handling
specificDate.tz('America/New_York').format('YYYY-MM-DD HH:mm:ss Z'); 
// "2025-02-18 09:30:00 -05:00"

// Manipulation (immutable - returns new instances)
const nextWeek = specificDate.add(1, 'week');
const lastMonth = specificDate.subtract(1, 'month');

مزایای کلیدی Day.js

  • حجم بسیار کم: حدود ۲ کیلوبایت (minified و gzipped، فقط هسته اصلی)
  • API آشنا: مشابه Moment.js و مناسب برای انتقال آسان
  • Immutable: تمام عملیات، آبجکت جدید برمی‌گردانند
  • مبتنی بر پلاگین: فقط قابلیت‌هایی را اضافه می‌کنیم که واقعاً به آن‌ها نیاز داریم
  • Chainable API: زنجیره‌کردن متدها به‌شکل ساده و خوانا

معماری پلاگین‌محور Day.js بسیار هوشمندانه طراحی شده است؛ زیرا فقط هزینه حجمی قابلیت‌هایی را می‌پردازیم که در پروژه از آن‌ها استفاده می‌کنیم، بدون آن‌که انعطاف‌پذیری در فرمت تاریخ در جاوااسکریپت کاهش پیدا کند.

// Only import the plugins you need
import relativeTime from 'dayjs/plugin/relativeTime';
import calendar from 'dayjs/plugin/calendar';
dayjs.extend(relativeTime);
dayjs.extend(calendar);

// Now you can use these features
dayjs('2025-02-18').fromNow(); // "in X years" (depends on current date)
dayjs('2025-02-18').calendar(); // "02/18/2025" or "Tuesday" based on how far in future

Moment.js: گزینه قدیمی

کتابخانه Moment.js زمانی انتخاب اول توسعه‌دهندگان برای مدیریت و فرمت تاریخ در جاوااسکریپت بود، اما امروز به‌عنوان یک گزینه قدیمی شناخته می‌شود. تیم Moment.js به‌صورت رسمی اعلام کرده که این کتابخانه در حالت نگه‌داری (Maintenance Mode) قرار دارد و استفاده از جایگزین‌های مدرن‌تر را توصیه می‌کند.

با این حال، هنوز پروژه‌های زیادی از Moment.js استفاده می‌کنند و آشنایی با رویکرد آن می‌تواند مفید باشد.

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

import moment from 'moment';
import 'moment-timezone';

// Creating moments
const now = moment(); // Current date/time
const fromString = moment('2025-02-18T14:30:00Z');
const fromFormat = moment('18/02/2025', 'DD/MM/YYYY');

// Formatting
fromString.format('YYYY-MM-DD'); // "2025-02-18"
fromString.format('dddd, MMMM Do YYYY'); // "Tuesday, February 18th 2025"
fromString.format('h:mm a'); // "2:30 pm"

// Operations (modifies the original moment)
fromString.add(7, 'days');
fromString.subtract(2, 'months');

// Timezone handling
const tokyoTime = fromString.clone().tz('Asia/Tokyo').format('YYYY-MM-DD HH:mm:ss');
const nyTime = fromString.clone().tz('America/New_York').format('YYYY-MM-DD HH:mm:ss');

معایب اصلی Moment.js

با وجود محبوبیت تاریخی Moment.js، این کتابخانه محدودیت‌هایی دارد که در پروژه‌های مدرن، به‌ویژه در زمینه فرمت تاریخ در جاوااسکریپت، می‌تواند مشکل‌ساز شود:

  • Tree-shakable نیست: باعث افزایش قابل‌توجه حجم باندل نهایی می‌شود
  • API قابل تغییر: امکان تغییر ناخواسته تاریخ‌ها وجود دارد
  • زنجیره‌کردن پیچیده متدها: در برخی سناریوها می‌تواند منجر به رفتارهای غیرمنتظره و باگ‌های سخت‌یاب شود

Temporal: آینده مدیریت تاریخ در جاوااسکریپت

پروپوزال Temporal در استاندارد ECMAScript با هدف جایگزینی API مشکل‌دار Date ارائه شده است. این API جدید، جامع‌تر، immutable و کاملاً آگاه از Time Zone طراحی شده و بسیاری از مشکلات رایج در فرمت تاریخ در جاوااسکریپت را به‌صورت ریشه‌ای حل می‌کند.

اگرچه Temporal هنوز به‌طور رسمی وارد استاندارد نشده است، اما به‌عنوان آینده مدیریت تاریخ در جاوااسکریپت، ارزش بررسی و توجه جدی را دارد.

در قطعه کد زیر، رویکرد مدرن Temporal را مشاهده می‌کنیم: ساخت تاریخ‌های immutable و آگاه از Time Zone و انجام محاسبات زمانی به‌شکلی ایمن و قابل پیش‌بینی.

// This syntax is not yet available in browsers without polyfills

// Creating a date (Temporal.PlainDate is timezone-independent)
const date = Temporal.PlainDate.from({ year: 2025, month: 2, day: 18 });

// Creating a specific time in a timezone
const nyDateTime = Temporal.ZonedDateTime.from({
  timeZone: 'America/New_York',
  year: 2025, month: 2, day: 18, hour: 9, minute: 30
});

// Formatting
date.toString(); // "2025-02-18"
nyDateTime.toString(); // "2025-02-18T09:30:00-05:00[America/New_York]"

// Duration and arithmetic (returns new instances)
const futureDate = date.add({ days: 7 });
const duration = date.until(futureDate);

در حال حاضر می‌توانیم Temporal را با استفاده از npmjs.com/package/@js-temporal/polyfill تجربه و بررسی کنیم.

مقایسه کتابخانه‌ها و راهنمای انتخاب

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

از نظر حجم باندل

  • Native Date: بدون هزینه اضافی، چون بخشی از خود جاوااسکریپت است
  • date-fns: حدود ۱۳ کیلوبایت (minified و gzipped)
  • Day.js: حدود ۲ کیلوبایت و بسیار سبک
  • Moment.js: حدود ۶۷ کیلوبایت و سنگین‌ترین گزینه

از نظر immutability

  • Native Date و Moment.js: مقادیر تاریخ را به‌صورت mutable مدیریت می‌کنند
  • date-fns و Day.js: کاملاً immutable هستند و همیشه آبجکت جدید برمی‌گردانند

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

از نظر tree-shaking

  • date-fns: عملکرد بسیار عالی؛ فقط توابع موردنیاز وارد باندل می‌شوند
  • Day.js: وضعیت مناسب، به‌ویژه با معماری پلاگین‌محور
  • Moment.js: عملکرد ضعیف در این زمینه
  • Native Date: نیازی به tree-shaking ندارد

از نظر پشتیبانی از Time Zone

  • Native Date: امکانات محدود
  • date-fns: با کتابخانه‌ی مکمل date-fns-tz پشتیبانی قدرتمند ارائه می‌دهد
  • Day.js و Moment.js: از طریق پلاگین‌ها امکان مدیریت Time Zone دارند

در حوزه Localization

  • Native Date: پشتیبانی قابل‌قبول
  • date-fns و Moment.js: پشتیبانی بسیار قوی
  • Day.js: عملکرد خوب، اما برخی localeها نیاز به پلاگین دارند

از نظر پشتیبانی از تایپ اسکریپت

  • date-fns: بهترین تجربه توسعه
  • Day.js: پشتیبانی مناسب
  • Moment.js: عمدتاً از طریق DefinitelyTyped
  • Native Date: امکانات پایه‌ای

از نظر وضعیت توسعه نیز، date-fns و Day.js به‌صورت فعال در حال توسعه هستند، در حالی که Moment.js فقط در حالت نگه‌داری قرار دارد.

هر گزینه را چه زمانی استفاده کنیم؟

  • متدهای Native Date: برای پروژه‌های کوچک، نمایش‌های ساده تاریخ و زمانی که حداقل بودن حجم باندل اهمیت بالایی دارد.
  • date-fns: انتخابی عالی برای اپلیکیشن‌های مدرن، زمانی که رویکرد تابع‌محور یا tree-shaking برایمان اهمیت دارد و کنترل دقیق‌تری روی فرمت تاریخ در جاوااسکریپت می‌خواهیم.
  • Day.js: گزینه‌ای ایده‌آل هنگام انتقال از Moment.js یا زمانی که به حجم باندل بسیار کم نیاز داریم، اما امکانات Native Date کافی نیستند.
  • Moment.js: مناسب پروژه‌های قدیمی یا شرایطی که هزینه انتقال و بازنویسی بیشتر از مزایای آن است.

ملاحظات مربوط به عملکرد

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

تأثیر حجم باندل

  • Native Date: بدون افزایش حجم
  • date-fns: حدود ۱۳ کیلوبایت
  • date-fns با همه localeها: حدود ۵۰۰ کیلوبایت (قبل از tree-shaking)
  • Day.js (هسته اصلی): حدود ۲ کیلوبایت
  • Day.js با پلاگین‌ها: حدود ۵ تا ۱۰ کیلوبایت
  • Moment.js: حدود ۶۷ کیلوبایت
  • Moment.js با همه localeها: حدود ۳۰۲ کیلوبایت

مصرف حافظه و CPU

  • Native Date: سریع‌ترین گزینه، اما با امکانات محدود
  • date-fns: مصرف حافظه مناسب و عملکرد پایدار به‌دلیل توابع خالص
  • Day.js: بسیار سریع در عملیات پایه؛ استفاده سنگین از پلاگین‌ها ممکن است کمی سربار ایجاد کند
  • Moment.js: مصرف حافظه بیشتر و عملکرد ضعیف‌تر در پردازش‌های سنگین

در ادامه، یک مقایسه ساده شده از عملکرد ارائه شده است:

// Test with 100,000 operations
const COUNT = 100000;

// Native JS
console.time('Native');
for (let i = 0; i < COUNT; i++) {
  new Date().toISOString();
}
console.timeEnd('Native'); // Typically fastest

// date-fns
console.time('date-fns');
for (let i = 0; i < COUNT; i++) {
  format(new Date(), 'yyyy-MM-dd\'T\'HH:mm:ss.SSS\'Z\'');
}
console.timeEnd('date-fns'); // Close second

// Day.js
console.time('Day.js');
for (let i = 0; i < COUNT; i++) {
  dayjs().format('YYYY-MM-DDTHH:mm:ss.SSS[Z]');
}
console.timeEnd('Day.js'); // Usually faster than Moment

// Moment.js
console.time('Moment');
for (let i = 0; i < COUNT; i++) {
  moment().format('YYYY-MM-DDTHH:mm:ss.SSS[Z]');
}
console.timeEnd('Moment'); // Usually slowest

جلوگیری از خطاهای رایج

برای کاهش باگ‌های متداول هنگام کار با فرمت تاریخ در جاوااسکریپت، به نکات زیر توجه داشته باشیم:

  • ماه‌ها در API مربوط به Date به‌صورت zero-indexed هستند (ژانویه = ۰)
  • در محاسبات زمانی نزدیک به تغییر ساعت تابستانی (DST) دقت ویژه داشته باشیم
  • آبجکت‌های Date را مستقیماً با == یا === مقایسه نکنیم و از .getTime() استفاده نماییم
  • به تفاوت رفتار parse تاریخ در مرورگرهای مختلف توجه داشته باشیم
//Incorrect: Direct comparison
const date1 = new Date('2025-02-18');
const date2 = new Date('2025-02-18');
if (date1 === date2) { / This will never execute / }// Correct version: Compare timestamps
if (date1.getTime() === date2.getTime()) { /*This works / }

جمع‌بندی

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

  • Native Date: مناسب سناریوهای ساده
  • date-fns: رویکردی مدرن و تابع‌محور با پشتیبانی عالی از tree-shaking
  • Moment.js: قابل استفاده، اما برای پروژه‌های جدید توصیه نمی‌شود
  • Temporal API: نمایانگر آینده مدیریت تاریخ در جاوااسکریپت

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

دیدگاه‌ها:

افزودن دیدگاه جدید