پرش به محتوا

چندریختی (برنامه‌نویسی)

از ویکی‌پدیا، دانشنامهٔ آزاد
چندریختی (برنامه‌نویسی)

یکی از ویژگی‌های کلیدی در کلاس‌های مشتق‌شده، این است که یک اشاره‌گر به یک کلاس مشتق‌شده، از نظر نوع داده‌ای با اشاره‌گر، به کلاس اصلی آن سازگار است. چندریختی یا پلی‌مورفیسم (به انگلیسی: Polymorphism)، هنر بهره‌برداری از این ویژگی ساده و در عین حال قدرتمند و کاربردی است که متدلوژی شی‌گرایی را به حد کامل توانایی‌اش می‌رساند.[۱][۲]

مفهوم چندریختی ویژگی است که به رابط‌ها امکان می‌دهد تا برای گروهی از عملیات‌ها مورد استفاده قرار گیرند. در این مقاله مفهوم چندریختی از طریق دو مفهوم بازنویسی متدها (به انگلیسی: override) و سربارشده (به انگلیسی: Overload) پشتیبانی می‌شود. ما با بررسی این دو مفهوم به دنبال بررسی تأثیر مفهوم چندریختی روی برنامه‌نویسی شیء گرا هستیم. روش به کار برده شده در این تحقیق، مطالعه منابعی از جمله کتب مرتبط با برنامه‌نویسی، راهنمای برنامه‌های شیء گرا و اجرای برنامه‌هایی با ماهیت چندریختی است. با توجه به مطالب ارائه شده در این مقاله می‌توان گفت که مفهوم چندریختی یکی از سه اصول مهم در برنامه‌نویسی شیء گرا است. در نتیجه مفاهیم نهان‌سازی و وراثت به همراه مفهوم چندریختی سه رکن اصلی برنامه‌نویسی شئ‌گرا را تشکیل می‌دهند. کلمات کلیدی: چندریختی، سربار کردن، لغو کردن و متد.

چندریختی به معنای چندین شکل از یک متد است. مفهوم چندریختی اغلب به وسیلهٔ عبارت «یک رابط، چند متد» بیان می‌شود. این بدین معناست که این امکان وجود دارد که یک رابط عمومی برای گروهی از فعالیت‌های مرتبط به هم طراحی شود.[۳]

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

چنانچه نام و امضای متدی از یک زیر کلاس (به انگلیسی: Sub Class) 《در ساختار سلسله مراتبی کلاس‌ها 》 با نام و نوع امضای متدی از ابر کلاس آن یکسان باشد، در آن صورت متد موجود در زیر کلاس، متد موجود در ابر کلاسش را لغو می‌کند.[۴]

مفاهیم سربار کردن و لغو کردن متدها از جمله مفاهیمی هستند که چندریختی بر پایهٔ آن‌ها بنا می‌شود. در این مقاله پس از بررسی اجمالی مفهوم چندریختی در بخش اول، در بخش دوم، مفهوم سربار کردن متدها را با ذکر چند مثال خواهیم دید. در بخش سوم نیز مفهوم متدهای لغو شده یا همان توابع مجازی را خواهیم آورد و در بخش پایانی نیز به بررسی تأثیر وجود مفهوم چندریختی در برنامه‌نویسی شیءگرا 《مانند جاوا》 و مقایسهٔ آن با برنامه‌هایی که چندریختی را پشتیبانی نمی‌کنند 《مانند زبان برنامه‌نویسی C》 خواهیم پرداخت.

مفهوم چندریختی

[ویرایش]
چندریختی (یا polymorphism که از زبان یونانی گرفته می‌شود)، ویژگی است که رابط‌ها از طریق آن به گروهی از عملیات‌ها دست می‌یابند. عمل خاص مورد نظر توسط ماهیت دقیق شرایط جاری تعیین می‌شود.

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

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

به‌طور کلی، چندریختی به وسیلهٔ عبارت «یک رابط، چند متد» بیان می‌شود. این امر با فراهم ساختن امکان استفاده از یک رابط برای مشخص کردن یک کلاس عمومی از عملیات، به کاهش پیچیدگی کمک می‌کند؛ و این کامپایلر است که عمل مورد نظر را (یعنی متد مربوطه) متناسب با هر یک از شرایط انتخاب می‌کند. به عنوان مثال حس بویایی سگ‌ها نمونه‌ای از چندریختی می‌باشد. چنانچه بوی گربه‌ای به مشام سگی برسد، در آن صورت واق واق می‌کند و به دنبال گربه خواهد رفت. اما اگر بوی غذا به مشامش برسد، بزاق آن ترشح خواهد کرد و به طرف ظرف غذا خواهد رفت. در هر دو شرایط مزبور، یک نوع حس بویایی کار می‌کند. تفاوت این دو حالت، بویی است که به مشام سگ می‌رسد؛ یعنی، نوع داده‌ای که بینی سگ بر روی آن کار انجام می‌دهد!

چندریختی، نهان‌سازی و وراثت با هم کار می‌کنند

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

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

سربار کردن متدها

[ویرایش]

تعریف و چگونگی انجام آن

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

سربار کردن سازنده‌ها

[ویرایش]
همان‌طور که می‌دانیم، هر کلاسی که در جاوا ایجاد می‌کنیم، می‌تواند دارای سازنده باشد. اگر سازنده‌ای برای کلاسی ننویسیم، جاوا یک سازنده برای آن کلاس می‌نویسد که از طریق آن می‌توان اشیای آن کلاس را ایجاد کرد. این کلاس به فیلدهای اشیاء نیز مقادیر اولیه مناسبی می‌دهد؛ یعنی، به عنوان مثال فیلدهای عددی را برابر صفر و فیلدهای رشته‌ای را برابر تهی قرار می‌دهد.
متدهای سازنده را نیز می‌توان همچون متدهای معمولی سربار کرد. درحقیقت، در بیشتر کلاس‌های مربوط به کارهای واقعی، سازنده‌های سربار شده، نه تنها استثنا به‌شمار نمی‌آیند، بلکه کاملاً معمول خواهند بود.

مثالی از سازنده‌های سربار شده

[ویرایش]

در ادامه مثالی از سازنده‌های سربار شده نمایش داده شده‌است. در این مثال کلاس Book طراحی شده‌است که دارای سه سازنده با امضاهای متفاوت می‌باشد.

public class Book{

    private String name;
    private String author;
    //overloaded constructors
    public Book(){

    }

    public Book(String name){
        this.name = name;
    }

    public Book(String name, String author){
        this.Book(name);
        this.author = author;
    }

}

نحوه ساخت ارجاعی از نوع این کلاس به یکی از روش‌های زیر قابل انجام است.

Book newBook = new Book();
Book newBook = new Book("Shahnameh");
Book newBook = new Book("Shahnameh", "Ferdosi");

تعریف و چگونگی انجام آن

[ویرایش]
چنانچه نام و نوع امضای (هدر متد که از نام و پارامترها تشکیل می‌شود) متدی از یک زیر کلاس با نام و امضای متدی از ابر کلاسش یکسان باشد، در آن صورت اصطلاحاً می‌گوییم که متد موجود در زیرکلاس، متد موجود در ابر کلاس را بازنویسی می‌کند.
وقتی متد بازنویسی شده‌ای از یک زیر کلاس فراخوانده می‌شود، همیشه از نگارش تعریف شده در زیر کلاس استفاده خواهد شد. نگارش تعریف شده در ابر کلاس پنهان خواهد شد.

تعیین پویای متدها

[ویرایش]
مکانیزم لغو کردن متدها، پایه و اساس یکی از قدرتمندترین مفاهیم برنامه‌نویسی شیء گرا را تشکیل می‌دهد، تعیین پویای متدها. تعیین پویای متدها مکانیزمی است که جاوا با استفاده از آن، حاصل عبارت فراخوانی متدهای لغو شده را به جای زمان کامپایل، در زمان اجرا تعیین می‌کند. دلیل اهمیت این مکانیزم آن است که جاوا، چندریختی زمان اجرا را با آن پیاده‌سازی می‌کند.
اینک کار خود را با بیان مجدد یک اصل مهم آغاز می‌کنیم: متغیرهای ارجاع ابر کلاس‌ها می‌توانند به شیءهای زیرکلاس‌ها ارجاع داشته باشند. جاوا با استفاده از این امر، مسئلهٔ تعیین متدهای لغو شده را در زمان اجرا حل می‌کند. چگونگی انجام این کار به این شرح است. وقتی متد لغو شده‌ای از طریق ارجاع یک ابر کلاس فراخوانده می‌شود، جاوا بر اساس نوع شیئی که در زمان فراخوانی به آن ارجاع می‌شود، تعیین می‌کند که کدام نگارش از متد اجرا شود. از این رو این کار در زمان اجرا انجام می‌شود.[۵]
وقتی ارجاع به انواع مختلف شیءها صورت می‌گیرد، نگارش‌های مختلفی از یک متد لغو شده فراخوانده خواهند شد. به عبارت دیگر نوع شیء مورد ارجاع (ونه متغیر ارجاع) است که تعیین می‌کند که کدام نگارش ازیک متد لغو شده، اجرا خواهدشد.

منابع

[ویرایش]
  1. دکتر بابک بشری راد (۱۳۹۴). برنامه‌سازی پیشرفته با ++C. تهران: ناقوس. صص. ۱۴۳. شابک ۹۷۸-۹۶۴-۳۷۷-۷۴۶-۳.
  2. Cardelli, Luca; Wegner, Peter (December 1985). "On understanding types, data abstraction, and polymorphism" (PDF). ACM Computing Surveys. 17 (4): 471–523. CiteSeerX 10.1.1.117.695. doi:10.1145/6041.6042. S2CID 2921816.: "Polymorphic types are types whose operations are applicable to values of more than one type."
  3. Stroustrup, Bjarne (February 19, 2007). "Bjarne Stroustrup's C++ Glossary". polymorphism – providing a single interface to entities of different types.
  4. Conallen, J.; Engle, M.; Houston, K.; Maksimchuk, R.; Young, B.; Booch, G. (2007). Object-Oriented Analysis and Design with Applications (3rd ed.). Pearson Education. ISBN 9780132797443.
  5. Strachey, Christopher (2000). "Fundamental Concepts in Programming Languages". Higher-Order and Symbolic Computation. 13 (1/2): 11–49. CiteSeerX 10.1.1.332.3161. doi:10.1023/A:1010000313106. ISSN 1573-0557. S2CID 14124601.