توضیحات
سه رکن شی گرایی عبارتند از کپسوله کردن (encapsulation)، وراثت (inheritance) و چند ریختی (polymorphism).
بعضی از برنامه نویس ها abstraction هم به عنوان رکن چهارم شی گرایی در نظر میگیرن که اینجا موضوع بحث ما نیست.
در دو بخش قبل با کپسوله کردن (Encapsulation) و وراثت (inheritance) آشنا شدیم در این بخش میخوایم با رکن سوم شی گرایی یعنی چند ریختی (Polymorphism) آشنا بشیم. و سپس به پیوند پویا (Dynamic Binding) و کست کردن (Type Casting) که با چند ریختی مرتبط هستند می پردازیم.
برای پرداختن به پیوند پویا (dynamic binding) و کست کردن (type casting) ابتدا لازمه با چند ریختی (polymorphism) در کاتلین آشنا بشیم.
چند ریختی (Polymorphism)
چند ریختی یعنی متغیر از جنس یکی از superclass ها اشاره کنه به آبجکت ایجاد شده از subclass
همینطور که در قسمت قبل (وراثت) گفته شد subclass ویژگی های superclass رو به ارث میبره به علاوه ویژگی های جدید خودشو میتونه داشته باشه پس میشه گفت subclass یک کلاس اختصاصی از superclass است.
هر دایره شکل هندسیه اما هر شکل هندسی دایره نیست (میتونه مستطیل باشه).
بنابراین هر دایره علاوه بر ویژگی های یک شکل هندسی ویژگی های اختصاصی خودشم داره که دایره رو از بقیه شکل های هندسی متمایز میکنه.
با توجه به این استدلال میشه گفت هر متغیر از GeometricShape میتونه به هر آبجکت از Circle یا Rectangle اشاره کنه اما برعکسش نمیشه.
در بخش قبل کلاس های Circle، Rectangle و GeometricShape تعریف شدن. همینطور کلاس GeometricShape از Shape نیز ارث بری میکنه.
در مثال زیر برای خلاصه شدن از دوباره نویسیشون خودداری کردیم میتونید به بخش قبل مراجعه و در مورد مطالعه کد هاشونو کپی کنید.
در مثال زیر کلاس Person سوپر کلاس Student و Citizen است.
پیوند پویا (Dynamic Binding)
یک تابع در طی ارث بری میتونه به چند روش پیادهسازی بشه زمان اجرای برنامه jvm تصمیم میگیره کدوم پیادهسازی به اجرا در بیاد.
یک تابع میتونه در superclass تعریف بشه و در subclass باز نویسی بشه به عنوان مثال تابع toString در کلاس Any تعریف شده و چون کلاس Any سوپر کلاس تمام کلاس هاست تابع toString میتونه در هر کلاسی بازنویسی و به طور اختصاصی پیادهسازی بشه.
به قطعه کد زیر توجه کنید:
سؤال: هنگام اجرای کد کدوم toString اجرا میشه؟ اونی که در Shape بازنویسی شده یا اونی که در Any تعریف شده؟
برای پاسخ به این سؤال ابتدا با دو مفهوم نوع تعریف شده (declared type) و نوع واقعی (actual type) باید اشنا بشیم.
نوع تعریف شده (declared type)
هنگام تعریف متغیر اگه نوع متغیر رو مشخص کنیم بهش declared type میگیم در بالا declared type کلاس Any است.
نوع واقعی (actual type)
به آبجکتی که ایجاد میکنیم نوع واقعی (actual type) میگیم. actual type میتونه نوعش با declared type یکی باشه یا یکی از subclass های declared type باشه. در بالا actual type کلاس Shape است.
بهتره برگردیم به سوالمون اینکه کدوم toString اجرا میشه؟
جواب: بستگی به actual type داره.
زمان اجرا JVM از actual type شروع میکنه به بررسی تا به declared type برسه و به اولین پیادهسازی تابع که رسید بررسی رو متوقف و تابع رو اجرا میکنه. در بالا چون اولین پیادهسازی toString داخل Shape اتفاق افتاده همونو اجرا میکنه.
مثال
کست کردن (Type Casting)
تبدیل آبجکت به یک آبجکت از نوع دیگه رو کست کردن (type casting) میگیم.
فرض کنید declared type سوپر کلاس actual type است.
به عبارتی یک متغیر تعریف میکنیم که نوع متغیر superclass آبجکتی است که بهش داره اشاره میکنه.
در این حالت هنگام صدا زدن پراپرتی و توابع با استفاده از متغیر فقط توابع و پراپرتی هایی که در superclass تعریف شدن در دسترس هستند.
گاهی در کد نیاز داریم از توابع و پراپرتی های actual type در کد استفاده کنیم برای همین رو میاریم به کست کردن declared type به actual type.
برای کست کردن (type casting) از کلیدواژه ی as در کاتلین استفاده میکنیم.
مثال
در این حالت کامپایلر اخطار میده که ممکنه o به Shape در حافظه اشاره نکنه و این کست کردن safe نیست. به هر حال کست کردن انجام میشه و برنامه بدون مشکل اجرا میشه.
میتونیم با استفاده از is بررسی کنیم declared type سوپر کلاس actual type هست یا خیر.
کاتلین از ویژگی کست هوشمند (smart casting) بهره می بره یعنی اگه با is معلوم بشه declared type به کدوم actual type داره اشاره میکنه دیگه نیاز به کست کردن آشکار نیست.
خلاصه
- سه رکن شی گرایی عبارتند از encapsulation، inheritance، polymorphism
- چند ریختی یعنی یک متغیر از جنس superclass اشاره کنه به آبجکتی از جنس subclass
- هنگامی که یک تابع صدا زده میشه JVM از actual type شروع میکنه به بررسی و اولین پیادهسازی تابع رو اجرا میکنه.
- کست کردن یعنی تبدیل آبجکت از یک نوع به آبجکتی از نوع دیگه.
- کاتلین از ویژگی کست هوشمند یا (smart casting) پشتیبانی میکنه.