Obyektga yo'naltirilgan dasturlashdagi vorislik - bu tizim bo'ylab ota-sinfdagi(base class) xususiyatlarni bola sinflari(child class) ichida qayta ishlatishga imkon beruvchi xususiyat bo'lib, bu vorislikning asosiy afzalliklaridan biridir. Ammo, biz hal qilmoqchi bo'lgan yoki mavhumlashtirmoqchi bo'lgan ma'lum bir domen(model) uchun sinflarni loyihalashda, ba'zi yaxshi amaliyotlar (yoki yomonlari) uzoq muddatda dasturiy ta'minotning umumiy barqarorligiga ta'sir qilishi mumkin.
Asosan, meros ushbu manbadan foydalanishni oqlash uchun yetarlicha o'xshashliklarga ega bo'lgan sinflar o'rtasida foydalanish uchun mo'ljallangan. Agar bola sinflari(child class) ular uchun mantiqiy bo'lmagan xususiyat va metodlarga ega bo'lishni boshlasa, hatto ular ota-ona sinfidan(base class) bo'lsa ham, meros haqida yana bir bor o'ylab ko'rish vaqti keladi.
Liskovning almashtirish printsipi (Liskov Substitution Principle) - yuqori sinf obyektlari dasturni buzmasdan uning kichik sinflari obyektlari bilan almashtirilishi kerak. Bunda sizning pastki sinflaringiz obyektlari sizning yuqori sinfingiz obyektlari xususiyatlarini o'z ichiga olishi talab qiladi.
Yuqoridagi ta'riflar tushunish uchun ozgina qiyin bo'lishi mumkin. Shuning uchun keling bularni amalda qo'llab tushinishga harakat qilamiz.
Tasavvur qiling biz telegram guruhidagi a'zolar va adminlar uchun berilgan ruxsatlarni muhokama qilaylik. Bizda telegram foydalanuvchisi - TelegramUser, guruh a'zosi - GroupSubscriber va guruh admini - GroupAdmin sinflari bor.
TelegramUser sinfi:
publicclassTelegramUser {publicstring FullName { get; set; } =string.Empty;publicstring Email { get; set; } =string.Empty;publicstring Password { get; set; } =string.Empty;publicvirtualvoidAccessToAddAdmin() {Console.WriteLine("Guruhga admin qo'shish uchun ruxsat berilgan"); }publicvirtualvoidAccessToChangeGroupInfo() {Console.WriteLine("Guruh ma'lumotlarini o'zgartirish uchun ruxsat berilgan"); }publicvirtualvoidAccessToReadMessages() {Console.WriteLine("Xabarlarni o'qish uchun ruxsat berilgan"); }publicvirtualvoidAccessToRemoveMessages() {Console.WriteLine("Xabarlarni o'chirish uchun ruxsat berilgan"); } }
Yuqorida ko'rib turganingizdek, ikkala bola sinf(child class) ham TelegramUser sinfidan meros qilib olinyapti. GroupSubscriber(guruh a'zosi)ga guruh ma'lumotlarini o'zgartirish va admin qo'shish kabi xususiyatlar xos emas. Demak, TelegramUser bu ikki sinf uchun umumiy bo'la olmadi.
Liskov printsipini qo'llash uchun ikkita yondashuv mavjud:
Ota-sinfda faqat umumiy xususiyatlar va metodlarni belgilang, har qanday o'ziga xoslikni bolalar sinflariga qoldiring.
Ota-sinfni bola sinflari o'ziga xosliklarini to'g'ri taqsimlaydigan bir nechta interfeyslarda ajrating.
1-usul - o'ziga xos bo'lgan xususiyatlar va metodlar o'sha sinfning o'zida e'lon qilinadi:
publicclassTelegramUser {publicstring FullName { get; set; } =string.Empty;publicstring Email { get; set; } =string.Empty;publicstring Password { get; set; } =string.Empty; }publicclassGroupSubscriber:TelegramUser {publicvirtualvoidAccessToReadMessages() {Console.WriteLine("Xabarlarni o'qish uchun ruxsat berilgan"); }publicvirtualvoidAccessToRemoveMessages() {Console.WriteLine("Xabarlarni o'chirish uchun ruxsat berilgan"); } }publicclassGroupAdmin:TelegramUser { publicvirtualvoidAccessToAddAdmin() {Console.WriteLine("Guruhga admin qo'shish uchun ruxsat berilgan"); }publicvirtualvoidAccessToChangeGroupInfo() {Console.WriteLine("Guruh ma'lumotlarini o'zgartirish uchun ruxsat berilgan"); }publicvirtualvoidAccessToReadMessages() {Console.WriteLine("Xabarlarni o'qish uchun ruxsat berilgan"); }publicvirtualvoidAccessToRemoveMessages() {Console.WriteLine("Xabarlarni o'chirish uchun ruxsat berilgan"); } }
2-usul - quyida ko'rsatilganidek, har bir holat uchun maxsus interfeyslarni yaratishdir:
publicinterfaceIGroupUser {voidAccessToReadMessages();voidAccessToRemoveMessages(); }publicinterfaceIGroupAdmin {voidAccessToAddAdmin();voidAccessToChangeGroupInfo(); }publicclassTelegramUser {publicstring FullName { get; set; } =string.Empty;publicstring Email { get; set; } =string.Empty;publicstring Password { get; set; } =string.Empty; }publicclassGroupSubscriber:TelegramUser,IGroupUser {publicvirtualvoidAccessToReadMessages() {Console.WriteLine("Xabarlarni o'qish uchun ruxsat berilgan"); }publicvirtualvoidAccessToRemoveMessages() {Console.WriteLine("Xabarlarni o'chirish uchun ruxsat berilgan"); } }publicclassGroupAdmin:TelegramUser,IGroupUser,IGroupAdmin {publicvoidAccessToAddAdmin() {Console.WriteLine("Admin qo'shish uchun ruxsat berilgan"); }publicvoidAccessToChangeGroupInfo() {Console.WriteLine("Guruh ma'lumotlarini o'zgartirish uchun ruxsat berilgan"); }publicvirtualvoidAccessToReadMessages() {Console.WriteLine("Xabarlarni o'qish uchun ruxsat berilgan"); }publicvirtualvoidAccessToRemoveMessages() {Console.WriteLine("Xabarlarni o'chirish uchun ruxsat berilgan"); } }