اندازهی قلم متن
تخمین مدت زمان مطالعهی مطلب:
سه دقیقه
این الگو اجازهی تعریف کردن عملیاتی جدید را برای مجموعهای از شیءها، بدون تغیر دادن ساختار خود شیءها، میدهد. همچنین اجازهی جدا کردن کلاس را از منطقی که کلاس پیاده سازی میکند، به ما میدهد.
عملیات بیشتری میتوانند در شیء Visitor کپسوله سازی شوند. شیءها میتوانند یک متد visit داشته باشند که یک شیء Visitor را دریافت میکند. Visitor میتواند تغییرات مورد نیاز را ایجاد کند و عملیاتی را بر روی شیءهایی که دریافت کردهاست، انجام دهد.
این الگو به توسعه دهندگان این اجازه را میدهد که کتابخانهها (libraries)، فریم ورکها (frameworks) و ... را گسترش دهند.
مثال:
class Visitor { visit(item){} } class BookVisitor extends Visitor { visit(book) { var cost=0; if(book.getPrice() > 50) { cost = book.getPrice()*0.50 } else{ cost = book.getPrice() } console.log("Book name: "+ book.getName() + "\n" + "ID: " + book.getID() + "\n" + "cost: "+ cost); return cost; } } class Book{ constructor(id,name,price){ this.id = id this.name = name this.price = price } getPrice(){ return this.price } getName(){ return this.name } getID(){ return this.id } accept(visitor){ return visitor.visit(this) } } var visitor = new BookVisitor() var book1 = new Book("#1234","lordOftheRings",80) book1.accept(visitor)
در مثال بالا ما یک کتابفروشی داریم. کلاس Book برای نشان دادن یک کتاب در فروشگاه استفاده شدهاست. این کلاس همانند زیر تعریف شدهاست:
class Book{ constructor(id,name,price){ this.id = id this.name = name this.price = price } //code... }
یک کتاب خصوصیات زیر را دارد:
- id
- name
- price
هم چنین شامل توابع زیر میباشد:
getPrice(){ return this.price } getName(){ return this.name } getID(){ return this.id }
متد getPrice ، قیمت را برگشت میدهد، getName ، نام را برگشت میدهد و getID، شناسهی کتاب را برگشت میدهد.
اکنون کتابفروشی یک تخفیف را برای کتابهایی که هزینهی آنها بیشتر از 50 دلار است، معرفی میکند. در ادامه، میخواهیم یک عملیات دیگر را انجام دهیم و تخفیف را بر روی آنها پیاده سازی کنیم. در اینجا از الگوی visitor استفاده خواهیم کرد. ما یک Visitor را معرفی میکنیم که کتابها را بازدید خواهد کرد و قیمت آنها را بهروزرسانی میکند. بنابراین شیءهای کتاب باید تابعی داشته باشند که اجازه دهد visitor، آنها را بازدید (visit) کند و عملیات مد نظر را بر روی آنها انجام دهد. برای این منظور، یک متد به نام accept در کلاس Book تعریف کردهایم:
accept(visitor){ return visitor.visit(this) }
متد accept یک شیء visitor را به عنوان یک آرگومان دریافت میکند و به آن اجازه میدهد که با فراخوانی کردن تابع visit خودش، کتاب جاری را بازدید (visit) کند (this اشاره به کتاب جاری دارد) .
اکنون اجازه دهید نگاهی به کلاس Visitor داشته باشیم:
class Visitor { visit(item){} }
این کلاس، یک تابع به نام visit دارد و itemی را که میخواهد بازدید (visit ) کند، به عنوان پارامتر دریافت میکند. در این سناریو، میخواهیم که کتابها را بازدید (visit ) کنیم. از این رو، در ابتدا یک کلاس را به نام BookVisitor تعریف میکنیم که کلاس Visitor را extend میکند:
class BookVisitor extends Visitor { visit(book) { var cost=0; if(book.getPrice() > 50) { cost = book.getPrice()*0.50 } else{ cost = book.getPrice() } console.log("Book name: "+ book.getName() + "\n" + "ID: " + book.getID() + "\n" + "cost: "+ cost); return cost; } }
تابع visit، قیمت کتابی را که دارد بازدید میکند، بررسی میکند. اگر بزرگتر از 50 باشد، 50 درصد تخفیف را بر روی آن اعمال میکند؛ در غیر این صورت، قیمت به حالت قبلی خودش باقی میماند.
چه زمانی از این الگو استفاده کنیم:
- زمانیکه نیاز است عملیاتی مشابه، بر روی شیءهای متفاوتی از یک data structure انجام شود.
- زمانیکه نیاز است عملیاتی خاص، بر روی شیءهای متفاوتی از data structure انجام شود.
- زمانیکه میخواهید توسعه پذیری را برای کتابخانهها (libraries) یا فریم ورکها (frameworks) اضافه کنید.