narjes Mansoori
narjes Mansoori
خواندن ۴ دقیقه·۱ سال پیش

درک مفهوم انتزاع (Abstraction) در شی‌گرایی به زبان ساده

توی شی‌گرایی دونستن انتزاع یا Abstraction کافی نیست. بلکه باید اون رو درک کنیم و با درک صحیح اون، دیدمون به دنیای شی‌گرایی وسیع‌تر و باز تر میشه. اینکه ویژگی کلاس‌های Abstract و اینترفیس‌ها چیه به تنهایی مهم نیست. مهم اینه که درک کنیم اینها کجا به کار ما میان و چه مشکلاتی رو حل میکن.

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

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

توی این مثال، خودرو و حیوان مفاهیم انتزاعی (Abstract) هستن که قابل پیاده‌سازی نیستن. بلکه فقط الگو و طرحی هستن برای چیزهای عینی و قابل پیاده‌سازی. به گربه و هیوندار سانتافه قرمز و هر چیزی توی دنیای واقعی وجود دارن و ما اونها رو می‌بینیم، میگن Concrete یعنی واقعی.

مفاهیم انتزاعی، یک طرح کلی و "الگو" هستن برای چیزهای عینی. چند تا مثال از الگو، و چیزهای عینی از اون الگو:

  • حیوان برای گربه
  • خودرو برای دوچرخه
  • نوشیدنی برای چایی
  • غذا برای آبگوشت
  • ورزش برای پیاده‌روی
  • میوه برای موز
  • آجیل برای پسته

اگه بریم مغازه خشکبار و بگیم آجیل می‌خوایم، مطمئناً فروشنده میگه چه نوع آجیلی! (اگه خواستید دفعه بعد رفتید خشکبار فروشی امتحان کنین :)) ). چون آجیل یک مفهموم کلی و انتزاعی هست.

توی برنامه‌نویسی شی‌گرا مفاهیم Abstract و Concrete هم وجود دارن.

به کلاس‌هایی که قابل نمونه‌سازی باشن میگن Concrete Classes و کلاس‌هایی که قابل نمونه‌سازی نباشن میگن Abstract Classes

منظور از نمونه سازی همون new کردن یک کلاس هست.

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

abstract class Animal { abstract makeSound(); move() { // Moving } } new Animal; // Error Cannot create an instance of an abstract

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

class Penguin extends Animal { makeSound() { // Ghizhzhzh } }

چرا از Abstraction استفاده کنیم؟

مثال زیر رو در نظر بگیرید که توی اون از Abstraction استفاده نشده:

class Orange {} class Apple {} // ... Rest of fruits class FruitsBasket { private items; public add(items) { items.forEach (item => { if ( item instanceof Orange || item instanceof Apple || item instanceof Banana ) { this.items.push(item); } }); } } let fruits = new FruitsBasket; fruits.add([orange, apple, banana, umbrella, meat, wall, glass, book]);

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

public add(items) { items.forEach (item => { if ( item instanceof Orange || item instanceof Apple || item instanceof Banana || item instanceof Peach || item instanceof Khiar // Cucumbers :) || item instanceof Melon ) { this.items.push(item); } }); }

خب این اصلاً خوب نیست.خوانایی و زیبایی این کد کم هست و همیشه در معرض تغییر قرار داره. همچنین قانون دوم SOLID (Open/Closed Principle) هم داره نقض میشه.

با استفاده از یک اینترفیس یا کلاس Abstract خیال خودمون رو راحت می‌کنیم:

abstract class Fruit {} class Orange extends Fruit {} class Apple extends Fruit {} // ... Rest of fruits class FruitsBasket { private items; public add(items) { items.forEach (item => { if (item instanceof Fruit) { this.items.push(item); } }); } } let fruits = new FruitsBasket; fruits.add([orange, apple, banana, umbrella, meat, wall, glass, book]);

در واقع ما اینجا به قول معروف یک لایه انتزاعی (Abstraction Layer) اضافه کردیم. اینطوری متد add وابسته به کلاس‌های بی‌نهایت Concrete نیست. بلکه وابسته به انتزاع هست. پس می‌تونیم هر چقدر که دلمون میخواد کلاس میوه اضافه کنیم؛ بدون اینکه کلاس FruitsBasket و متد add رو دستکاری کنیم.

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

abstraction
Android Developer
شاید از این پست‌ها خوشتان بیاید