تایپ اسکریپت چیست؟ چرا باید تایپ اسکریپت یاد بگیریم؟

تایپ اسکریپت چیست؟ چرا باید تایپ اسکریپت یاد بگیریم؟

سلام به همه همراهان سکان آکادمی،

امروز می‌خواهیم درباره "تایپ‌اسکریپت" (TypeScript) صحبت کنیم و بعضی از مطالب مربوط به این زبان را با هم یاد بگیریم. چون واقعیت این است که خود من هم تازه دارم با این زبان آشنا می‌شوم.

در این مقاله، "انواع داده‌ها (data types)" و اینکه "چرا باید تایپ‌اسکریپت یاد بگیریم؟" را با هم آموزش می‌بینیم.

تایپ‌اسکریپت (TypeScript) چیست؟

همانطور که می‌دانید، جاوا‌اسکریپت (JavaScript)، یکی از محبوب‌ترین زبان‌های برنامه‌نویسی حال حاضر دنیاست. یکی از دلایلش می‌تواند این باشد که نسبت به دیگر زبان‌ها، در نحوه نوشتن (syntax)، محدودیت و سخت‌گیری ندارد.

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

// Can you guess the return value?
const variable = (a, b) => {
  return a + b;
}

آیا می‌توانید مقدار خروجی این قطعه کد را حدس بزنید؟

من که نمی‌توانم!

واقعا در جاوا‌اسکریپت نمی‌توان خروجی این کد را حدس زد. زیرا این تابع می‌تواند هر مقداری از انواع مختلف داده را دریافت کند. رشته (String)، عدد (Number) و غیره.

1. اگر مقادیر ورودی عدد باشند:

// Arguments is number.
const variable = (a, b) => {
    return a + b;
  }
  const result = variable(2021, 9);
  console.log(result); // 2030

2. اگر مقادیر ورودی رشته باشند:

  // Arguments is string.
const variable = (a, b) => {
    return a + b;
  }
  const result = variable("2021", "9");
  console.log(result); // 20219

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

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

حالا به این قطعه کد نگاه کنید:

// Can you guess the return value?
const variable = (a: number, b: number): number => {
   return a + b;
}

اکنون می‌توانید حدس بزنید این تابع چه خروجی می‌تواند داشته باشد؟

اگر من هم چیزی نگویم کاملا مشخص است...

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

این نوع ساختار بسیار مفید و کاربردی است؛ زیرا در پروژه‌ها و شرکت‌های بزرگ، خواندن و فهمیدن کدهایی که در گذشته نوشته شده اند، بسیار سخت و پیچیده است.

انواع داده در تایپ‌اسکریپت به زبان ساده

تایپ‌اسکریپت، به صورت پیشفرض چندین نوع داده (data type) اولیه دارد. مثل رشته‌ها، اعداد، boolean، null، undefined و غیره.

در مثال زیر بخشی از data type های موجود در این زبان را با هم بررسی می‌کنیم:

// string, number and boolean.
const caterpie01: number = 2021;    // OK
const caterpie02: number = false;   // NG

const Metapod01: string = "sleepy"; // OK
const Metapod02: string = true;     // NG

const Wartortle01: boolean = true;  // OK
const Wartortle02: boolean = 1111;  // NG

با اجرای این قطعه کد، در console اخطارهای زیر را می‌بینیم:

typescript.ts:10:7 - error TS2322: Type 'boolean' is not assignable to type 'number'.

10 const caterpie02: number = false;   // NG
         ~~~~~~~~~~

typescript.ts:13:7 - error TS2322: Type 'boolean' is not assignable to type 'string'.

13 const Metapod02: string = true;     // NG
         ~~~~~~~~~

typescript.ts:16:7 - error TS2322: Type 'number' is not assignable to type 'boolean'.

16 const Wartortle02: boolean = 1111;  // NG
         ~~~~~~~~~~~

اما درباره مقادیر null و undefined چطور؟

// null and undefined.
const Butterfree: null = null;
const ButterfreeNull: string = Butterfree;
console.log(ButterfreeNull) // null

const Kakuna: undefined = undefined;
const KakunaNull: string = Kakuna;
console.log(KakunaNull) //undefined

این قطعه کد اجرا می‌شود و مشکلی ندارد. ما می‌توانیم به یک رشته (string) مقادیر null و undefined بدهیم.

اما نکته ای که باید به آن توجه کنیم این است که، هنوز حالت strict mode در تایپ‌اسکریپت را فعال نکردیم.

(Strict Mode در تایپ‌اسکریپت به این صورت عمل می‌کند که حساسیت بررسی "انواع داده‌ها" را بالا برده و ما نمی‌توانیم حتی مقادیر null و undefined را به یک نوع داده‌ی دیگر بدهیم.)

زمانی که حالت strict mode را true قرار دهیم، در هنگام اجرای کد به اخطار زیر بر می‌خوریم:

typescript.ts:21:7 - error TS2322: Type 'null' is not assignable to type 'string'.

21 const ButterfreeNull: string = Butterfree;
         ~~~~~~~~~~~~~~

typescript.ts:25:7 - error TS2322: Type 'undefined' is not assignable to type 'string'.

25 const KakunaNull: string = Kakuna;

حالت strict mode در تنظیمات خود تایپ اسکریپت و در فایل tsconfig.json یا همچنین توسط دستور strict— قابل تغییر و تنظیم است.

اگر تا به حال این تنظیمات را انجام نداده اید، از این لینک وارد وبسایت اصلی تایپ‌اسکریپت شده و شیوه انجام آن را مرور کنید.

نوع داده‌ی ANY در تایپ‌اسکریپت به چه معنا است؟

تایپ‌اسکریپت یک "نوع داده" به نام any دارد. این نوع، به ما اجازه می‌دهد که از همه نوع داده‌ها، بدون خطا استفاده کنیم. این نوع داده، شبیه چیزی است که در جاوا‌اسکریپت داریم.

به قطعه کد زیر نگاه کنید:

// any data type
let birthday: any = 1991;
console.log(typeof birthday) // number

birthday = "bird";
console.log(typeof birthday) // string

birthday = false;
console.log(typeof birthday) // boolean

birthday = null;
console.log(typeof birthday) // object

birthday = undefined;
console.log(typeof birthday) // undefined

متغیر "birthday" می‌تواند هر نوع داده‌ای داشته باشد.

به نوعی می‌توان گفت، زمانی که از نوع داده‌ی any استفاده می‌کنیم، اصلا از typescript استفاده نمی‌کنیم! بلکه در حال نوشتن همان کدهای جاوااسکریپت هستیم.

*این بدیهی است، زیرا همانطور که از اسم typescript پیداست، نوعی از اسکریپت نویسی است که تمرکز ویژه‌ای بر تایپ‌ها دارد.

حتی اگر هم مشخص نکنیم، تایپ‌اسکریپت می‌تواند نوع داده‌ها را تشخیص دهد.

مثال زیر را با هم ببینیم:

// typescript can guess data types.
const caterpie01: number = 2021;     // number   - we tell it
const caterpie001 = 2021;            // number   - typescript guess

const Metapod01: string = "sleepy";  // string   - we tell it
const Metapod001 = "sleepy";         // string   - typescript guess

const Wartortle01: boolean = true;   // boolean  - we tell it
const Wartortle001 = true;           // boolean  - typescript guess

یقینا این روش (که دیگر نیازی نیست ما نوع داده را مشخص کنیم و خود تایپ‌اسکریپت متوجه می‌شود) کوتاه‌تر است. همچنین در این روش هم ما نمی‌توانیم داده از نوع دیگری را به این متغیر بدهیم.

let caterpie001 = 2021;            // number
caterpie001 = "text";              // type error

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

مثال زیر را ببینیم:

const pikachu = (a, b): number => {
  return a + b;
}
pikachu(2021, 9);

در این مثال به این خطا بر می‌خوریم:

typescript.ts:57:18 - error TS7006: Parameter 'a' implicitly has an 'any' type.

57 const pikachu = (a, b): number => {
                    ~

typescript.ts:57:21 - error TS7006: Parameter 'b' implicitly has an 'any' type.

57 const pikachu = (a, b): number => {

*البته این نکته را نباید فراموش نکنیم که در این لحظه strict mode فعال و true است. اگر این حالت رو false قرار دهیم، کامپایل با موفقیت انجام شده و به ما خطایی نمی‌دهد.

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

const pikachu = (a: number, b: number): number => {
  return a + b;
}

توصیه می‌شود که از نوع داده any فقط در صورت لزوم استفاده کنیم. برای مثال زمانی که بعضی از کدها را از JavaScript به TypeScript انتقال می‌دهیم.

نوع داده‌ی Object در تایپ اسکریپت

تایپ‌اسکریپت نوع داده‌ی آبجکت را با استفاده از interface ایجاد می‌کند.

قبل از توضیح کامل، به مثال زیر نگاهی می‌اندازیم:

// define object data type with interface.
interface UserObj {
  name: string,
  age: number,
  skill: string
}
// assign data type to object.
const user: UserObj = {
  name: "pikachu",
  age: 6,
  skill: "Developer"
}

ما با استفاده از سینتکس interface می‌توانیم یک نوع داده‌ی آبجکت بسازیم و سپس آن را به یک آبجکت نسبت بدهیم.

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

همانند مثال زیر:

// define object data type with interface.
interface UserObj{
  name: string,
  age: number,
  skill: string
}
// assign data type to object.
const user: UserObj = {
  name: "pikachu",
  age: "change age",       // change 
  skill: "Developer
}

خطا:

typescript.ts:75:3 - error TS2322: Type 'string' is not assignable to type 'number'.

75   age: "change age",
     ~~~

  typescript.ts:69:3
    69   age: number,
         ~~~
    The expected type comes from property 'age' which is declared here on type 'UserObj'

خیلی خوب است اگر نوع داده یک آبجکت را با استفاده از interface تعریف کنیم. اما این را هم باید بدانیم که به راحتی می‌توان data type یک Object را به صورت مستقیم و بدون استفاده از interface هم تعریف کرد.

// assign data type directly to object.
const user: {name: string, age: number, skill: string} = {
  name: "pikachu",
  age: 6,
  skill: "Developer"
}

نوع داده‌ی Array در تایپ اسکریپت

نحوه تعریف یک آرایه در تایپ‌اسکریپت به صورت زیر است:

// define array data type
const userName: string[] = ["pikachu", "Raichu", "Charizard"];

اما اگر در این مورد نوع داده‌ی این آرایه را تغییر بدیم... :

/ change array data type
const userName: string[] = ["pikachu", "Raichu", false];

با خطای زیر مواجه می‌شویم:

typescript.ts:80:49 - error TS2322: Type 'boolean' is not assignable to type 'string'.

80 const userName: string[] = ["pikachu", "Raichu", false];

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

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

// defined array with another way.
const userName: Array<string> = ["pikachu", "Raichu", "Charizard"];

نوع داده‌ی generics در تایپ اسکریپت چیست؟

Data type بعدی، نوع داده‌ی generics در تایپ‌اسکریپت است. همانطور که از اسم این نوع داده مشخص است، یک انواع داده‌ی عمومی است. 

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

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

مثال زیر کاملا گویای این نوع داده است:

// defined array with generics data type.
type UserName<T> = T[];
// After defined generics type, we can define specific data type.
const userName: UserName<string> = ["pikachu", "Raichu", "Charizard"];

// Above code is the same as this.
const userName: string[] = ["pikachu", "Raichu", "Charizard"];

در تایپ‌اسکریپت با استفاده از T می‌توانیم نوع داده‌های generics را تعریف کنیم.

مثال زیر زیاد خوب نیست و نباید از این نسبت دادن داده‌ها به متغیرها استفاده کرد. اما فقط به عنوان مثال، می‌تواند مفهوم انواع داده‌های generics را به خوبی برساند. این مثال را با هم بررسی می‌کنیم:

// defined array with generics data type.
type user<T> = T[];
// After defined generics type, we can define specific data type.
const user01: user<string> = ["pikachu", "Raichu", "Charizard"];
const user02: user<number> = [6, 14, 16];
const user03: user<boolean> = [true, true, false];

این مثال نشان می‌دهد که بعد از تعریف متغیر user به عنوان یک داده‌ی generics، این متغیر می‌تواند هر نوعی را دریافت کند. اما بعد از تعیین اولین نوع داده‌ی این متغیر، نمی‌تواند نوع دیگری بپذیرد و به خطا می‌خورد. و به نوعی این رفتار در TypeScript طبیعی است.

نوع داده‌ی Union در تایپ اسکریپت چیست؟

نوع داده union به این صورت است که می‌توانیم چندین نوع داده تعریف کنیم. مثال ساده‌ی زیر به درک بهتر این نوع داده کمک می‌کند و با هم آن را بررسی می‌کنیم:

let userName: (string | number) = "pikachu";  // OK
userName = 6; 

این قطعه کد به خوبی و بدون خطا اجرا می‌شود. زیرا متغیر userName می‌تواند هر دو نوع داده‌ی Number و String را داشته باشد. پس Union به این معناست که یک متغیر می‌تواند چندین نوع داده را بپذیرد.

اما کد زیر درست نیست و به خطا برمی‌خوریم:

let userName: (string | number) = "pikachu";
userName = 6;
userName = false;      // NG

این بدیهی است. زیرا این متغیر نمی‌تواند نوع داده‌ی Boolean را بپذیرد.

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

مثال زیر را با هم ببینیم:

// define data type with array and union
let userName: (string | number)[] = ["pikachu", "Raichu", 6, 14];

این قطعه کد کاملا درست است. اما زمانی که در این آرایه یک نوع داده‌ی دیگر اضافه شود، به خطای type error برمی‌خوریم:

// define data type with array and union
let userName: (string | number)[] = ["pikachu", "Raichu", 6, 14, false];

خطا:

typescript.ts:105:65 - error TS2322: Type 'boolean' is not assignable to type 'string | number'.

105 let userName: (string | number)[] = ["pikachu", "Raichu", 6, 14, false];

نوع داده‌ی تاپل (Tuple) در تایپ اسکریپت چیست؟

Tuple یک نوع داده‌ی خیلی محدود است. برای درک بهتر این نوع داده، در ابتدا مثال زیر را درنظر می‌گیریم:

let userName: [string, number] = ["pikachu", 6];

در این مثال، نوع داده‌ی tuple اجازه می‌دهد که تنها دو المان با نوع داده‌های String (تنها در سمت چپ) و Number (تنها در سمت راست)  تعریف کنیم. این نوع داده را می‌توان با [ ] مشخص و استفاده کرد.

چندین مثال که منجر به خطا می‌شوند را با هم بررسی می‌کنیم:

مثال1:

let userName1: [string, number] = [6, "pikachu"];

خطا:

typescript.ts:109:36 - error TS2322: Type 'number' is not assignable to type 'string'.

109 let userName1: [string, number] = [6, "pikachu"];      // NG
                                       ~

typescript.ts:109:39 - error TS2322: Type 'string' is not assignable to type 'number'.

109 let userName1: [string, number] = [6, "pikachu"];      // NG
                                         ~~~~~~~~~

مثال2:

let userName2: [string, number] = ["pikachu", "text"]; // NG

خطا:

typescript.ts:110:47 - error TS2322: Type 'string' is not assignable to type 'number'.

110 let userName2: [string, number] = ["pikachu", "text"]; // NG
                                                 ~~~~~~

مثال3:

let userName3: [string, number] = ["pikachu", 6, 14];  // NG

خطا:

typescript.ts:111:5 - error TS2322: Type '[string, number, number]' is not assignable to type '[string, number]'.
  Source has 3 element(s) but target allows only 2.

111 let userName3: [string, number] = ["pikachu", 6, 14];  // NG

با بررسی مثال‌های بالا، کاملا واضح است که نوع داده‌ی Tuple در تایپ اسکریپت در این آرایه چه کاربردی دارد. برای توضیح بیشتر، این آرایه با نوع داده‌ی Tuple مشخص شده و می‌تواند تنها دو المان در آرایه داشته باشد. اولین مقدار در آرایه باید از نوع string و دومین مقدار در آرایه باید از نوع number باشد.

نتیجه گیری

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

اگر در شرکت‌های بزرگ مشغول به کار هستید و در پروژه‌های خود خط‌های زیادی کد دارید، باید زمان خیلی زیادی را صرف خواندن و فهمیدن این کدها کنید. در صورتی که TypeScript به خواندن و فهمیدن این کدها کمک کرده و آن‌ها را خواناتر می‌کند.

ما در این مقاله سعی کردیم با بیانی ساده و همراه با مثال، انواع داده در تایپ اسکریپت را برایتان توضیح دهیم. این مقاله شامل مباحث اولیه و ابتدایی از تایپ اسکریپت بود.

 

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

مرسی از شما بابت وقتی که برای خواندن این مقاله گذاشتید. 😊

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