Çift gönderim - Double dispatch

İçinde yazılım Mühendisliği, çift ​​gönderim özel bir şeklidir çoklu gönderim ve çağrıda yer alan iki nesnenin çalışma zamanı türlerine bağlı olarak farklı somut işlevlere bir işlev çağrısı gönderen bir mekanizma. Çoğunlukla nesne odaklı sistemler, koddaki bir işlev çağrısından çağrılan somut işlev, tek bir nesnenin dinamik türüne bağlıdır ve bu nedenle bunlar olarak bilinir tek gönderim aramalar veya basitçe sanal işlev aramalar.

Dan Ingalls ilk olarak çift dağıtmanın nasıl kullanılacağını anlattı Smalltalk, onu çağırmak çoklu polimorfizm.[1]

Genel Bakış

Ele alınan genel sorun, yalnızca alıcıya değil aynı zamanda argümanlara da bağlı olarak farklı yöntemlere bir mesajın nasıl gönderileceğidir.

Bu amaçla, sistemler CLOS uygulamak çoklu gönderim. İkili gönderim, çoklu gönderimi desteklemeyen sistemlerde polimorfizmi aşamalı olarak azaltan başka bir çözümdür.

Kullanım durumları

Çift gönderim, hesaplama seçiminin bağımsız değişkenlerinin çalışma zamanı türlerine bağlı olduğu durumlarda yararlıdır. Örneğin, bir programcı aşağıdaki durumlarda çift dağıtımı kullanabilir:

  • Karışık bir nesne kümesini sıralama: algoritmalar, bir nesne listesinin bazı kanonik sıraya göre sıralanmasını gerektirir. Bir elemanın başka bir elemandan önce gelip gelmediğine karar vermek, her iki tür ve muhtemelen alanların bazı alt kümeleri hakkında bilgi sahibi olmayı gerektirir.
  • Uyarlanabilir çarpışma algoritmaları genellikle farklı nesneler arasındaki çarpışmaların farklı şekillerde ele alınmasını gerektirir. Tipik bir örnek, bir uzay gemisi ile bir asteroid arasındaki çarpışmanın, bir uzay gemisi ile bir uzay istasyonu arasındaki çarpışmadan farklı bir şekilde hesaplandığı bir oyun ortamındadır.[2]
  • Boyama algoritmaları örtüşen kesişme noktalarını gerektiren Sprite farklı bir şekilde işlenecek.
  • Çalışan yönetimi sistemler olabilir sevk etmek farklı personel için farklı işler. Bir program Muhasebeci olarak yazılan bir kişi nesnesi ve mühendislik olarak yazılan bir iş nesnesi verilen algoritma, o kişinin o iş için planlamasını reddeder.
  • Olay işleme Doğru olay işleme rutinini çağırmak için alıcı nesnenin hem olay türünü hem de türünü kullanan sistemler.
  • Kilit ve anahtar birçok kilit türü ve birçok anahtar türü olan ve her anahtar türünün birden fazla kilit türü açtığı sistemler. Yalnızca ilgili nesnelerin türlerini bilmeniz gerekmez, aynı zamanda "belirli bir anahtarın belirli bir kilidi açıp açmadığını görmeyle ilgili olan belirli bir anahtar hakkında bilgi" alt kümesi de farklı kilit türleri arasında farklıdır.

Yaygın bir deyim

Yukarıda sunulan örneklerde olduğu gibi ortak deyim, uygun algoritmanın seçiminin çalışma zamanında çağrının argüman türlerine dayalı olmasıdır. Bu nedenle çağrı, çağrıların dinamik çözümlemesiyle ilişkili tüm olağan ek performans maliyetlerine tabidir, genellikle yalnızca tek yöntem gönderimini destekleyen bir dilden daha fazladır. İçinde C ++, örneğin, bir dinamik işlev çağrısı genellikle bir tek ofset hesaplama - bu mümkündür çünkü derleyici, işlevin nesnenin yöntem tablosu ve böylece ofseti statik olarak hesaplayabilir. Destekleyen bir dilde çift dağıtım, bu biraz daha maliyetlidir, çünkü derleyicinin yöntemin çalışma zamanında yöntem tablosundaki ofsetini hesaplamak için kod üretmesi gerekir, böylece genel komut yolu uzunluğu (işleve yapılan toplam çağrı sayısından fazla olmayacak bir miktara göre, bu çok önemli olmayabilir).

Ruby'de Örnek

Yaygın bir kullanım durumu, ekran veya yazıcı olabilen bir nesneyi ekran bağlantı noktasında veya henüz var olmayan başka bir şeyi görüntülemektir. Bu, bu farklı medyayla nasıl başa çıkılacağının saf bir uygulamasıdır.

sınıf Dikdörtgen  def display_on(Liman)    # nesne sınıfına göre doğru kodu seçer    durum Liman      ne zaman DisplayPort        DisplayPort'ta görüntülemek için # kod      ne zaman Yazıcı Bağlantı Noktası        # PrinterPort üzerinde görüntülemek için kod      ne zaman RemotePort        RemotePort'ta görüntülemek için # kod    son  sonson

Oval, Üçgen ve kendisini bir ortamda göstermek isteyen diğer nesneler için de aynısının yazılması gerekir. Sorun, birden fazla polimorfizm derecesinin mevcut olmasıdır: biri display_on yöntemini bir nesneye göndermek için, diğeri ise görüntüleme için doğru kodu (veya yöntemi) seçmek için.

Çok daha temiz ve bakımı daha kolay olan bir çözüm, ikinci bir gönderim yapmaktır, bu sefer nesneyi ortamda görüntülemek için doğru yöntemi seçmek için:

sınıf Dikdörtgen  def display_on(Liman)    # ikinci gönderim    Liman.display_rectangle(kendini)  sonsonsınıf Oval  def display_on(Liman)    # ikinci gönderim    Liman.display_oval(kendini)  sonsonsınıf DisplayPort  def display_rectangle(nesne)    # DisplayPort'ta bir dikdörtgeni görüntülemek için kod  son  def display_oval(nesne)    # DisplayPort'ta oval görüntülemek için kod  son  # ...sonsınıf Yazıcı Bağlantı Noktası  def display_rectangle(nesne)    # PrinterPort üzerinde bir dikdörtgeni görüntülemek için kod  son  def display_oval(nesne)    # PrinterPort üzerinde oval görüntülemek için kod  son  # ...son

C ++ ile çift gönderim

İlk bakışta, çift gönderim doğal bir sonucu gibi görünüyor fonksiyon aşırı yükleme. Fonksiyon aşırı yükleme, çağrılan fonksiyonun argümanın türüne bağlı olmasına izin verir. Bununla birlikte, işlev aşırı yüklemesi, "isim değiştirme "burada işlevin dahili adı bağımsız değişkenin türünü kodlar. Örneğin, bir işlev foo (int) dahili olarak çağrılabilir __foo_i ve işlev foo (çift) çağrılabilir __Gıda. Böylece, isim çakışması ve sanal tablo araması yoktur. Buna karşılık, dinamik gönderme, çağıran nesnenin türüne dayanır, yani sanal işlevler (geçersiz kılma) yerine fonksiyon aşırı yükleme ve bir vtable aramasına neden olur. Şu dilde yazılmış örneği düşünün C ++, bir oyundaki çarpışmaların:

sınıf Uzay Gemisi {};sınıf Apollo Uzay Aracı : halka açık Uzay Gemisi {};sınıf Asteroit {halka açık:  gerçek geçersiz Ters düşmek(Uzay Gemisi&) {    std::cout << "Asteroid bir SpaceShip'e çarptı n";  }  gerçek geçersiz Ters düşmek(Apollo Uzay Aracı&) {    std::cout << "Asteroid bir Apollo Uzay Gemisine çarptı n";  }};sınıf Patlayan Asteroid : halka açık Asteroit {halka açık:  geçersiz Ters düşmek(Uzay Gemisi&) geçersiz kılmak {    std::cout << "ExplodingAsteroid bir SpaceShip'e çarptı n";  }  geçersiz Ters düşmek(Apollo Uzay Aracı&) geçersiz kılmak {    std::cout << "PatlayanAsteroid bir Apollo Uzay gemisine çarptı n";  }};

Eğer varsa:

Asteroit asteroit;Uzay Gemisi uzay gemisi;Apollo Uzay Aracı theApolloSpacecraft;

daha sonra, işlevin aşırı yüklenmesi nedeniyle,

asteroit.Ters düşmek(uzay gemisi); asteroit.Ters düşmek(theApolloSpacecraft);

sırasıyla yazdıracak, Asteroid bir SpaceShip'e çarptı ve Asteroid bir Apollo'ya çarptı, herhangi bir dinamik gönderim kullanmadan. Ayrıca:

Patlayan Asteroid PatlayanAsteroid;PatlayanAsteroid.Ters düşmek(uzay gemisi); PatlayanAsteroid.Ters düşmek(theApolloSpacecraft);

yazdıracak Asteroid bir SpaceShip'e çarptı ve Asteroid bir Apollo uzay aracına çarptı sırasıyla dinamik gönderim olmadan.

Bir referansla Asteroitdinamik gönderim kullanılır ve bu kod:

Asteroit& theAsteroidReference = PatlayanAsteroid;theAsteroidReference.Ters düşmek(uzay gemisi); theAsteroidReference.Ters düşmek(theApolloSpacecraft);

baskılar Asteroid bir SpaceShip'e çarptı ve Asteroid bir Apollo uzay aracına çarptıyine beklendiği gibi. Ancak, aşağıdaki kod istendiği gibi çalışmıyor:

Uzay Gemisi& theSpaceShipReference = theApolloSpacecraft;asteroit.Ters düşmek(theSpaceShipReference);theAsteroidReference.Ters düşmek(theSpaceShipReference);

İstenen davranış, bu çağrıları alan işleve bağlamaktır. theApolloSpacecraft argümanı olarak, değişkenin somutlaştırılmış türü olduğu için, beklenen çıktı Asteroid bir Apollo'ya çarptı ve Asteroid bir Apollo uzay aracına çarptı. Ancak çıktı aslında Asteroid bir SpaceShip'e çarptı ve Asteroid bir SpaceShip'e çarptı. Sorun şu ki, sanal işlevler C ++ 'da dinamik olarak gönderilirken, işlev aşırı yüklemesi statik olarak yapılır.

Yukarıda açıklanan sorun şu şekilde çözülebilir: simülasyon çift ​​gönderim, örneğin bir ziyaretçi düzeni. Mevcut kodun, her ikisinin de Uzay Gemisi ve Apollo Uzay Aracı fonksiyon verilir

gerçek geçersiz Ters düşmek(Asteroit& inAsteroid) {  inAsteroid.Ters düşmek(*bu);}

Daha sonra, önceki örnek hala doğru çalışmıyorken, aramaları uzay gemisi ajan olacak şekilde yeniden çerçevelendirmek bize istenen davranışı verir:

Uzay Gemisi& theSpaceShipReference = theApolloSpacecraft;Asteroit& theAsteroidReference = PatlayanAsteroid;theSpaceShipReference.Ters düşmek(asteroit);theSpaceShipReference.Ters düşmek(theAsteroidReference);

Yazdırılıyor Asteroid bir Apollo'ya çarptı ve Asteroid bir Apollo uzay aracına çarptı, beklenildiği gibi. Anahtar şu ki theSpaceShipReference.CollideWith (theAsteroidReference); çalışma zamanında aşağıdakileri yapar:

  1. theSpaceShipReference bir referanstır, bu nedenle C ++ vtable'da doğru yöntemi arar. Bu durumda arayacak ApolloSpacecraft :: CollideWith (Asteroid ve).
  2. İçinde ApolloSpacecraft :: CollideWith (Asteroid ve), inAsteroid bir referanstır, bu yüzden inAsteroid.CollideWith (* this) sonuçlanacak başka bir vtable araması. Bu durumda, inAsteroid bir referanstır Patlayan Asteroid yani ExplodingAsteroid :: CollideWith (ApolloSpacecraft &) Aranacak.

C # ile çift gönderim

İçinde C #, bir bağımsız değişkeni kabul eden bir örnek yöntemi çağırırken, ziyaretçi modeli kullanılmadan çoklu gönderim gerçekleştirilebilir. Bu, geleneksel polimorfizm kullanılarak yapılırken aynı zamanda argümanı da oluşturarak yapılır. dinamik.[3] Çalışma zamanı bağlayıcı, çalışma zamanında uygun yöntem aşırı yüklemesini seçecektir. Bu karar, nesne örneğinin çalışma zamanı türünü (polimorfizm) ve bağımsız değişkenin çalışma zamanı türünü dikkate alacaktır.

Eyfel'de çift gönderim

Eyfel programlama dili, aracılar kavramını çift dağıtım sorununa taşıyabilir. Aşağıdaki örnek, aracı dili yapısını çift dağıtım sorununa uygular.

Çeşitli ŞEKİL biçimlerine sahip bir sorun alanını ve üzerine bir ŞEKİL çizmek için YÜZEY çizmeyi düşünün. Hem SHAPE hem de SURFACE, kendi içlerinde `` çiz '' adı verilen bir işlevi bilir, ancak birbirlerinde değil. İki türdeki nesnelerin, bir ziyaretçi kalıbı kullanarak ikili bir gönderimde birbirleriyle birlikte değişken olarak etkileşime girmesini istiyoruz.

Buradaki zorluk, kendi üzerine bir polimorfik ŞEKİL çizmek için bir polimorfik YÜZEY elde etmektir.

Çıktı

Aşağıdaki çıktı örneği, iki SURFACE ziyaretçi nesnesinin, polimorfik SHAPE nesnelerinin bir listesi üzerinden polimorfik olarak geçirilmesinin sonuçlarını göstermektedir. Ziyaretçi kodu kalıbı, yalnızca SHAPE ve SURFACE'in jenerik olarak farkındadır ve her ikisinin de belirli bir türünün farkındadır. Bunun yerine kod, bu iki ertelenmiş sınıf ve onların soyundan gelenler arasında oldukça esnek bir eş-değişken ilişki elde etmek için çalışma zamanı polimorfizmine ve aracıların mekaniğine dayanır.

çizmek kırmızı POLYGON ETCHASKET üzerinde kırmızı GRAFFITI_WALL üzerinde POLYGONdraw a gri ETCHASKET üzerinde RECTANGLE gri GRAFFITI_WALL üzerinde RECTANGLE yeşil ETCHASKET üzerinde QUADRILATERAL yeşil GRAFFITI_WALL üzerinde QUADRILATERAL mavi ETCHASKET üzerinde PARALELOGRAMCHASET mavi GRAFFITI_WALL üzerinde PARALLELOGRAM Sarı POLYGON ETCHASKET üzerinde Sarı GRAFFITI_WALL üzerinde POLYGONdraw a mor ETCHASKET üzerinde RECTANGLE mor GRAFFITI_WALL üzerinde RECTANGLE

Kurmak

SHAPE veya SURFACE'e bakmadan önce, çift dağıtımımızın yüksek düzeyde ayrıştırılmış kullanımını incelememiz gerekir.

Ziyaretçi modeli

Ziyaretçi örüntüsü, bir veri yapısının öğelerini (ör. Liste, ağaç vb.) Polimorfik olarak ziyaret eden bir ziyaretçi nesnesi yoluyla çalışır, ziyaret edilen hedef yapısındaki polimorfik öğe nesnelere karşı bazı eylemler (çağrı veya aracı) uygular.

Aşağıdaki örneğimizde, her birini polimorfik bir YÜZEY ile ziyaret ederek, ŞEKİL'in YÜZEY üzerine çizilmesini isteyen polimorfik SHAPE nesnelerinin bir listesini yapıyoruz.

 1 	Yapmak 2 			- Yüzeylere şekiller yazdırın. 3 		yerel 4 			l_shapes: ARRAYED_LIST [ŞEKİL] 5 			l_surfaces: ARRAYED_LIST [YÜZEY] 6 		yapmak 7 			oluşturmak l_shapes.Yapmak (6) 8 			l_shapes.uzatmak (oluşturmak {ÇOKGEN}.make_with_color ("kırmızı")) 9 			l_shapes.uzatmak (oluşturmak {DİKDÖRTGEN}.make_with_color ("gri"))10 			l_shapes.uzatmak (oluşturmak {DÖRTLÜ}.make_with_color ("yeşil"))11 			l_shapes.uzatmak (oluşturmak {PARALELKENAR}.make_with_color ("mavi"))12 			l_shapes.uzatmak (oluşturmak {ÇOKGEN}.make_with_color ("Sarı"))13 			l_shapes.uzatmak (oluşturmak {DİKDÖRTGEN}.make_with_color ("mor"))14 15 			oluşturmak l_surfaces.Yapmak (2)16 			l_surfaces.uzatmak (oluşturmak {ETCHASKETCH}.Yapmak)17 			l_surfaces.uzatmak (oluşturmak {GRAFFITI_WALL}.Yapmak)18 19 			karşısında l_shapes gibi ic_shapes döngü20 				karşısında l_surfaces gibi ic_surfaces döngü21 					ic_surfaces.eşya.drawing_agent (ic_shapes.eşya.drawing_data_agent)22 				son23 			son24 		son

SHAPE ve SURFACE nesnelerinden oluşan bir koleksiyon oluşturarak başlıyoruz. Daha sonra listelerden birini (ŞEKİL) yineleriz, diğerinin (YÜZEY) öğelerinin sırayla her birini ziyaret etmesine izin veririz. Yukarıdaki örnek kodda, SURFACE nesneleri SHAPE nesnelerini ziyaret ediyor.

Kod, {SURFACE} üzerinde polimorfik bir çağrı yapar. Çift gönderim modelinin ilk çağrısı (gönderim) olan `drawing_agent 'aracılığıyla dolaylı olarak geri çekilir. Dolaylı ve polimorfik bir aracı (`` drawing_data_agent '') geçirerek ziyaretçi kodumuzun yalnızca iki şeyi bilmesini sağlar:

  • Yüzeyin çizim aracısı nedir (ör. 21. satırdaki al_surface.drawing_agent)?
  • Şeklin çizim verisi aracısı nedir (ör. 21. satırdaki al_shape.drawing_data_agent)?

Hem SURFACE hem de SHAPE kendi temsilcilerini tanımladığından, ziyaretçi kodumuz polimorfik veya başka bir şekilde yapmak için uygun çağrının ne olduğunu bilmek zorunda kalmaz. Bu seviyeye indirgeme ve ayrıştırma, C, C ++ ve Java gibi diğer yaygın dillerde, ya bir tür yansıma ya da imza eşleştirme ile aşırı özellik yüklemesi haricinde, elde edilemez.

YÜZEY

{SURFACE} .draw'a yönelik polimorfik çağrı içinde, çift gönderim modelinde ikinci polimorfik çağrı veya gönderim haline gelen bir aracıya çağrıdır.

 1 	ertelenmiş sınıf 2 		YÜZEY 3 	 4 	özellik {YOK} - Başlatma 5 	 6 		Yapmak 7 				- Akımı Başlat. 8 			yapmak 9 				drawing_agent := ajan çizmek10 			son11 	12 	özellik -- Giriş13 14 		drawing_agent: PROSEDÜR [HİÇ, TUPLE [STRING, STRING]]15 				- Akımın çizim temsilcisi.16 	17 	özellik {YOK} - Uygulama18 	19 		çizmek (a_data_agent: FONKSİYON [HİÇ, TUPLE, TUPLE [isim, renk: STRING]])20 				- Current üzerine “a_shape '' çizin.21 			yerel22 				l_result: TUPLE [isim, renk: STRING]23 			yapmak24 				l_result := a_data_agent (Geçersiz)25 				Yazdır ("çizmek " + l_result.renk + " " + l_result.isim + "açık" + tip + "% N")26 			son27 	28 		tip: STRING29 				- Current adını yazın.30 			ertelenmiş son31 	32 	son

19. satırdaki aracı argümanı ve 24. satırdaki çağrı hem polimorfik hem de ayrıştırılmıştır. Ajan ayrıştırılmıştır çünkü {SURFACE} .draw özelliği "a_data_agent" in hangi sınıfa dayandığına dair hiçbir fikri yoktur. Operasyon aracısının hangi sınıftan türetildiğini söylemenin bir yolu yoktur, bu nedenle SHAPE'den veya onun soyundan gelenlerden gelmek zorunda değildir. Bu, Eyfel ajanlarının diğer dillerin tek miras, dinamik ve polimorfik bağlanmasına göre belirgin bir avantajıdır.

Aracı, çalışma zamanında dinamik olarak polimorfiktir çünkü nesne ihtiyaç duyulduğu anda dinamik olarak yaratılır, burada nesneleştirilmiş yordamın sürümü o zamanda belirlenir. Güçlü bir şekilde bağlanan tek bilgi, aracı imzasının Sonuç tipidir - yani - iki öğeli bir adlandırılmış TUPLE. Bununla birlikte, bu özel gereklilik, çevreleyen özelliğin talebine dayanmaktadır (örneğin, 25 numaralı satır, YÜZEY'in `` çizim '' özelliğini yerine getirmek için TUPLE'ın adlandırılmış öğelerini kullanır), bu gerekli ve kaçınılmamıştır (ve belki de olamaz) .

Son olarak, yalnızca `` drawing_agent '' özelliğinin HERHANGİ bir istemciye aktarıldığına dikkat edin! Bu, ziyaretçi desen kodunun (bu sınıfın YALNIZCA müşterisidir) yalnızca işini yapmak için temsilci hakkında bilgi sahibi olması gerektiği anlamına gelir (örneğin, aracıyı ziyaret edilen nesnelere uygulanan özellik olarak kullanma).

ŞEKİL

SHAPE sınıfı, belki bir YÜZEYDE çizilenler için temele (örneğin çizim verileri) sahiptir, ancak olması zorunlu değildir. Yine, aracılar, SHAPE ile eş-değişken ilişkisini mümkün olduğunca ayrıştırılmış hale getirmek için gereken dolaylı ve sınıf agnostiklerini sağlar.

Ek olarak, lütfen SHAPE'ın yalnızca `` drawing_data_agent '' özelliğini herhangi bir istemciye tamamen dışa aktarılan bir özellik olarak sağladığını unutmayın. Bu nedenle, oluşturma dışında SHAPE ile etkileşime girmenin tek yolu, HERHANGİ bir müşteri tarafından SHAPE için çizim verilerini dolaylı ve polimorfik olarak toplamak için kullanılan `` drawing_data_agent '' olanaklarıdır!

 1 	ertelenmiş sınıf 2 		ŞEKİL 3 	 4 	özellik {YOK} - Başlatma 5 	 6 		make_with_color (bir renk: sevmek renk) 7 				- “a_color '' ile“ color '' yapın. 8 			yapmak 9 				renk := bir renk10 				drawing_data_agent := ajan drawing_data11 			sağlamak12 				color_set: renk.same_string (bir renk)13 			son14 15 	özellik -- Giriş16 	17 		drawing_data_agent: FONKSİYON [HİÇ, TUPLE, sevmek drawing_data]18 				- Çizim için veri aracısı.19 	20 	özellik {YOK} - Uygulama21 	22 		drawing_data: TUPLE [isim: sevmek isim; renk: sevmek renk]23 				- Akımın çizimi için gerekli veriler.24 			yapmak25 				Sonuç := [isim, renk]26 			son27 	28 		isim: STRING29 				- Current'ın nesne adı.30 			ertelenmiş son31 	32 		renk: STRING33 				- Akımın Rengi.34 35 	son

Klasik Uzay Gemisi örneği

Klasik Uzay Gemisi örneğinin bir varyasyonu, haydut asteroitler ve uzay istasyonları gibi diğer öğelerle dolu bir evrende dolaşan bir veya daha fazla uzay gemisi nesnesine sahiptir. İstediğimiz şey, inanma evrenimizdeki iki eş değişken nesne arasındaki karşılaşmaları (örneğin olası çarpışmaları) ele almak için çift gönderim yöntemidir.Aşağıdaki örneğimizde, USS Enterprise ve USS Excelsior'umuzun çıktı gezintisi:

Starship Enterprise, pozisyonunu A-001'den A-002'ye değiştiriyor.Starship Enterprise, Asteroid `` Rogue 1 '' den kaçınarak kaçınma eylemi gerçekleştiriyor! Starship Enterprise, pozisyonunu A-002'den A-003'e değiştiriyor.Starship Enterprise, Asteroid `` Rogue 2 '' den kaçınarak kaçınma eylemi gerçekleştiriyor Starship Enterprise, geçerken bir bilim ekibini Starship Excelsior'a ışınlıyor! Starship Enterprise, pozisyonunu A-003'ten A-004'e değiştiriyor.Starship Excelsior, konumunu A-003'ten A-005'e değiştiriyor.Starship Enterprise, Asteroid'den kaçınarak kaçınma eylemi yapıyor '' Rogue 3 '! Starship Excelsior, Space Station Deep Space 9'un yakınında ve yuvaya yerleştirilebilir.Starship Enterprise, konumunu A-004'ten A-005'e değiştiriyor.Starship Enterprise, bir bilim ekibini geçerken Starship Excelsior'a ışınlıyor! Starship Enterprise, Uzay İstasyonu Derin'in yakınında Alan 9 ve yuvaya yerleştirilebilir.

Ziyaretçi

Klasik Uzay Gemisi örneğinin ziyaretçisi ayrıca çift gönderim mekanizmasına sahiptir.

 1 Yapmak 2 		- SPACESHIP nesnelerinin bir evreni ziyaret etmesine ve hareket etmesine izin verin. 3 	yerel 4 		l_universe: ARRAYED_LIST [SPACE_OBJECT] 5 		l_enterprise, 6 		l_excelsior: UZAY GEMİSİ 7 	yapmak 8 		oluşturmak l_enterprise.make_with_name ("Kurumsal", "A-001") 9 		oluşturmak l_excelsior.make_with_name ("Excelsior", "A-003")10 		oluşturmak l_universe.Yapmak (0)11 		l_universe.güç (l_enterprise)12 		l_universe.güç (oluşturmak {Asteroit}.make_with_name ("Rogue 1", "A-002"))13 		l_universe.güç (oluşturmak {Asteroit}.make_with_name ("Rogue 2", "A-003"))14 		l_universe.güç (l_excelsior)15 		l_universe.güç (oluşturmak {Asteroit}.make_with_name ("Rogue 3", "A-004"))16 		l_universe.güç (oluşturmak {UZAY İSTASYONU}.make_with_name ("Derin Uzay 9", "A-005"))17 		ziyaret etmek (l_enterprise, l_universe)18 		l_enterprise.pozisyonu ayarla ("A-002")19 		ziyaret etmek (l_enterprise, l_universe)20 		l_enterprise.pozisyonu ayarla ("A-003")21 		ziyaret etmek (l_enterprise, l_universe)22 		l_enterprise.pozisyonu ayarla ("A-004")23 		l_excelsior.pozisyonu ayarla ("A-005")24 		ziyaret etmek (l_enterprise, l_universe)25 		ziyaret etmek (l_excelsior, l_universe)26 		l_enterprise.pozisyonu ayarla ("A-005")27 		ziyaret etmek (l_enterprise, l_universe)28 	son29 özellik {YOK} - Uygulama30 ziyaret etmek (a_object: SPACE_OBJECT; a_universe: ARRAYED_LIST [SPACE_OBJECT])31 		- "a_object" a_universe'yi ziyaret eder.32 	yapmak33 		karşısında a_universe gibi ic_universe döngü34 			Kontrol ekli {SPACE_OBJECT} ic_universe.eşya gibi al_universe_object sonra35 				a_object.meet_agent.telefon etmek ([al_universe_object.sensor_data_agent])36 			son37 		son38 	son

Çift gönderim, iki dolaylı aracının birbirleriyle mükemmel polimorfik uyum içinde çalışan iki eş değişken çağrısı sağlamak için birlikte çalıştığı 35 numaralı satırda görülebilir. "Ziyaret" özelliğinin "a_object" öğesi, "al_universe_object" den gelen "sensor_data_agent" sensör verileriyle çağrılan bir "meet_agent" e sahiptir.Bu özel örneğin diğer ilginç kısmı, SPACE_OBJECT sınıfı ve onun `` karşılaşma özellik:

Ziyaretçi eylemi

Bir SPACE_OBJECT'in dışa aktarılan tek özellikleri, karşılaşma ve sensör verileri için aracılar ve yeni bir konum belirleme kapasitesidir. Bir nesne (uzay gemisi) evrendeki her nesneyi ziyaret ettiğinde, sensör verileri toplanır ve karşılaşma aracısındaki ziyaret nesnesine iletilir. Burada, sensor_data_agent'ten gelen sensör verileri (yani sensor_data_agent sorgusu tarafından döndürülen sensor_data TUPLE'ın veri öğesi öğeleri) mevcut nesneye karşı değerlendirilir ve bu değerlendirmeye dayalı olarak bir eylem süreci gerçekleştirilir (bkz. SPACE_OBJECT aşağıda). Diğer tüm veriler {NONE} olarak dışa aktarılır. Bu, Private'ın C, C ++ ve Java kapsamlarına benzer. Dışa aktarılmayan özellikler olarak, veriler ve rutinler yalnızca her bir SPACE_OBJECT tarafından dahili olarak kullanılır. Son olarak, karşılaşma çağrılarının, SPACE_OBJECT'in olası soy sınıfları hakkında belirli bilgileri içermediğini unutmayın! Kalıtımda bu düzeyde bulunan tek şey, tamamen genel bir SPACE_OBJECT'in özniteliklerinden ve rutinlerinden bilinebileceklere dayanan genel ilişkisel yönlerdir. Yıldız gemileri, uzay istasyonları ve asteroitler hakkında bildiklerimize veya hayal ettiklerimize dayanarak, insanlar olarak `` baskı''nın çıktısının bize mantıklı gelmesi, yalnızca mantıklı bir planlama veya tesadüftür. SPACE_OBJECT, soyundan gelenler hakkında herhangi bir spesifik bilgi ile programlanmamıştır.

 1 ertelenmiş sınıf 2 SPACE_OBJECT 3 özellik {YOK} - Başlatma 4 make_with_name (bir isim: sevmek isim; bir pozisyon: sevmek durum) 5     - "a_name 've" a_position' ile Current'ı başlatın. 6   yapmak 7     isim := bir isim 8     durum := bir pozisyon 9     sensor_data_agent := ajan sensor_data10     meet_agent := ajan karşılaşmak11   sağlamak12     name_set: isim.same_string (bir isim)13     position_set: durum.same_string (bir pozisyon)14   son15 özellik -- Giriş16 meet_agent: PROSEDÜR [HİÇ, TUPLE]17     - Current ile karşılaşmaları yönetmek için ajan.18 sensor_data_agent: FONKSİYON [HİÇ, TUPLE, ekli sevmek sensor_data_anchor]19     - Akımın sensör verilerini döndürmek için aracı.20 özellik - Ayarlar21 pozisyonu ayarla (bir pozisyon: sevmek durum)22     - "a_position" ile "konumu" ayarlayın.23   yapmak24     Yazdır (tip + " " + isim + "konumundan konumunu değiştirir" + durum + "to" + bir pozisyon + ".% N")25     durum := bir pozisyon26   sağlamak27     position_set: durum.same_string (bir pozisyon)28   son29 özellik {YOK} - Uygulama30 karşılaşmak (a_sensor_agent: FONKSİYON [HİÇ, TUPLE, ekli sevmek sensor_data_anchor])31     - “a_radar_agent '' ile Akımın çarpışma durumunu tespit edin ve rapor edin.32   yapmak33     a_sensor_agent.telefon etmek ([Geçersiz])34     Kontrol ekli {sevmek sensor_data_anchor} a_sensor_agent.last_result gibi al_sensor_data sonra35       Eğer değil isim.same_string (al_sensor_data.isim) sonra36         Eğer (durum.same_string (al_sensor_data.durum)) sonra37           Eğer ((al_sensor_data.is_dockable ve is_dockable) ve38               (is_manned ve al_sensor_data.is_manned) ve39               (is_manueverable ve al_sensor_data.is_not_manueverable)) sonra40             Yazdır (tip + " " + isim + " yakınında " + al_sensor_data.tip + " " +41                 al_sensor_data.isim + "ve yuvaya yerleştirilebilir.% N")42           Aksi takdirde ((is_dockable ve al_sensor_data.is_dockable) ve43                 (is_manned ve al_sensor_data.is_manned) ve44                 (is_manueverable ve al_sensor_data.is_manueverable)) sonra45             Yazdır (tip + " " + isim + "bir bilim ekibini ışınlıyor" + al_sensor_data.tip + " " +46                 al_sensor_data.isim + "geçtikçe!% N")47           Aksi takdirde (is_manned ve al_sensor_data.is_not_manned) sonra48             Yazdır (tip + " " + isim + "kaçınarak kaçınma hareketi yapar" +49                 al_sensor_data.tip + " `" + al_sensor_data.isim + "'!% N")50           son51         son52       son53     son54   son55 isim: STRING56     - Akımın Adı.57 tip: STRING58     - Akım Tipi.59   ertelenmiş60   son61 durum: STRING62     - Akım Konumu.63 is_dockable: BOOLE64     - Current başka bir insanlı nesneyle kenetlenebilir mi?65   ertelenmiş66   son67 is_manned: BOOLE68     - Current insanlı bir nesne mi?69   ertelenmiş70   son71 is_manueverable: BOOLE72     - Current hareket ettirilebilir mi?73   ertelenmiş74   son75 sensor_data: ekli sevmek sensor_data_anchor76     - Akımın Sensör verileri.77   yapmak78       Sonuç := [isim, tip, durum, is_dockable, değil is_dockable, is_manned, değil is_manned, is_manueverable, değil is_manueverable]79     son80 81   sensor_data_anchor: çıkarılabilir TUPLE [isim, tip, durum: STRING; is_dockable, is_not_dockable, is_manned, is_not_manned, is_manueverable, is_not_manueverable: BOOLE]82       - Sensör veri tipi Akım çapası.83 84 son

SPACE_OBJECT'in üç alt sınıfı vardır:

SPACE_OBJECTAsteroitUZAY GEMİSİUZAY İSTASYONU

Örneğimizde, ASTEROID sınıfı `` Rogue '' öğeler için, SPACESHIP için iki yıldızlı gemi ve Deep Space Nine için SPACESTATION kullanılmaktadır. Her sınıfta, tek uzmanlık, "tür" özelliğinin ve nesnenin belirli özelliklerinin ayarlanmasıdır. `` Ad '', oluşturma rutininde ve `` pozisyonda '' verilir.Örneğin: Aşağıda SPACESHIP örneği verilmiştir.

 1 sınıf 2 UZAY GEMİSİ 3 miras almak 4 SPACE_OBJECT 5 oluşturmak 6 make_with_name 7 özellik {YOK} - Uygulama 8 tip: STRING = "Starship" 9   - <Ön imleç>10 is_dockable: BOOLE = Doğru11   - <Ön imleç>12 is_manned: BOOLE = Doğru13   - <Ön imleç>14 is_manueverable: BOOLE = Doğru15   - <Ön imleç>16 son

Yani, evrenimizdeki herhangi bir UZAY GEMİSİ yanaşabilir, insanlı ve manevra kabiliyetine sahiptir. Asteroitler gibi diğer nesneler bunların hiçbiri değildir. Öte yandan, bir UZAY YOLU hem yerleştirilebilir hem de insanlıdır, ancak manevra kabiliyeti yoktur. Böylece, bir nesne diğeriyle karşılaştığı zaman, önce konumların onları birbirinin yakınına getirip getirmediğini kontrol eder ve eğer öyleyse, nesneler temel özelliklerine göre etkileşime girer. name aynı nesne olarak kabul edildiğinden, mantıksal olarak bir etkileşime izin verilmez.

Eyfel örneği sonucu

İkili gönderime gelince, Eiffel, tasarımcı ve programcının, sınıf rutinlerini aracılar yaparak sınıflarından ayırarak ve daha sonra doğrudan nesne özelliği yapmak yerine bu aracıları geçerek doğrudan nesneden nesneye bilgi düzeyini daha da kaldırmasına olanak tanır. aramalar. Temsilcilerin ayrıca belirli imzaları ve olası sonuçları (sorgular durumunda) vardır, bu da onları ideal kılar statik tip kontrolü belirli nesne detaylarından vazgeçmeden araçlar. Aracılar tamamen polimorfiktir, böylece ortaya çıkan kod yalnızca yerel işini yapmak için gereken özel bilgiye sahiptir. Aksi takdirde, birçok eş-değişken nesnenin etrafına yayılmış belirli dahili sınıf özelliği bilgisine sahip olunmasıyla eklenen hiçbir bakım yükü yoktur. Aracıların kullanımı ve mekaniği bunu sağlar. Temsilci kullanımının olası bir dezavantajı, bir temsilcinin hesaplama açısından doğrudan arama muadilinden daha pahalı olmasıdır. Bunu akılda tutarak, aracıların ikili dağıtımda kullanıldığını ve bunların ziyaretçi modellerinde uygulandığını asla varsaymamak gerekir. Ortak değişken etkileşimlerine dahil olacak sınıf türlerinin alanına ilişkin bir tasarım sınırını açıkça görebiliyorsa, o zaman doğrudan çağrı, hesaplama masrafı açısından daha verimli bir çözümdür. Bununla birlikte, katılan türlerin sınıf alanının önemli ölçüde artması veya farklılık göstermesi bekleniyorsa, aracılar, çift dağıtım modelinde bakım yükünü azaltmak için mükemmel bir çözüm sunar.

Ayrıca bakınız

Referanslar

  1. ^ Çoklu Polimorfizmi Ele Almak İçin Basit Bir Teknik. Bildirilerinde OOPSLA '86, Nesneye Yönelik Programlama Sistemleri, Diller ve Uygulamalar, sayfa 347–349, Kasım 1986. SIGPLAN Bildirimleri, 21 (11) olarak basılmıştır. ISBN  0-89791-204-7
  2. ^ Daha Etkili C ++, Scott Meyers (Addison-Wesley, 1996)
  3. ^ "Type dynamic (C # Programlama Kılavuzu) kullanma". Microsoft Geliştirici Ağı. Microsoft. 30 Eyl 2009. Alındı 25 Mayıs 2016. ... Aşırı yük çözümlemesi, bir yöntem çağrısındaki bir veya daha fazla bağımsız değişkenin dinamik türüne sahip olması durumunda derleme zamanında değil çalışma zamanında gerçekleşir ...