درس سیزدهم – واسطها (Interfaces)

درس سیزدهم – واسطها (Interfaces)

 در این درس با واسطها در زبان C# آشنا خواهیم شد. اهداف این درس بشرح زیر می‌باشند :

1-     آشنایی با مفهوم کلی واسطها

2-     تعریف یک واسط

3-     استفاده از یک interface

4-     پیاده‌سازی ارث‌بری در interface ها

5- نکات مهم و پیشرفته

6- مثالی کاربردی از واسطها

7- منابع مورد استفاده

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

 حال باید دید چرا با توجه به اینکه interface ها دارای پیاده‌سازی نیستند مورد استفاده قرار می‌گیرند یا بهتر بگوئیم سودمندی استفاده از interface ها در چیست؟ تصور کنید که در یک برنامه با مولفه‌هایی سروکار دارید که متغیرند ولی دارای فیلدها یا متدهایی با نامهای یکسانی هستند و باید نام این متدها نیز یکسان باشد. با استفاده از یک interface مناسب می‌توان تنها متدها و یا فیلدهای مورد نظر را اعلان نمود و سپس کلاسها و یا property های مورد از آن interface ارث‌بری نمایند. در این حالت تمامی کلاسها و property ها دارای فیلدها و یا متدهایی همنام هستند ولی هر یک پیاده‌سازی خاصی از آنها را اعمال می‌نمایند.

 نکته مهم دیگر درباره interface ها، استفاده و کاربرد آنها در برنامه‌های بزرگی است که برنامه‌ها و یا اشیاؤ مختلفی در تماس و تراکنش (transact) هستند. تصور کنید کلاسی در یک برنامه با کلاسی دیگر در برنامه‌ای دیگر در ارتباط باشد. فرض کنید این کلاس متدی دارد که مقداری از نوع int بازمیگرداند. پس از مدتی طراح برنامه به این نتیجه می‌رسد که استفاده از int پاسخگوی مشکلش نیست و باید از long استفاده نماید. حال شرایط را در نظر بگیرید که برای تغییر یک چنین مسئله ساده‌ای چه مشکل بزرگی پیش خواهد آمد. تمامی فیلدهای مورتبط با این متد باید تغییر داده شوند. در ضمن از مسئله side effect نیز نمی‌توان چشم پوشی کرد.( تاثیرات ناخواسته و غیر منتظره و یا به عبارتی پیش بینی نشده که متغیر یا فیلدی بر روی متغیر یا فیلدی دیگر اعمال می‌کند، در اصطلاح side effect گفته می‌شود.) حال فرض کنید که در ابتدا interface ای طراحی شده بود. درصورت اعمال جزئیترین تغییر در برنامه مشکل تبدیل int به long قابل حل بود، چراکه کاربر یا برنامه و در کل user برنامه در هنگام استفاده از یک interface با پیادهسازی پشت پرده آن کاری ندارد و یا بهتر بگوئیم امکان دسترسی به آن را ندارد. از اینرو اعمال تغییرات درون آن تاثیری بر رفتار کاربر نخواهد داشت و حتی کاربر از آن مطلع نیز نمی‌شود.  در مفاهیم کلی شیء گرایی، interface ها یکی از مهمترین و کاربردی ترین اجزاء هستند که در صورت درک صحیح بسیار مفید واقع می‌شوند. یکی از مثالهای مشهود درباره interface ها (البته در سطحی پیشرفته تر و بالاتر) رابطهای کاربر گرافیکی (GUI) هستند. کاربر تنها با این رابط سروکار دارد و کاری به نحوه عملیات پشت پرده آن ندارد و اعمال تغییرات در پیاده‌سازی interface کاربر را تحت تاثیر قرار نمی‌دهد.

 از دیدگاه تکنیکی، واسطها بسط مفهومی هستند که از آن به عنوان انتزاع (Abstract) یاد می‌کنیم. در کلاسهای انتزاعی (که با کلمه کلید abstract مشخص می‌شدند.) سازندة کلاس قدر بود تا فرم کلاس خود را مشخص نماید : نام متدها، نوع بازگشتی آنها و تعداد و نوع پارامتر آنها، اما بدون پیاده‌سازی بدنه متد. یک interface همچنین می‌تواند دارای فیلدهایی باشد که تمامی آنها static و final هستند. یک interface تنها یک فرم کلی را بدون پیاده‌سازی به نمایش می‌گذارد.

 از این دیدگاه، یک واسط بیان می‌دارد که : " این فرم کلی است که تمامی کلاسهایی که این واسط را پیاده‌سازی می‌کنند، باید آنرا داشته باشند." از سوی دیگر کلاسها و اشیاء دیگری که از کلاسی که از یک واسط مشتق شده استفاده می‌کنند، می‌دانند که این کلاس حتماً تمامی متدها و اعضای واسط را پیاده‌سازی می‌کند و می‌توانند به راحتی از آن متدها و اعضا استفاده نمایند. پس به طور کلی می‌توانیم بگوئیم که واسطها بمنظور ایجاد یک پروتکل (protocol) بین کلاسها مورد استفاده قرار می‌گیرند. (همچنان که برخی از زبانهای برنامه‌سازی بجای استفاده از کلمه کلیدی interface از protocol استفاده می‌نمایند.)

  به دلیل اینکه کلاسها و ساختارهایی که از interface ها ارث‌بری می‌کنند موظف به پیاده‌سازی و تعریف آنها هستند، قانون و قاعده‌ای در این باره ایجاد می‌گردد. برای مثال اگر کلاس A از واسط IDisposable ارث‌بری کند، این ضمانت بوجود می‌آید که کلاس A دارای متد Dispose() است، که تنها عضو interface نیز می‌باشد. هر کدی که می‌خواهد از کلاس A استفاده کند، ابتدا چک می‌نماید که آیا کلاس A واسط IDisposable را پیاده‌سازی نموده یا خیر. اگر پاسخ مثبت باشد آنگاه کد متوجه می‌شود که می‌تواند از متد A.Dispose() نیز استفاده نماید. در زیر نحوه اعلان یک واسط نمایش داده شده است.

 

interface IMyInterface

{

void MethodToImplement();

}

 

در این مثال نحوه اعلان واسطی با نام IMyInterface نشان داده شده است. یک قاعده (نه قانون!) برای نامگذاری واسطها آنست که نام واسطها را با "I" آغاز کنیم که اختصار کلمه interface است. در interface این مثال تنها یک متد وجود دارد. این متد می‌توان هر متدی با انواع مختلف پارامترها و نوع بازگشتی باشد. توجه نمایید همانطور که گفته شد این متد دارای پیاده‌سازی نیست و تنها اعلان شده است.  نکته دیگر که باید به ان توجه کنید آنست که این متد به جای داشتن {} به عنوان بلوک خود، دارای ; در انتهای اعلان خود می‌باشد. علت این امر آنست که interface تنها نوع بازگشتی و پارامترهای متد را مشخص می‌نماید و کلاس یا شی‌ای که از آن ارث می‌برد باید آنرا پیاده‌سازی نماید. مثال زیر نحوه استفاده از این واسط را نشان می‌دهد.

 مثال 1-13 : استفاده از واسطها و ارث‌بری از آنها

class InterfaceImplementer : IMyInterface

{

static void Main()

{

InterfaceImplementer iImp = new InterfaceImplementer();

iImp.MethodToImplement();

}

public void MethodToImplement()

{

Console.WriteLine("MethodToImplement() called.");

}

}

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

 مثال 2-13 : ارث‌بری واسطها از یکدیگر

using System;

 

interface IParentInterface

{

void ParentInterfaceMethod();

}

 

interface IMyInterface : IParentInterface

{

void MethodToImplement();

}

 

class InterfaceImplementer : IMyInterface

{

static void Main()

{

InterfaceImplementer iImp = new InterfaceImplementer();

iImp.MethodToImplement();

iImp.ParentInterfaceMethod();

}

 

public void MethodToImplement()

{

Console.WriteLine("MethodToImplement() called.");

}

 

public void ParentInterfaceMethod()

{

Console.WriteLine("ParentInterfaceMethod() called.");

}

}

 

مثال 2-13 دارای 2 واسط است : یکی IMyInterface و واسطی که از آن ارث می‌برد یعنی IParentInterface. هنگامیکه واسطی از واسط دیگری ارث‌بری می‌کند، کلاس یا ساختاری که این واسطها را پیاده‌سازی می‌کند، باید تمامی اعضای واسطهای موجود در سلسله مراتب ارث‌بری را پیاده‌سازی نماید. در مثال 2-13، چون کلاس InterfaceImplementer از واسط IMyInterface ارث‌بری نموده، پس از واسط IParentInterface نیز ارث‌بری دارد، از اینرو باید کلیه اعضای این دو واسط را پیاده‌سازی نماید.

 چند نکته مهم :

       1-     با استفاده از کلمه کلید interface در حقیقت یک نوع مرجعی (Reference Type) جدید ایجاد نموده‌اید.

2-   از لحاظ نوع  ارتباطی که واسطها و کلاسها در ارث‌بری ایجاد می‌نمایند باید به این نکته اشاره کرد که، ارث‌بری از کلاس رابطه "است" یا "بودن" (is-a relation) را ایجاد می‌کند (ماشین یک وسیله نقلیه است) ولی ارث‌بری از یک واسط یا interface نوع خاصی از رابطه، تحت عنوان "پیاده‌سازی" (implement relation) را ایجاد می‌کند. ("می‌توان ماشین را با وام بلند مدت خرید" که در این جمله ماشین می‌تواند خریداری شدن بوسیله وام را پیاده‌سازی کند.)

3-     فرم کلی اعلان interface ها بشکل زیر است :

[attributes] [access-modifier] interface interface-name [:base-list]{interface-body}

      که در اعضای آن بشرح زیر می باشند :

attributes : صفتهای واسط

access-modifiers : private   یا public سطح دسترسی به واسط از قبیل

interface-name : نام واسط

:base-list : لیست واسطهایی که این واسط آنها را بسط می‌دهد.

Interface-body : بدنه واسط که در آن اعضای آن مشخص می‌شوند

      توجه نمایید که نمی‌توان یک واسط را بصورت virtual اعلان نمود.

4-     هدف از ایجاد یک interface تعیین توانائیهاییست که می‌خواهیم در یک کلاس وجود داشته باشند.

5-     به مثالی در زمینه استفاده از واسطها توجه کنید :

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

در این واسط دو متد Read() و Write() وجود دارند که در بدنه واسط تعریف می‌شوند ک

interface IStorable

{

void Read( );

void Write(object);

}

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

public class Document : IStorable

{

public void Read( ) {...}

public void Write(object obj) {...}

// ...

}

حال بعنوان طراح برنامه، شما وظیفه داری تا به پیاده‌سازی این واسط بپردازید، بطوریکه کلیه نیازهای شما را برآورده نماید. نمونه‌ای از این پیاده‌سازی در مثال 3-13 آورده شده است. 

مثال 3-13 : پیاده‌سازی واسط و ارث‌بری – مثال کاربردی

using System;

 

// interface اعلان

interface IStorable

{

void Read( );

void Write(object obj);

int Status { get; set; }

}

 

public class Document : IStorable

{

public Document(string s)

{

Console.WriteLine("Creating document with: {0}", s);

}

 

public void Read( )

{

Console.WriteLine("Implementing the Read Method for IStorable");

}

 

public void Write(object o)

{

Console.WriteLine("Implementing the Write Method for IStorable");

}

 

public int Status

{

get

{

return status;

}

set

{

status = value;

}

}

private int status = 0;

}

 

public class Tester

{

static void Main( )

{

Document doc = new Document("Test Document");

doc.Status = -1;

doc.Read( );

Console.WriteLine("Document Status: {0}", doc.Status);

IStorable isDoc = (IStorable) doc;

isDoc.Status = 0;

isDoc.Read( );

Console.WriteLine("IStorable Status: {0}", isDoc.Status);

}

}

 

                خروجی برنامه نیز بشکل زیر است :

Output:

Creating document with: Test Document

Implementing the Read Method for IStorable

Document Status: -1

Implementing the Read Method for IStorable

IStorable Status: 0

 

6-   در مثال فوق توجه نمایید که برای متدها واسط IStorable هیچ سطح دسترسی (public,private و ...) در نظر گرفته نشده است. در حقیقت تعیین سطح دسترسی باعث ایجاد خطا می‌شود چراکه هدف اصلی از ایجاد یک واسط ایجاد شیء است که تمامی اعضای آن برای تمامی کلاسها قابل دسترسی باشند.

7-     توجه نمایید که از روی یک واسط نمی‌توان نمونه‌ای جدید ایجاد کرد بلکه باید کلاسی از آن ارث‌بری نماید.

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

خلاصه :

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

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

درس دوازدهم – ساختارها در C# (Struct)

درس دوازدهم – ساختارها در C# (Struct)

 

در این درس با ساختارها (Struct) در زبان C# آشنا می‌شویم. اهداف این درس بشرح زیر می‌باشند

  • یک struct یا ساختار (Structure) چیست؟
  • پیاده‌سازی ساختارها(Struct)
  • استفاده از ساختارها(Struct)
  • نکات مهم و مطالب کمکی دربارة struct ها

 

ساختار (struct) چیست؟

همانطور که با استفاده از کلاسها می‌توان انواع (types) جدید و مورد نظر را ایجاد نمود، با استفاده از struct ها می‌توان انواع مقداری (value types) جدید و مورد نظر را ایجاد نمود. از آنجائیکه struct ها بعنوان انواع مقداری در نظر گرفته می‌شوند، از اینرو تمامی اعمال مورد استفاده بر روی انواع مقداری را می‌توان برای struct ها در نظر گرفت. struct ها بسیار شبیه به کلاس‌ها هستند و می‌توانند دارای فیلد، متد و property باشند. عموماً ساختارها مجموعه کوچکی از عناصری هستند که منطقی با یکدیگر دارای رابطه می‌باشند. برای نمونه می‌توان به ساختار Point موجود در Framework SDK اشاره کرد که حاوی دو property با نامهای X و Y است.

 

با استفاده از ساختارها (struct) می‌توان اشیایی با انواع جدید ایجاد کرد که این اشیاء می‌توانند شبیه به انواع موجود (int, float, …) باشند. حال سوال اینست که چه زمانی از ساختارها(struct)  بجای کلاس استفاده می‌کنیم؟ در ابتدا به نحوه استفاده از انواع موجود در زبان ‍C# توجه نمایید. این انواع دارای مقادیر و عملگرهای معینی جهت کار با این مقادیر هستند. حال اگر نیاز به شی‌ای دارید که همانند این انواع رفتار نمایند لازم است تا از ساختارها (struct) استفاده نمایید. در ادامه این مبحث نکات و قوانینی را ذکر می‌کنیم که با استفاده از آنها بهتر بتوانید از ساختارها (struct) استفاده نمایید.

 

اعلان و پیاده‌سازی struct

برای اعلان یک struct کافیست تا با استفاده از کلمه کلیدی struct که بدنبال آن نام مورد نظر برای ساختار آمده استفاده کرد. بدنة ساختار نیز بین دو کروشة باز و بسته {} قرار خواهد گرفت. به مثال زیر توجه نمایید :

 

مثال 1-12 : نمونه‌ای از یک ساختار (Struct)

using System;

struct Point

{

public int x;

public int y;

public Point(int x, int y)

{

this.x = x;

this.y = y;

}

public Point Add(Point pt)

{

Point newPt;

newPt.x = x + pt.x;

newPt.y = y + pt.y;

return newPt;

}

}

///

/// struct مثالی از اعلان و ساخت یک

///

class StructExample

{

static void Main(string[] args)

{

Point pt1 = new Point(1, 1);

Point pt2 = new Point(2, 2);

Point pt3;

pt3 = pt1.Add(pt2);

Console.WriteLine("pt3: {0}:{1}", pt3.x, pt3.y);

}

}

 

مثال 1-12 نحوة ایجاد و استفاده از struct را نشان می‌دهد. به راحتی می‌توان گفت که یک نوع(type)  ، یک struct است، زیرا از کلمه کلیدی struct در اعلان خود بهره می‌گیرد. ساختار پایه‌ای یک ساختار پایه‌ای یک struct بسیار شبیه به یک کلاس است، ولی تفاوتهایی با آن دارد که این تفاوتها در پاراگراف بعدی مورد بررسی قرار می‌گیرند. ساختار Point دارای سازنده ایست که مقادیر داده شده با آنرا به فیلدهای x و y تخصیص می‌دهد. این ساختار همچنین دارای متد Add() می‌باشد که ساختار Point دیگری را دریافت می‌کند و آنرا به struct کنونی می‌افزاید و سپس struct جدیدی را باز می‌گرداند.

 

توجه نمایید که ساختار Point جدیدی درون متد Add() تعریف شده است. توجه کنید که در اینجا همانند کلاس، نیازی به استفاده از کلمه کلیدی new جهن ایجاد یک شیء جدید نمی‌باشد. پس از آنکه نمونة جدیدی از یک ساختار ایجاد شد، سازندة پیش فرض (یا همان سازندة بدون پارامترش)   برای آن در نظر گرفته می‌شود. سازندة بدون پارامتر کلیه مقادیر فیلدهای ساختار را به مقادیر پیش فرض تغییر می‌دهد. بعنوان مثال فیلدهای صحیح به صفر و فیلدهای Boolean به false تغییر می‌کنند. تعریف سازندة بدون پارامتر برای یک ساختار صحیح نمی‌باشد. (یعنی شما نمی‌توانید سازندة بدون پارامتر برای یک struct تعریف کنید.)

 

ساختارها (structs) با استفاده از عملگر new نیز قابل نمونه‌گیری هستند (هر چند نیازی به استفاده از این عملگر نیست.) در مثال 1-12 pt1 و pt2 که ساختارهایی از نوع Point هستند، با استفاده از سازندة موجود درون ساختار Point مقداردهی می‌شوند. سومین ساختار از نوع Point، pt3 است و از سازندة بدون پارامتر استفاده می‌کند زیرا در اینجا مقدار آن اهمیتی ندارد. سپس متد Add() از ساختار pt1 فراخوانده می‌شود و ساختار pt2 را بعنوان پارامتر دریافت می‌کند. نتیجه به pt3 تخصیص داده می‌شود، این امر نشان می‌دهد که یک ساختار می‌تواند همانند سایر انواع مقداری مورد استفاده قرار گیرد. خروجی مثال 1-12 در زیر نشان داده شده است :

pt3 : 3 : 3

 

یکی دیگر از تفاوتهای ساختار و کلاس در اینست که ساختارها نمی‌توانند دارای تخریب کننده (deconstructor) باشند. همچنین ارث‌بری در مورد ساختارها معنی ندارد. البته امکان ارث‌بری بین ساختارها و interface ها وجود دارد. یک interface نوع مرجعی زبان C# است که دارای اعضایی بدون پیاده‌سازی است. هر کلاس و یا ساختاری که از یک interface ارث‌بری نماید باید تمامی متدهای آنرا پیاده‌سازی کند. دربارة interface ها در آینده صحبت خواهیم کرد.

 

خلاصه :

هم اکنون شما با چگونگی ایجاد یک ساختار آشنا شدید. هنگامیکه قصد دارید نوعی را بصورت ساختار یا کلاس پیاده‌سازی کنید، باید به این نکته توجه کنید که این نوع چگونه مورد استفاده قرار می‌گیرد. اگر می‌خواهید سازنده‌ای بدون پارامتر داشته باشید، در اینصورت کلاس تنها گزینه شماست. همچنین توجه نمایید از آنجائیکه یک ساختار بعنوان یک نوع مقداری در نظر گرفته می‌شود، در پشته (Stack) ذخیره می‌شود و حال آنکه کلاس در heap ذخیره می‌گردد.

 

نکات مهم و مطالب کمکی

 

  1. تفاوتهای اصلی بین کلاس و ساختار در چیست؟

همانطور که بطور مختصر در بالا نیز اشاره شد، از نظر نوشتاری (syntax) struct و کلاس بسیار شبیه به یکدیگر هستند اما دارای تفاوتهای بسیار مهمی با یکدیگر می‌باشند.

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

struct Time
{
    public Time() { ... } // خطای زمان کامپایل رخ می‌دهد
   
}

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

یکی دیگر از تفاوتهای بین کلاس و struct در آن است که، اگر در کلاس برخی از فیلدهای موجود در  سازندة کلاس را مقداردهی نکنید، کامپایلر مقدار پیش فرض صفر، false و یا null را برای آن فیلد در نظر خواهد گرفت ولی در struct تمامی فیلدهای سازنده باید بطور صریح مقداردهی شوند و درصورتیکه شما فیلدی را مقداردهی نکید کامپایلر هیچ مقداری را برای آن در نظر نخواهدگرفت و خطای زمان کامپایل رخ خواهد داد. بعنوان مثال در کد زیر اگر Time بصورت کلاس تعریف شده بود خطایی رخ نمی‌داد ولی چون بصورت struct تعریف شده خطای زمان کامپایل رخ خواهد داد :

struct Time
{
    public Time(int hh, int mm) 
    {
        hours = hh;
        minutes = mm;
    }   // خطای زمان کامپایلی بدین صورت رخ می‌دهد : seconds not initialized
   
    private int hours, minutes, seconds;
}
 

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

 

struct Time

{

   

    private int hours = 0; // خطای زمان کامپایل رخ می‌دهد

    private int minutes;

    private int seconds;

}

 

آخرین تفاوت بین کلاس و struct که ما به آن خواهیم پرداخت در مورد ارث‌بری است. کلاسها می‌توانند از کلاس پایة خود ارث‌بری داشته باشند در حالیکه ارث‌بری در struct ها معنایی ندارد و یک struct تنها می‌تواند از واسطها (interface) ارث‌بری داشته باشد.

 

  1. پس از ایجاد یک  ساختار چگونه می‌توان از آن استفاده نمود؟

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

struct Time 
   
    private int hours, minutes, seconds;
}
 
class Example
{
    public void Method(Time parameter) 
    {
        Time localVariable;
       
    }
    private Time field;
}

 

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

 

درس یازدهم – اندیکسرها در C# (Indexers)

درس یازدهم – اندیکسرها در C# (Indexers)

 

در این درس با اندیکسرها در C# آشنا می‌شویم. اهداف این درس به شرح زیر می‌باشند :

  • پیاده‌سازی اندیکسر
  • سرریزی اندیکسرها (Overload)
  • درک چگونگی پیاده‌سازی اندیکسرهای چند پارامتری
  • خلاصه
  • نکات مهم و مطالب کمکی در زمینه اندیکسرها

 

اندیکسرها

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

 

مثال 11-1 : نمونه‌ای از یک اندیکسر

using System;

///

/// مثالی ساده از یک اندیکسر

///

class IntIndexer

{

private string[] myData;

public IntIndexer(int size)

{

myData = new string[size];

for (int i=0; i < size; i++)

{

myData[i] = "empty";

}

}

public string this[int pos]

{

get

{

return myData[pos];

}

set

{

myData[pos] = value;

}

}

static void Main(string[] args)

{

int size = 10;

IntIndexer myInd = new IntIndexer(size);

myInd[9] = "Some Value";

myInd[3] = "Another Value";

myInd[5] = "Any Value";

Console.WriteLine(" Indexer Output ");

for (int i=0; i < size; i++)

{

Console.WriteLine("myInd[{0}]: {1}", i, myInd[i]);

}

}

}

مثال 11-1 نحوه پیاده‌سازی اندیکسر را نشان می‌دهد. کلاس IntIndexer دارای آرایة رشته‌ای بنام myData می‌باشد. این آرایه، عنصری خصوصی (private) است و کاربران خارجی (external users) نمی‌توانند به آن دسترسی داشته باشند. این آرایه درون سازندة (constructor) کلاس تخصیص‌دهی می‌گردد که در آن پارامتر size از نوع int دریافت می‌شود، از آرایه myData نمونه‌ای جدید ایجاد می‌گردد، سپس هر یک از المانهای آن با کلمه "empty" مقدار‌دهی می‌گردد.

 عضو بعدی کلاس، اندیکسر است که بوسیلة کلمه کلیدی this و دو براکت تعریف شده است، this[int pos]. این اندیکسر پارامتر موقعیتی pos را دریافت می‌نماید. همانطور که حتماً تا کنون دریافته‌اید پیاده‌سازی اندیکسر بسیار شبیه به پیاده‌سازی یک ویژگی (property) است. اندیکسر نیز دارای accessor های set و get است که دقیقاً همانند property عمل می‌کنند. همانطور که در اعلان این اندیکسر نیز مشاهده می‌شود، متغیری از نوع رشته‌ای را باز می‌گرداند.

 در متد Main() شیء جدیدی از IntIndexer ایجاد شده است و مقادیری به آن افزوده می‌شود و سپس نتایج چاپ می‌گردند. خروجی این برنامه به شکل زیر است :

Indexer Output

myInd[0]: empty

myInd[1]: empty

myInd[2]: empty

myInd[3]: Another Value

myInd[4]: empty

myInd[5]: Any Value

myInd[6]: empty

myInd[7]: empty

myInd[8]: empty

myInd[9]: Some Value

 

استفاده از integer جهت دسترسی به آرایه‌ها در اغلب زبانهای برنامه‌سازی رایج است ولی زبان  C# چیزی فراتر از آنرا نیز پشتیبانی می‌کند. در C# اندیکسرها را می‌توان با چندین پارامتر تعریف کرد و هر پارامتر می‌تواند از نوع خاصی باشد. پارامتر‌های مختلف بوسیلة کاما از یکدیگر جدا می‌شوند. پارامترهای مجاز برای اندیکسر عبارتند از : integer، enum و string. علاوه بر آن، اندیکسرها قابل سرریزی (Overload) هستند. در مثال 2-11 تغییراتی در مثال قبل ایجاد کرده‌ایم تا برنامه قابلیت دریافت اندیکسرهای سرریز شده را نیز داشته باشد.

 سرریزی اندیکسرها

مثال 2-11 : اندیکسرهای سرریز شده (Overloaded Indexers)

using System;

///

/// پیاده‌سازی اندیکسرهای سرریز شده

///

class OvrIndexer

{

private string[] myData;

private int arrSize;

 

public OvrIndexer(int size)

{

arrSize = size;

myData = new string[size];

for (int i=0; i < size; i++)

{

myData[i] = "empty";

}//end of for

}//end of constructor

 

public string this[int pos]

{

get

{

return myData[pos];

}

set

{

myData[pos] = value;

}

}//end of indexer

 

public string this[string data]

{

get

{

int count = 0;

for (int i=0; i < arrSize; i++)

{

if (myData[i] == data)

{

count++;

}//end of if

}//end of for

return count.ToString();

}//end of get

set

{

for (int i=0; i < arrSize; i++)

{

if (myData[i] == data)

{

myData[i] = value;

}//end of if

}//end of for

}//end of set

}//end of overloaded indexer

 

static void Main(string[] args)

{

int size = 10;

 

OvrIndexer myInd = new OvrIndexer(size);

myInd[9] = "Some Value";

myInd[3] = "Another Value";

myInd[5] = "Any Value";

myInd["empty"] = "no value";

Console.WriteLine(" Indexer Output ");

for (int i=0; i < size; i++)

{

Console.WriteLine("myInd[{0}]: {1}", i, myInd[i]);

}//end of for

Console.WriteLine(" Number of "no value" entries: {0}", myInd["no value"]);

}//end of Main()

}//end of class

 

 مثال 2-11 نحوه سرریز کردن اندیکسر را نشان می‌دهد. اولین اندیکسر که دارای پارامتری از نوع int تحت عنوان pos است دقیقاً مشابه مثال 1-11 است ولی در اینجا اندیکسر جدیدی نیز وجود دارد که پارامتری از نوع string دریافت می‌کند. get accessor اندیکسر جدید رشته‌ای را برمی‌گرداند که نمایشی از تعداد آیتمهایی است که با پارامتر مقداری data مطابقت می‌کند. set accessor مقدار هر یک از مقادیر ورودی آرایه را که مقدارش با پارامتر data مطابقت نماید را به مقداری که به اندیکسر تخصیص داده می‌شود، تغییر می‌دهد.

 رفتار (behavior) اندیکسر سرریز شده که پارامتری از نوع string دریافت می‌کند، در متد Main() نشان داده شده است. در اینجا set accessor مقدار "No value" را به تمام اعضای کلاس myInd که مقدارشان برابر با "empty" بوده است، تخصیص می‌دهد. این accessor از دستور زیر استفاده نموده است : myInd["empty"] = "No value" . پس از اینکه تمامی اعضای کلاس myInd چاپ شدند، تعداد اعضایی که حاوی "No value" بوده‌اند نیز نمایش داده می‌شوند. این امر با استفاده از دستور زیر در get accessor روی می‌دهد : myInd["No value"]. خروجی برنامه بشکل زیر است :

Indexer Output

myInd[0]: no value

myInd[1]: no value

myInd[2]: no value

myInd[3]: Another Value

myInd[4]: no value

myInd[5]: Any Value

myInd[6]: no value

myInd[7]: no value

myInd[8]: no value

myInd[9]: Some Value

Number of "no value" entries: 7

 علت همزیستی هر دو اندیکسر در مثال 2-11 در یک کلاس مشابه، تفاوت اثرگذاری و فعالیت آنهاست. اثرگذاری و تفاوت اندیکسرها از تعداد و نوع پارامترهای موجود در لیست پارامترهای اندیکسر مشخص می‌گردد. در هنگام استفاده از اندیکسرها نیز، کلاس با استفاده از تعداد و نوع پارامترهای اندیکسرها، می‌تواند تشخیص دهد که در یک فراخوانی از کدام اندیکسر باید استفاده نماید. نمونه‌ای از پیاده‌سازی اندیکسری با چند نوع پارامتر در زیر آورده شده است :

public object this[int param1, ..., int paramN]

{

get

{

// process and return some class data

}

set

{

// process and assign some class data

}

}

خلاصه :

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

در آینده و در مباحث پیشرفته‌تر با موارد بیشتری از استفادة اندیکسرها آشنا خواهید شد.

 نکات :

  1. منظور از اندیکسر سرریز شده چیست؟

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

  1. از اندیکسر چگونه مانند آرایه استفاده می‌شود؟

همانطور که در این درس مشاهده کردید دسترسی به عناصر اندیکسر همانند آرایه‌ها با استفاده از یک اندیس صورت می‌پذیرد. با استفاده از این اندیس می‌توان به عنصر مورد نظر کلاس دسترسی پیدا نمود.

  1. یک مثال عملی  استفاده از  اندیکسرها چیست؟

یک نمونة بسیار جالب از استفادة اندیکسرها کنترل ListBox است. (ListBox عنصری است کنترلی که با استفاده از آن لیستی از عناصر رشته‌ای نمایش داده می‌شوند و کاربر با انتخاب یکی از این گزینه‌ها با برنامه ارتباط برقرار می‌کند. در حقیقت این عنصر کنترلی یکی از روشهای دریافت اطلاعات از کاربر است با این تفاوت که در این روش ورودی‌هایی که کاربر می‌تواند وارد نماید محدود شده هستند و از قبل تعیین شده‌اند. نمونه‌ای از یک ListBox قسمت انتخاب نوع فونت در برنامة Word است که در آن لیستی از فونتهای موجود در سیستم نمایش داده می‌شود و کاربر با انتخاب یکی از آنها به برنامه اعلام می‌کند که قصد استفاده از کدام فونت سیستم را دارد.) ListBox نمایشی از ساختمان داده ایست شبیه به آرایه که اعضای آن همگی از نوع string هستند. علاوه بر این این کنترل می‌خواهد تا در هنگام انتخاب یکی از گزینه‌هایش بتواند اطلاعات خود را بطور خودکار update نماید و یا به عبارتی بتواند ورودی دریافت نماید. تمامی این اهداف با استفاده از اندیکسر میسر می‌شود. اندیکسرها شبیه به property ها اعلان می‌شوند با این تفاوت مهم که اندیکسرها بدون نام هستند و نام آنها تنها کلمه کلیدی this است و همین this مورد اندیکس شدن قرار می‌گیرد و سایر موارد بشکل پارامتر به اندیکسر داده می‌شوند.

public class ListBox: Control

{

private string[] items;

 

public string this[int index]

{

get

{

return items[index];

}

set

{

items[index] = value;

Repaint();

}

}

}

با نگاه به نحوه استفاده از اندیکسر بهتر می‌توان با مفهوم آن آشنا شد. برای مثال دسترسی به ListBox بشکل زیر است :

ListBox listBox = ...;

listBox[0] = "hello";

Console.WriteLine(listBox[0]);

 

نمونه برنامه‌ای که در آن نحوة استفاده از اندیکسر در عنصر کنترلی ListBox نشان داده شده، در زیر آورده شده است :

 

Csharp-Persian_Indexer_Demo

 

using System;

 

public class ListBoxTest

{

 // تخصیص داده می‌شوند.ListBoxرشته‌های مورد نظر به

public ListBoxTest(params string[] initialStrings)

{

 // فضایی را برای ذخیره‌سازی رشته‌های تخصیص می‌دهد.

strings = new String[256];

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

foreach (string s in initialStrings)

{

strings[ctr++] = s;

}

}//end of constructor

 

 // رشته‌ای به انتهای کنترل افزوده می‌شود.

public void Add(string theString)

{

if (ctr >= strings.Length)

{

 // در این قسمت می‌توان کدی جهت کنترل پر شدن فضای تخصیص داده شده قرار داد.

}

else

strings[ctr++] = theString;

}//end of Add()

 

 // اعلان اندیکسر

public string this[int index]

{

get

{

if (index < 0 || index >= strings.Length)

{

// در این قسمت می‌توان کدی جهت کنترل پر شدن فضای تخصیص داده شده قرار داد.

}

return strings[index];

}//end of get

set

{

if (index >= ctr )

{

 // فراخوانی متدی جهت کنترل خطا

}

else

strings[index] = value;

}//end of set

}//end of indexer

 

// تعداد رشته‌های موجود را نشان می‌دهد

public int GetNumEntries( )

{

return ctr;

}

 

private string[] strings;

private int ctr = 0;

}//end of ListBoxTest class

 

public class Tester

{

static void Main( )

{

 //جدید و تخصیص دهی آن ListBox ساخت یک

ListBoxTest lbt = new ListBoxTest("Hello", "World");

 // رشته‌های مورد نظر به کنترل افزوده می‌شوند.

lbt.Add("Who");

lbt.Add("Is");

lbt.Add("John");

lbt.Add("Galt");

 // رشتة جدیدی در خانه شمارة یک فرار داده می‌شود.

string subst = "Universe";

lbt[1] = subst;

 // کلیه آیتمهای موجود نمایش داده می‌شوند.

for (int i = 0;i

{

Console.WriteLine("lbt[{0}]: {1}",i,lbt[i]);

}

}//end of Main()

}//end of Tester class

خروجی نیز بشکل زیر می‌باشد :

 

Output:

 

lbt[0]: Hello

lbt[1]: Universe

lbt[2]: Who

lbt[3]: Is

lbt[4]: John

lbt[5]: Galt

 

توجه :

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