در قسمت قبل اندکی با virtual method آشنا شدید. همانطور که ذکر شد، پروسهی تعریف مجدد virtual method در derived class را method overriding مینامند.
همانطور که گفته شد، virtual method در base class با کلمهیکلیدی virtual تعریف میشود. هنگامیکه یک virtual method در derived class مجدداً تعریف میشود، باید از override modifier استفاده کنید و هنگام override کردن یک متد، باید اسم متد، return type و پارامترهای آن را مطابق با virtual method بنویسید.
به مثال زیر توجه کنید:
using System; class Human { public virtual void SayHello(string name) { Console.WriteLine("SayHello in base class"); } } class Man : Human { public override void SayHello(string name) { Console.WriteLine("Hello " + name); } } class OverrideDemo { static void Main() { Man ob = new Man(); ob.SayHello("Stefan"); } }
متد ()SayHello درکلاس Human بهصورت virtual تعریف شده است و یک پارامتر دارد. در کلاس Man که از Human ارثبری کرده، متد مربوطه override شده است. همانطور که میبینید این متد در کلاس Man از override modifier استفاده کرده است. دقت کنید که override کردن یک متد ضروری نیست و در صورتیکه متدی را override نکنید، آن نسخه از متد که در base class وجود دارد اجرا خواهد شد.
به مثال زیر توجه کنید:
using System; class A { public virtual void SayHello() { Console.WriteLine("SayHello in base class"); } } class B : A { public override void SayHello() { Console.WriteLine("SayHello in B"); } } class C : A { // this class doesn't override SayHello() } class OverrideDemo { static void Main() { A a = new A(); B b = new B(); C c = new C(); a.SayHello(); b.SayHello(); c.SayHello(); } }
خروجی:
در اینجا، کلاس C متد ()SayHello را override نمیکند بنابراین زمانیکه متد ()SayHello از طریق شیء c فراخوانی میشود، متد ()SayHello در کلاس A اجرا خواهد شد.
هنگامیکه از سلسله مراتب ارثبری استفاده میکنید، اگر یک derived class، یک virtual method را override نکند، به طرف ابتدای زنجیرهی ارثبری حرکت کنید، اولین override آن متد که دیده شود اجرا خواهد شد.
به مثال زیر توجه کنید:
using System; class A { public virtual void SayHello() { Console.WriteLine("SayHello in base class"); } } class B : A { public override void SayHello() { Console.WriteLine("SayHello in B"); } } class C : B { // this class doesn't override SayHello() } class D : C { // this class doesn't override SayHello() } class OverrideDemo { static void Main() { D d = new D(); d.SayHello(); } }
خروجی:
همانطور که در مثال بالا میبینید، کلاس D از C و کلاس C از B و کلاس B از A ارثبری کرده است. کلاس D و C متد ()SayHello را override نکردهاند اما کلاس B این متد را override کرده است. بنابراین هنگامی که از طریق شیء کلاس D این متد را صدا میزنید، در زنجیرهی ارثبری اولین کلاسی که متد ()SayHello را فراخوانی کرده است کلاس B است. بنابراین همانطور که در خروجی میبینید، نسخهی override شدهی این متد، موجود در کلاس B، اجرا خواهد شد. قابل ذکر است که properties و indexers نیز میتوانند با استفاده از virtual و override به همین شکل مورد استفاده قرار گیرند.
علت استفاده از متدهای override شده چیست؟
متدهای override شده به سیشارپ اجازه میدهند تا از ویژگی runtime polymorphism بهره ببرد. Polymorphism توانایی ساخت متدهایی است که با توجه به موقعیت، میتوانند اجرای متفاوتی داشته باشند. برای مثال شما میتواند هم به ماشین و هم به سگ غذا بدهید اما خوب میدانید که معنای غذا دادن به ایندو کاملاً متفاوت است. Polymorphism به این دلیل برای برنامهنویسی شیگرا اهمیت دارد که به یک کلاس کلی، اجازه میدهد متدهایی داشته باشد که در همهی کلاسهای مشتق شده از آن کلاس، مشترک هستند. این درحالی است که به derived class ها این اجازه را میدهد تا هرطور که میخواهند آن متدها را اجرا کنند و درصورت نیاز، نحوهی اجرای آن متدها را تغییر دهند. متدهای override شده، روش دیگری برای اجرای این جنبه از polymorphism که میگوید “one interface, multiple methods” هستند.
به مثال زیر توجه کنید:
// Use virtual methods and polymorphism. using System; class TwoDShape { double pri_width; double pri_height; // A default constructor. public TwoDShape() { Width = Height = 0.0; name = "null"; } // Parameterized constructor. public TwoDShape(double w, double h, string n) { Width = w; Height = h; name = n; } // Construct object with equal width and height. public TwoDShape(double x, string n) { Width = Height = x; name = n; } // Construct a copy of a TwoDShape object. public TwoDShape(TwoDShape ob) { Width = ob.Width; Height = ob.Height; name = ob.name; } // Properties for Width and Height. public double Width { get { return pri_width; } set { pri_width = value < 0 ? -value : value; } } public double Height { get { return pri_height; } set { pri_height = value < 0 ? -value : value; } } public string name { get; set; } public void ShowDim() { Console.WriteLine("Width and height are " + Width + " and " + Height); } public virtual double Area() { Console.WriteLine("Area() must be overridden"); return 0.0; } } // A derived class of TwoDShape for triangles. class Triangle : TwoDShape { string Style; // A default constructor. public Triangle() { Style = "null"; } // Constructor for Triangle. public Triangle(string s, double w, double h) : base(w, h, "triangle") { Style = s; } // Construct an isosceles triangle. public Triangle(double x) : base(x, "triangle") { Style = "isosceles"; } // Construct a copy of a Triangle object. public Triangle(Triangle ob) : base(ob) { Style = ob.Style; } // Override Area() for Triangle. public override double Area() { return Width * Height / 2; } // Display a triangle's style. public void ShowStyle() { Console.WriteLine("Triangle is " + Style); } } // A derived class of TwoDShape for rectangles. class Rectangle : TwoDShape { // Constructor for Rectangle. public Rectangle(double w, double h) : base(w, h, "rectangle") { } // Construct a square. public Rectangle(double x) : base(x, "rectangle") { } // Construct a copy of a Rectangle object. public Rectangle(Rectangle ob) : base(ob) { } // Return true if the rectangle is square. public bool IsSquare() { if (Width == Height) return true; return false; } // Override Area() for Rectangle. public override double Area() { return Width * Height; } } class DynShapes { static void Main() { TwoDShape[] shapes = new TwoDShape[5]; shapes[0] = new Triangle("right", 8.0, 12.0); shapes[1] = new Rectangle(10); shapes[2] = new Rectangle(10, 4); shapes[3] = new Triangle(7.0); shapes[4] = new TwoDShape(10, 20, "generic"); for (int i = 0; i < shapes.Length; i++) { Console.WriteLine("object is " + shapes[i].name); Console.WriteLine("Area is " + shapes[i].Area()); Console.WriteLine(); } } }
خروجی:
در برنامهی بالا، ابتدا ()Area بهصورت virtual در کلاس TwoDShape تعریف شده و سپس توسط کلاسهای Triangle و Rectangle نیز override شده است. در TwoDShape میبینید که ()Area فقط بهصورت virtual تعریف شده است و تنها کاری که انجام میدهد این است که اطلاع میدهد این متد باید override شود. هر override از متد ()Area باید بستگی به شکل شیءای داشته باشد که derived class نشان دهندهی آن است. بهعنوان مثال اگر شکل مورد نظر مستطیل است نحوهی محاسبهی مساحت آن متناسب با مستطیل خواهد بود و اگر شکل مورد نظر مثلث باشد، نحوهی محاسبهی مساحت آن نیز متناسب با مثلث است. نکتهی مهم دیگر برنامهی بالا درون متد ()Main است. همانطور که میبینید shapes آرایهای از اشیای TwoDShape است اما عناصری که در این آرایه قرار دادیم reference های Triangle و Rectangle و TwoDShape هستند. همانطور که قبلاً ذکر شد، این مورد به این دلیل صحیح است که base class reference میتواند به derived class object رجوع کند. این برنامه سپس توسط یک حلقه، اطلاعات عناصر موجود در آرایه را نمایش میدهد.
استفاده از کلاسهای Abstract
گاهی قصد دارید یک base class بسازید که تنها یک فرم کلی را مشخص میکند و آن را با تمام کلاسهای مشتق شده، به اشتراک میگذارد و اجازه میدهد که خود derived class ها بدنه و جزئیات این فرم کلی را تکمیل کنند. بهعنوان مثال، این چنین کلاسی ماهیت یک متد را مشخص میکند و derived class ها باید این متد را override کنند اما خود base class دیگر نیازی ندارد که برای این متد یک اجرای پیشفرض داشته باشد. این حالت ممکن است زمانی رخ دهد که base class نتواند یک اجرای بامعنی برای متد مورد نظر داشته باشد، از اینرو اجرا را بر عهدهی derived class ها میگذارد. مانند مثال قبل که متد ()Area در کلاس TwoDShape، هیچگونه محاسباتی را انجام نمیداد. در چنین مواقعی، میتوانید مانند مثال قبل بهسادگی یک پیغام هشدار درون متد قرار دهید اما این روش چندان مناسب نیست و ممکن است در شرایط خاصی مثل debug کردن، مناسب باشد. گاهی ممکن است متدهایی در base class داشته باشید که derived class ها حتماً باید آنها را اجرا کنند، در چنین شرایطی باید از abstract method استفاده کنید.
یک متد abstract با abstract modifier ساخته میشود. abstract method بدنه ندارد و از اینرو درون base class اجرا نخواهد شد. derived class ها حتماً باید این abstract method را override کنند. یک abstract method بهصورت اتوماتیک virtual نیز است و در واقع نمیتوانید از virtual و abstract باهم در یک تعریف استفاده کنید.
فرم کلی abstract method بهشکل زیر است:
abstract type name(parameter-list);
همانطور که میبینید، در abstract method به بدنه نیاز ندارید. دقت کنید که abstract modifier را نمیتوانید برای متدهای static استفاده کنید. properties و indexers نیز میتوانند abstract باشند.
کلاسی که شامل یک یا بیشتر از یک متد abstract باشد باید بهصورت abstract تعریف شود. برای تعریف یک کلاس بهصورت abstract کافی است که قبل از کلمهی کلیدی class از abstract modifier استفاده کنید. از آنجا که abstract class نمیتواند بهطور کامل اجرا شود (بهدلیل وجود متدهای abstract که بدنه ندارند)، بههمین دلیل نمیتوانید از abstract class شیء بسازید.
هنگامیکه یک derived class از یک abstract class ارثبری میکند باید تمام متدهای abstract در base class را override کند در غیر اینصورت derived class نیز باید بهصورت abstract تعریف شود.
به مثال زیر توجه کنید:
using System; abstract class TwoDShape { double pri_width; double pri_height; // Parameterized constructor. public TwoDShape(double w, double h, string n) { Width = w; Height = h; name = n; } // Properties for Width and Height. public double Width { get { return pri_width; } set { pri_width = value < 0 ? -value : value; } } public double Height { get { return pri_height; } set { pri_height = value < 0 ? -value : value; } } public string name { get; set; } // Now, Area() is abstract. public abstract double Area(); } // A derived class of TwoDShape for triangles. class Triangle : TwoDShape { string Style; // Constructor for Triangle. public Triangle(string s, double w, double h) : base(w, h, "triangle") { Style = s; } // Override Area() for Triangle. public override double Area() { return Width * Height / 2; } } // A derived class of TwoDShape for rectangles. class Rectangle : TwoDShape { // Constructor for Rectangle. public Rectangle(double w, double h) : base(w, h, "rectangle") { } // Override Area() for Rectangle. public override double Area() { return Width * Height; } } class AbsShape { static void Main() { Triangle triangle = new Triangle("right", 8.0, 12.0); Rectangle rectangle = new Rectangle(10, 4); Console.WriteLine("Triangle | Area: " + triangle.Area()); Console.WriteLine("Rectangle | Area: " + rectangle.Area()); } }
همانطور که برنامه نشان میدهد، همهی derived class ها بایستی ()Area را override کنند (یا اینکه خودشان باید abstract باشند). نکتهی دیگر این است که یک abstract class میتواند متدهایی داشته باشد که abstract نیستند و derived class ها میتوانند در صورت نیاز آنها را override کنند درحالی که هیچ اجباری در کار نیست.
رها
19 September 2013
سلام و عرض ادب حضور اقای درویشیان
و تشکر بخاطر این موارد فوق العاده
اقای درویشیان من از اموزش #C هدفم بیشتر طراحی وب بود وقتی این سری اموزشی رو روع کردم اما مثل اینکه این مبحث اموزشی اصلا به وب و طراحی سایت مربوط نمیشه
من برا ساس انچه که در چند اله گذشته اموخته بودم (html , css ) میخواستم ادامه اون رو با asp.net پیش ببرم
حالا خواهشم اینه اگر برای شما مقدوره و در این زمینه اموزش خوبی رو یا وب سایت اموزشی (البته بجز اموزشها استاد کیانیان و اموزشهای انیاک ) اموزش کامل و جامعی برای طراحی وب سایت و قالب وب سایتها با استفاده ترکیبی از کدهای html ,css با ASP.NET در زمینه کاری میشناسید لطفا معرفی کنید واقعا ممنون میشم
من هرچقدر سرچ زدم اکثر اموزشها برای سطوح بالا بوده و برای مبتدیان در بین اونها چیزی پیدا نکردم
واقعا ممنون میشم اگر با مهارتی که دراین زمینه دارید منو برای استارت شروع در این زمینه راهنمایی کنید
با تشکر وارزوی توفیق ورزافزون
شاد باشید
مسعود درویشیان
19 September 2013
اگر قصد دارید asp.net کار کنید که حتماً باید #C یا VB.Net رو مسلط باشید و به همین دلیل خوندن مقالات زنگ سیشارپ کمک زیادی بهتون میکنه
مقاله فارسی خاصی تو این زمینه سراغ ندارم و پیشنهادم اینه که کتابهای انگلیسی رو مطالعه بفرمایید. توی سایت http://www.asp.net هم مطالب خیلی مقیدی میتونید پیدا کنید. کتابهای پیشنهادی توی سایت asp.net عالی هستن
رها
23 September 2013
سلام
خیلی ممنون و تشکر بخاطر معرفی مقاله و وب سایت ASP.NET
موفق باشید
تکران سرویس
23 September 2013
سلام..ممنون به خاطر این سری از اموزش ها.. لطفا ادامه بدیدو
سروش
24 September 2013
سلام
خسته نباشید
به سفارش دوستان به این صفحه اومدم و با اینکه رشته ام نیست ولی از اموزشها واقعا لذت بردم خیلی کامل توضیح داده شده
در صورتی که مقدوره و در توان شما هست خواهش میکنم در کنار این سری اموزشی آموزش ساخت وب اپلیکیشن (MVC) رو هم در کنارش شروع کنید و حداقل یه پروژه خیلی مختصر در مورد طراحی وب با این زبان رو بصورتی که الان دارید #C رو اموزش میدین بذارید
اموزشهای شما خیلی روان توضیح داده شده تا بحال اموزشی رو در این مقوله تا این حد کامل ندیده بودم که تمام نکات و خط به خط کدها رو کامل نوشته باشه
واقعا ممنون میشم اگر برای کسانی که بدنبال طراحی وب در قالب پروژه mvc هستند هم یه بخش اموزشی رو راه اندازی کنید
با ارزوی توفیق و راه اندازی این بخش
طراحی وب سایت
23 October 2013
بسیار عالی خسته نباشید
رها
23 October 2013
سلام و عرض ادب
امیدوارم با سوالات و درخواستهام شما رو خسته نکرده باشم
اما من بعد از توصیه شما به فراگیری #C برای اینکه بتونم در نهایت با ASP.NET کار کنم شروع بکار کردم
اموزشها شما رو وقتی تمرین میکنم بیشتر در صفحه داس اجرا میشه اما ASP.NET در مرورگر اجرا میشه
امیدکه بتونم منظورم رو بیان کنم . من در ASP همونطور که گفتم تازه کارم و وفقط با html , css اشنایی دارم
حالا نمیدونم یادگیری #C میتونه در نهایت برای طراحی سایت با ASP کمکی بکنه
اصلا کار سی شارپ و نقش اون در ASP چی هست
در صورت امکان و اگر فرصتی بود لطفا راهنمایی کنید که ایا با دنبال کردن این اموزش میشه بعدا از این کدها در ASP هم استفاده کرد ؟
با تقدیر وتشکر و ارزوی توفیق
مسعود درویشیان
23 October 2013
سلام. شما برای استفاده از asp.net نیاز دارید که سیشارپ (یا VB.net) رو یاد بگیرید چون زبان برنامهنویسیای که توی asp.net استفاده میشه، #C هست. برای این مجموعه آموزشی تصمیم بر این بود که سیشارپ رو روی کنسول آموزش بدیم تا یادگیری خیلی سادهتر بشه. شما از طریق این مجموعه زبان سیشارپ رو یاد میگیرید و میتونید در برنامهنویسی وب، موبایل، دسکتاپ، ساخت بازی و حتی سختافزار از سیشارپ استفاده کنید. در نهایت زبان سیشارپ (چیزی که توی این مجموعه آموزشی یاد میگیرید) توی محیطهای مختلف عوض نمیشه فقط باید با نحوهی برنامهنویسی روی هر محیط آشنا بشید. مثلاً وب و موبایل و دسکتاپ هرکدوم تکنیک برنامهنویسی خاص خودشون رو دارند اما میتونید برای همشون از سیشارپ استفاده کنید.
در نهایت اگر میخواهید روی وب کار کنید با خوندن کتابها و مقالات مربوط به asp.net یاد میگیرید که چطوری از سیشارپ توی وب استفاده میشه.
حمیدرضا
31 December 2013
آقا مسعود سلام
یه نظر من override کردن و نکردن یک متد هیچ فرقی نمیکند مثالی که شما بالا زدید با این نمونه چه فرقی می کند؟
شما تو توضیحات گفتید که اگر از override و virtual استفاده نکنیم و در صورتیکه متدی را override نکنید، آن نسخه از متد که در base class وجود دارد اجرا خواهد شد.ولی همانطور که بالا میبینید من نه از virtual و نه از override استفاده کردم و با اجرای این برنامه هیچ تفاوتی ندارد.
این طور که من متوجه شدم این مورد تنها در کلاسهایی کاربرد دارد که هیچ متدی در آن تعریف نشده است.
اگر امکان دارد بیشتر توضیح دهید.
مسعود درویشیان
31 December 2013
سلام. دوست عزیز اگه توجه کنی وقتی این برنامه رو میخوای compile کنی یه warning بهت میده که این هست:
Warning 1 ‘Man.SayHello(string)’ hides inherited member ‘Human.SayHello(string)’. Use the new keyword if hiding was intended
این یعنی اینکه متد SayHello شما توی کلاس Man دیگه اجازه نمیده شما مستقیماً از طریق اشیای این کلاس به متد SayHello کلاس Human دسترسی داشته باشی. چرا؟ چون دقیقاً همنام هستند و پارامترهاشون هم یکیه! پس وقتی شما از طریق شیء Man متد SayHello رو صدا میزنی، چون اسم متد با متد کلاس پدرش یکی هست باعث میشه کامپایلر یه هشدار بده که دیگه شما مستقیماً به متد کلاس پدر دسترسی نداری! به همین خاطر هم موقع اجرا، متد موجود توی کلاس Man اجرا میشه. کلاً فلسفهی virtual و override این هست که شما بدونی یه متد ممکنه کامل نباشه و فقط بخشی از کارهارو انجام بده و شما این اجازه رو داری که واسه کلاس خودت کاملش کنی یا اینکه کاملاً تغییرش بدی
یه بار دیگه مطالب عنوان “علت استفاده از متدهای override شده چیست” رو مرور کنی بهتر متوجه میشی
اگه بازم سوالی بود حتما بپرسید
حمیدرضا
31 December 2013
ممنون از پاسختون ولی من این جا رو متوجه نشدم که شما مستقیماً از طریق اشیای این کلاس به متد SayHello کلاس Human دسترسی داشته باشی. من مثال دوم شما رو که در مورد مستطیل و … هست رو overrid و virtual هایش رو برداشتم و تفاوت رو حس کردم ولی چون برنامه تعداد خطوط زیاد داشت متوجه نشدم که چه اتفاقی افتاد. در مثالی که براتون نوشتم برداشتم ولی هیچ اتفاقی نیفتاد. الان مشکل من تو همینه که چرا تو یکیش فرق داشت تو اون یکی نداشت.
بازم ممنون از جوابتون
elina
23 January 2015
سلام جناب آقای درویشیان
من قبلا شنیدم بودم برای یادگیری asp.net نیاز به یادگیری c# نیست.
بخاطر همین پروژه رو asp.net انتخاب کردم.
ساخت فروشگاه انلاین هست.
خواستم بپرسم، الان بدبخت شدم؟
مسعود درویشیان
23 January 2015
البته که نیازه، هرچند با ویژوال بیسیک هم میشه برنامهنویسیاش رو انجام داد. با این حال، باید سیشارپ رو یاد بگیرید. یا اینکه کلاً پروژهتون رو با php انجام بدید.
فرزانه
6 November 2015
وافعا عااالی بود
ابراهیم
19 November 2015
ش