Miyrasxorlıq boyınsha kompoziciya

Obyektke baǵdarlanǵan programmalastırıwda (OBP) miyrasxorlıq boyınsha kompoziciya (yamasa kompoziciyalıq qayta paydalanıw principi) ‒ klasslardıń tiykarǵı yamasa ata-analıq klasstan miyras alıwdıń ornına, polimorflıq minez-qulıq hám kodtı qayta paydalanıwdı olardıń kompoziciyası arqalı (qálegen funkcionallıqtı ámelge asıratuǵın basqa klasslardıń mısalların óz ishine alıw arqalı) jaqlawı kerek degen princip[2]. Ideal jaǵdayda barlıq qayta paydalanıw bar bolǵan komponentlerdi jıynaw arqalı ámelge asırılıwı múmkin, biraq ámelde jańaların jaratıw ushın kóbinese miyras alıw kerek boladı. Sonlıqtan, miyras alıw hám obekt kompoziciyası ádette birge isleydi, bul "Dizayn úlgileri" (1994) kitabında talqılanǵan.[3]
Tiykarlar
Miyrasxorlıq boyınsha kompoziciyanıń ámelge asırılıwı ádette sistemada bolıwı kerek bolǵan minez-qulıqlardı kórsetetuǵın hár qıylı interfeyslerdi jaratıwdan baslanadı. Interfeysler polimorflıq minez-qulıqtı jeńillestire aladı. Anıqlanǵan interfeyslerdi ámelge asıratuǵın klasslar dúziledi hám kerek bolǵanda biznes taraw klasslarına qosıladı. Solay etip, sistema minez-qulıqları miyras alıwsız ámelge asırıladı.
Haqıyqatında, biznes taraw klassları hesh qanday miyras alıwsız tiykarǵı klasslar bolıwı múmkin. Sistema minez-qulıqlarınıń alternativ ámelge asırılıwı qálegen minez-qulıq interfeysin ámelge asıratuǵın basqa klasstı beriw arqalı orınlanadı. Interfeyske silteme saqlaytuǵın klass interfeystiń ámelge asırılıwların qollay aladı - bul tańlaw orınlanıw waqtına shekem keyinge qaldırılıwı múmkin.
Mısal
Miyras alıw
C++ tilindegi mısal tómendegishe:
class Obekt
{
public:
virtual void jańala() {
// hesh nárse islemeydi
}
virtual void sızıw() {
// hesh nárse islemeydi
}
virtual void soqlıǵısıw(Obekt obektler[]) {
// hesh nárse islemeydi
}
};
class Kórinetuǵın : public Obekt
{
Model* model;
public:
virtual void sızıw() override {
// usı obekttiń ornında modeldi sızıw ushın kod
}
};
class Qattı : public Obekt
{
public:
virtual void soqlıǵısıw(Obekt obektler[]) override {
// basqa obektler menen soqlıǵısıwdı tekseriw hám oǵan juwap beriw ushın kod
}
};
class Jıljıwshı : public Obekt
{
public:
virtual void jańala() override {
// usı obekttiń ornın jańalaw ushın kod
}
};
Keyin, mınaday konkret klasslardıń bar ekenin kóz aldıńızǵa keltiriń:
- class Oyınshı - ol Qattı, Jıljıwshı hám Kórinetuǵın
- class Bult - ol Jıljıwshı hám Kórinetuǵın, biraq Qattı emes
- class Imarat - ol Qattı hám Kórinetuǵın, biraq Jıljıwshı emes
- class Duzaq - ol Qattı, biraq Kórinetuǵın da, Jıljıwshı da emes
Esińizde bolsın, kóp miyrasxorlıq dıqqat penen ámelge asırılmasa qáwipli bolıwı múmkin, sebebi ol romb máselesine alıp keliwi múmkin. Bunıń bir sheshimi - hárbir kerekli kombinaciya ushın KórinetuǵınHámQattı, KórinetuǵınHámJıljıwshı, KórinetuǵınHámQattıHámJıljıwshı hám t.b. sıyaqlı klasslar jaratıw bolıp tabıladı; biraq, bul kóp muǵdarda qaytalanatuǵın kodqa alıp keledi. C++ kóp miyrasxorlıqtıń romb máselesin sheshiw ushın virtual miyrasxorlıqtı qollanadı.
Kompoziciya hám interfeysler
Bul bólimdegi C++ mısalları kod qayta paydalanıw hám polimorfizmge erisiw ushın kompoziciya hám interfeyslerdi qollanıw principin kórsetedi. C++ tilinde interfeyslerdi járiyalaw ushın arnawlı gilt sóz bolmaǵanlıqtan, tómendegi C++ mısalı taza abstrakt tiykarǵı klasstan miyras alıwdı qollanadı. Kópshilik maqsetler ushın, bul funkcional jaqtan Java[4]: 87 hám C#[5].: 144 sıyaqlı basqa tillerde berilgen interfeyslerge uqsas.
VisibilityDelegate dep atalatuǵın abstrakt klass kirgiziń, onıń NotVisible hám Visible degen subklassları bar, ol obektti sızıw usılın támiyinleydi:
class KórinisligiWákil
{
public:
virtual void sızıw() = 0;
};
class Kórinbeytuǵın : public KórinisligiWákil
{
public:
virtual void sızıw() override {
// hesh nárse islemeydi
}
};
class Kórinetuǵın : public KórinisligiWákil
{
public:
virtual void sızıw() override {
// bul obekttiń poziciyasında modeldi sızıw ushın kod
}
};
JańalawWákil dep atalatuǵın abstrakt klasstı kirgiziń, onıń Qozǵalmaytuǵın hám Qozǵalatuǵın degen subklassları bar, bul obektti jıljıtıw usılın támiyinleydi:
class JańalawWákil
{
public:
virtual void jańalaw() = 0;
};
class Qozǵalmaytuǵın : public JańalawWákil
{
public:
virtual void jańalaw() override {
// hesh nárse islemeydi
}
};
class Qozǵalatuǵın : public JańalawWákil
{
public:
virtual void jańalaw() override {
// bul obekttiń poziciyasın jańalaw ushın kod
}
};
SoqlıǵısıwWákil dep atalatuǵın abstrakt klasstı kirgiziń, onıń Qattı emes hám Qattı degen subklassları bar, bul obekt penen soqlıǵısıw usılın támiyinleydi:
class SoqlıǵısıwWákil
{
public:
virtual void soqlıǵısıw(Obekt obektler[]) = 0;
};
class QattıEmes : public SoqlıǵısıwWákil
{
public:
virtual void soqlıǵısıw(Obekt obektler[]) override {
// hesh nárse islemeydi
}
};
class Qattı : public SoqlıǵısıwWákil
{
public:
virtual void soqlıǵısıw(Obekt obektler[]) override {
// basqa obektler menen soqlıǵısıwdı tekseriw hám oǵan juwap beriw ushın kod
}
};
Aqırında, Obekt dep atalatuǵın klasstı kirgiziń, onıń kóriniwin basqarıw ushın (KóriniwWákil qollanıp), qozǵalıwshılıǵın basqarıw ushın (JańalawWákil qollanıp) hám qattılıǵın basqarıw ushın (SoqlıǵısıwWákil qollanıp) aǵzaları bar. Bul klassta óz aǵzalarına tapsırma beretuǵın metodlar bar, mısalı, jańalaw() ápiwayı túrde JańalawWákil ústindegi metodtı shaqıradı:
class Obekt
{
KóriniwWákil* _k;
JańalawWákil* _j;
SoqlıǵısıwWákil* _s;
public:
Obekt(KóriniwWákil* k, JańalawWákil* j, SoqlıǵısıwWákil* s)
: _k(k)
, _j(j)
, _s(s)
{}
void jańalaw() {
_j->jańalaw();
}
void sızıw() {
_k->sızıw();
}
void soqlıǵısıw(Obekt obektler[]) {
_s->soqlıǵısıw(obektler);
}
};
Sonnan keyin anıq klaslar tómendegishe boladı:
class Oyınshı : public Obekt
{
public:
Oyınshı()
: Obekt(new Kórinetuǵın(), new Jıljıtılatuǵın(), new Qattı())
{}
// ...
};
class Tútin : public Obekt
{
public:
Tútin()
: Obekt(new Kórinetuǵın(), new Jıljıtılatuǵın(), new QattıEmes())
{}
// ...
};
Artıqmashılıqları
Miyrasxorlıq boyınsha kompoziciyaǵa artıqmashılıq beriw - bul dizaynǵa joqarı qolaylılıq beretuǵın dizayn principi. Biznes-domen klasların túrli komponentlerden dúziw, olardıń arasındaǵı ulıwmalıqtı tabıwǵa hám shańaraq teregin jaratıwǵa urınıwǵa qaraǵanda tábiyiyirek. Mısalı, akselerator pedalı menen rul dóńgeleginiń ulıwma qásiyetleri júdá az, biraq ekewi de avtomobildiń áhmiyetli komponentleri bolıp tabıladı. Olardıń ne isley alatuǵını hám avtomobilge qanday payda keltire alatuǵını ańsat anıqlanadı. Sonday-aq, kompoziciya shańaraq aǵzalarınıń ózgesheliklerine beyim bolmaǵanlıǵı sebepli, uzaq múddetli perspektivada turaqlıraq biznes-domendi támiyinleydi. Basqasha aytqanda, obekttiń ne ekenligin (is-a) keńeytiwden góre, onıń ne isley alatuǵının (has-a) qurastırıw jaqsıraq.
Dáslepki dizayn, sistema obektleriniń háreketlerin bólek interfeyslerde anıqlaw arqalı ápiwayılastırıladı, biznes-domen klasları arasında háreketlerdi miyrasxorlıq arqalı bólistiriw ushın ierarxiyalıq qatnas jaratıwdıń ornına. Bul kózqaras keleshektegi talaplardıń ózgerislerine ańsat beyimlesedi, al miyrasxorlıq modelinde bunday ózgerisler biznes-domen klaslarınıń tolıq qayta dúziliwin talap eter edi. Bunnan tısqarı, bul kózqaras bir neshe áwlad klasların óz ishine alǵan miyrasxorlıqqa tiykarlanǵan modeldegi salıstırmalı túrde kishi ózgerisler menen baylanıslı máselelerden qashadı. Kompoziciya qatnası kóbirek iykemli, sebebi ol orınlanıw waqtında ózgertiliwi múmkin, al subtiplew qatnasları statikalıq bolıp, kóplegen tillerde qayta kompilyaciyanı talap etedi.
Ayırım tiller, ásirese Go[6] hám Rust[7], tek tip kompoziciyasın qollanadı.
Kemshilikleri
Miyrasxorlıqtıń ornına kompoziciyanı qollanıwdıń bir ulıwma kemshiligi - ayırım komponentler tárepinen beriletuǵın metodlar tuwındı tipte ámelge asırılıwı kerek bolıwı múmkin, hátte olar tek jiberiletuǵın metodlar bolsa da (bul kópshilik programmalastırıw tillerinde durıs, biraq hámmesinde emes; § Kemshiliklerden qashıw bólimin qarań). Kerisinshe, miyrasxorlıq tiykarǵı klasstıń barlıq metodların tuwındı klassta qayta ámelge asırıwdı talap etpeydi. Kerisinshe, tuwındı klass tek tiykarǵı klass metodlarınan ózgeshe háreketke iye metodlardı ámelge asırıwı (biykarlawı) kerek. Eger tiykarǵı klass kóp sanlı ádettegi háreketti támiyinleytuǵın metodlardı óz ishine alsa hám olardıń tek bir neshewi tuwındı klassta biykarlanıwı kerek bolsa, bul júdá az programmalastırıw miynetin talap etiwi múmkin.
Mısalı, tómendegi C# kodında, Employee tiykarǵı klassınıń ózgeriwshileri hám metodları HourlyEmployee hám SalariedEmployee tuwındı subklassları tárepinen miyras etip alınadı. Tek Pay() metodı hárbir tuwındı subklass tárepinen ámelge asırılıwı (arnawlı etiliwi) kerek. Basqa metodlar tiykarǵı klasstıń ózi tárepinen ámelge asırıladı hám onıń barlıq tuwındı subklassları tárepinen bólisiledi; olardı subklass anıqlamalarında qayta ámelge asırıw (biykarlaw) yamasa hátte esletiw kerek emes.
// Tiykarǵı klass
public abstract class Xızmetker
{
// Qásiyetler
protected string Atı { get; set; }
protected int ID { get; set; }
protected decimal TólemStavkası { get; set; }
protected int IslegenSaatlar { get; }
// Házirgi tólem dáwiri ushın tólemdi alıw
public abstract decimal Tólem();
}
// Tuwındı subklass
public class SaatlıqXızmetker : Xızmetker
{
// Házirgi tólem dáwiri ushın tólemdi alıw
public override decimal Tólem()
{
// Islengen waqıt saatlarda
return IslegenSaatlar * TólemStavkası;
}
}
// Tuwındı subklass
public class AylıǵıXızmetker : Xızmetker
{
// Házirgi tólem dáwiri ushın tólemdi alıw
public override decimal Tólem()
{
// Tólem stavkası saatlıq emes, al jıllıq maosh
return IslegenSaatlar * TólemStavkası / 2087;
}
}
Kemshiliklerdi boldırmaw
Bul kemshilikten treytler, miksinler, (tip) jaylastırıw yamasa protokol keńeytiwlerin qollanıw arqalı boldırmawǵa boladı.
Ayırım tiller onı jeńilletiw ushın arnawlı qurallar usınadı:
- C# 8.0 versiyasınan baslap interfeys aǵzasına deneni anıqlawǵa múmkinshilik beretuǵın standart interfeys metodların usınadı[8].: 28–29 [9]: 38 [10]: 466–468
- D tiptiń ishinde anıq "alias this" deklaraciyasın usınadı, ol basqa jaylastırılǵan tiptiń hárbir metodı hám aǵzasın alǵa jibere aladı[11].
- Dart bólisiwge bolatuǵın standart ámelge asırıwları bar miksinlerdi usınadı.
- Go tipin jaylastırıw metodlardı alǵa jiberiw zárúrligin saplastıradı[12].
- Java 8-versiyasınan baslap standart interfeys metodların usınadı. Project Lombok wákillik beriw maydannıń barlıq metodlarınıń atları hám tiplerin kóshirip alıw hám saqlaw ornına, maydanda @Delegate annotaciyasın qollanıw arqalı wákillikti qollaydı.
- Julia makrosları metodlardı alǵa jiberiw ushın qollanılıwı múmkin. Lazy.jl[13] hám TypedDelegation.jl sıyaqlı bir neshe ámelge asırıwlar bar[14][15].
- Kotlin til sintaksisine wákillik úlgisin kirgizgen[16].
- PHP 5.4 versiyasınan baslap treytlerdi qollaydı[17].
- Raku metodlardı alǵa jiberiwdi ańsatlastırıw ushın handles treytin usınadı[18].
- Rust standart ámelge asırıwları bar treytlerdi usınadı.
- Scala (3-versiyadan baslap) obekttiń tańlanǵan aǵzaları ushın laqaplardı anıqlaw ushın "export" bólimin usınadı[19].
- Swift keńeytiwleri ayırım tip ámelge asırıwınıń ishinde emes, al protokoldıń ózinde protokoldıń standart ámelge asırılıwın anıqlaw ushın qollanılıwı múmkin[20].
Empirikalıq izertlewler
2013-jılı 93 ashıq kodlı Java baǵdarlamasına (hár qıylı kólemdegi) júrgizilgen izertlew mınalardı anıqladı:
Miyrasxorlıqtı kompoziciya menen almastırıw ushın úlken múmkinshilik joq bolsa da (...), bul múmkinshilik áhmiyetli (miyrasxorlıqtı qollanıwdıń ortasha 2% tek ishki qayta qollanıw ushın, al qosımsha 22% tek sırtqı yamasa ishki qayta qollanıw ushın).
Biziń nátiyjelerimiz miyrasxorlıqtıń qáte qollanılıwına baylanıslı táshiwish etiwge sebep joq ekenin kórsetedi (hesh bolmaǵanda ashıq kodlı Java baǵdarlamalarında), biraq olar kompoziciyaǵa qarsı miyrasxorlıqtı qollanıw máselesin alǵa shıǵaradı. Eger kompoziciyanı qollanıwǵa bolatuǵın jaǵdayda miyrasxorlıqtı qollanıw menen baylanıslı áhmiyetli shıǵınlar bar bolsa, onda biziń nátiyjelerimiz bazı bir táshiwishleniwge sebep bar ekenin kórsetedi.
Derekler
- ↑ Freeman, Eric; Robson, Elisabeth; Sierra, Kathy; Bates, Bert. Head First Design Patterns. O'Reilly, 2004 — 23 bet. ISBN 978-0-596-00712-6.
- ↑ Knoernschild, Kirk. Java Design - Objects, UML, and Process: 1.1.5 Composite Reuse Principle (CRP). Addison-Wesley Inc., 2002. ISBN 9780201750447. 29-may 2012-jıl sánesinde qaraldı.
- ↑ Gamma, Erich. Design Patterns: Elements of Reusable Object-Oriented Software. Addison-Wesley, 1994. ISBN 0-201-63361-2.
- ↑ Bloch, Joshua. "Effective Java: Programming Language Guide", third, Addison-Wesley, 2018. ISBN 978-0134685991.
- ↑ Price, Mark J.. C# 8.0 and .NET Core 3.0 – Modern Cross-Platform Development: Build Applications with C#, .NET Core, Entity Framework Core, ASP.NET Core, and ML.NET Using Visual Studio Code. Packt. ISBN 978-1-098-12195-2.
- ↑ Pike. «Less is exponentially more» (25-iyun 2012-jıl). Qaraldı: 1-oktyabr 2016-jıl.
- ↑ «Characteristics of Object-Oriented Languages - The Rust Programming Language». doc.rust-lang.org. Qaraldı: 10-oktyabr 2022-jıl.
- ↑ «What's new in C# 8.0». Microsoft Docs. Microsoft. Qaraldı: 20-fevral 2019-jıl.
- ↑ Skeet, Jon. C# in Depth, 23 March 2019.
- ↑ Albahari, Joseph. C# 10 in a Nutshell, 2022.
- ↑ «Alias This». D Language Reference. Qaraldı: 15-iyun 2019-jıl.
- ↑ «(Type) Embedding». The Go Programming Language Documentation. Qaraldı: 10-may 2019-jıl.
- ↑ «MikeInnes/Lazy.jl». GitHub.
- ↑ «JeffreySarnoff/TypedDelegation.jl». GitHub.
- ↑ «Method forwarding macro» (en). JuliaLang (20-aprel 2019-jıl). Qaraldı: 18-avgust 2022-jıl.
- ↑ «Delegated Properties». Kotlin Reference. JetBrains. Qaraldı: 11-iyul 2018-jıl.
- ↑ «PHP: Traits». www.php.net. Qaraldı: 23-fevral 2023-jıl.
- ↑ «Type system». docs.raku.org. Qaraldı: 18-avgust 2022-jıl.
- ↑ «Export Clauses». Scala Documentation. Qaraldı: 6-oktyabr 2021-jıl.
- ↑ «Protocols». The Swift Programming Language. Apple Inc. Qaraldı: 11-iyul 2018-jıl.