540 likes | 839 Views
Polimorfizm. Altuğ Altıntaş kitabından Dr. Mustafa Cem Kasapbaşı. Bir çok biçim. Alt sınıf üst sınıfın tüm özelliklerini alır, isterse fazladan özelliğe da sahip olabilir. class Er extends Asker { public void selamVer () { System.out.println ("Er Selam verdi"); } }. class Asker {
E N D
Polimorfizm Altuğ Altıntaş kitabından Dr. Mustafa Cem Kasapbaşı
Bir çok biçim • Alt sınıf üst sınıfın tüm özelliklerini alır, isterse fazladan özelliğe da sahip olabilir. class Er extends Asker { publicvoidselamVer() { System.out.println("Er Selam verdi"); } } class Asker { publicvoidselamVer() { System.out.println("Asker Selam verdi"); } } publicclassPolimorfizmOrnekBir { publicstaticvoidhazirOl(Asker a) { a.selamVer(); // ! Dikkat ! } publicstaticvoid main(Stringargs[]) { Asker a = new Asker(); Er e = new Er(); Yuzbasi y = newYuzbasi(); hazirOl(a); // yukarı cevirim ! yok ! hazirOl(e); // yukarı cevirim (upcasting) hazirOl(y); // yukarı cevirim (upcasting) } } classYuzbasiextends Asker { publicvoidselamVer() { System.out.println("Yuzbasi Selam verdi"); } }
Üç kavram • Upcasting • hazirOl(Asker a) • Polymorfizm • a.selamVer(); // ! Dikkat • Asker a = new Asker() ; • Asker a = new Er(); • Asker a = newYuzbasi(); • a nesnesi her bir çok nesneye bağlanabildiğini görürüz, bu özellik polimorfizm 'dir -ki bu özelliğin temelinde kalıtım (inheritance) yatar • Latebinding
LateBinding /Dynamic /Run Time • Eğer bir yordamın hangi nesneye ait olduğu çalışma anında belli oluyorsa bu olaya geç bağlama (late-binding) denir. • Hazırol(Asker a) içinde selamver metodunun hangi nesne üzerinde çalışacağına karar verilmesi gibi. (er ve yüzbaşı ) • Bu olayın tam tersi ise erken bağlamadır (earlybinding); yani, hangi nesnenin hangi yordamının çağrılacağı derleme anında bilinmesi.
Örnek 2 class Hayvan { publicvoidavYakala() { System.out.println("Hayvan av Yakala"); } } class Kartal extends Hayvan { publicvoidavYakala() { System.out.println("Kartal av Yakala"); } } class Timsah extends Hayvan{ publicvoidavYakala() { System.out.println("Timsah av Yakala"); } } publicstaticvoid main(Stringargs[]) { Hayvan[] h = new Hayvan[3]; // diziyi doldur for (int i = 0 ; i < 3 ; i++) { h[i] = rasgeleSec(); //upcasting } // dizi elemanlarini ekrana bas for (int j = 0 ; j < 3 ; j++) { h[j].avYakala(); // !Dikkat! } } } publicclassPolimorfizmOrnekIki { publicstatic Hayvan rasgeleSec() { intsec = ( (int) (Math.random() *3) ) ; Hayvan h = null ; if (sec == 0) h = new Hayvan(); if (sec == 1) h = new Kartal(); if (sec == 2) h = new Timsah(); return h; }
Final ve Geç Bağlama • Bir yordamı final yaparak şunu demiş oluruz, bu yordam, türetilmiş olan alt sınıfların içerisindeki diğer yordamlar tarafından iptal edilemesin (override) Eğer bir yordam iptal edilemezse o zaman geç bağlama (latebinding) özelliği de ortadan kalkar. class Kaplan extends Kedi { publicstaticvoidgoster(Kedi k) { k.yakalaAv(); } publicvoidyakalaAv() { System.out.println("Kaplan sinifi Av yakaladi"); } publicstaticvoid main(Stringargs[] ) { Kedi k = new Kedi() ; Kaplan kp = new Kaplan(); goster(k); goster(kp); // yukaridogru cevirim (upcasting) } } class Kedi { publicvoidyakalaAv() { System.out.println("Kedi sinifi Av yakaladi"); } }
class Kedi2 { public final void yakalaAv() { System.out.println("Kedi sinifi Av yakaladi"); } } class Kaplan2 extends Kedi2 { publicstaticvoidgoster(Kedi2 k) { // k.yakalaAv(); // ! dikkat ! } /* iptal edemez publicvoidyakalaAv() { System.out.println("Kaplan sinifi Av yakaladi"); } */ publicstaticvoid main(Stringargs[] ) { Kedi2 k = new Kedi2() ; Kaplan2 kp = new Kaplan2(); goster(k); goster(kp); } }
Neden Polimorfizm? publicclassIsYeri { publicstaticvoidmesaiBasla(Calisan[] c ) { for(int i = 0 ; i < c.length ; i++) { c[i].calis(); // !Dikkat! } } publicstaticvoid main(Stringargs[]) { Calisan[] c = newCalisan[4]; c[0] = newCalisan(); // yukarı cevirim gerekmiyor c[1] = newProgramci(); // yukarı cevirim (upcasting) c[2] = newPazarlamaci(); // yukarı cevirim (upcasting) c[3] = new Mudur(); // yukarı cevirim (upcasting) mesaiBasla(c); } }
Genişletilebilirlik(Extensibility) • Büyük iş yeri publicclassBuyukIsYeri { publicstaticvoidmesaiBasla(Calisan[] c ) { for (int i = 0 ; i < c.length ; i++) { c[i].calis(); // ! Dikkat ! } } publicstaticvoid main(Stringargs[]) { Calisan[] c = newCalisan[7]; c[0]=newCalisan(); //yukarı cevirim gerekmiyor c[1]=newProgramci(); //yukarı cevirim (upcasting) c[2]=newPazarlamaci(); // yukarı cevirim (upcasting) c[3]=new Mudur(); // yukarı cevirim (upcasting) c[4]=newGenelMudur(); // yukarı cevirim (upcasting) c[5]=newAnalizProgramci(); // yukarı cevirim (upcasting) c[6]=newSistemProgramci(); // yukarı cevirim (upcasting) mesaiBasla(c); } ProgramciCalisiyor PazarlamaciCalisiyor Mudur Calisiyor GenelMudurCalisiyor ProgramciCalisiyor ProgramciCalisiyor
Soyut Sınıflar ve Yordamlar (AbstractClassesandMethods) • Soyut (abstract) sınıflarımızı direk new()ile oluşturamayız. • birleştirici bir rol • Soyut sınıflardan türeyen sınıflar yukarı çevirim yapabilirler • Bir sınıfın soyut olması için, bu sınıfın içerisinde en az bir adet soyut yordamının bulunması gerekir. • Soyut yordamların gövdesi bulunmaz; yani, içi boş hiçbir iş yapmayan yordam görünümündedirler • Soyut bir sınıfdan türetilmiş alt sınıflar, bu soyut sınıfın içerisindeki soyut yordamları kesin olarak iptal etmeleri (override) gerekmektedir. • Soyut sınıfların içerisinde soyut yordamlar olacağı gibi, gövdeleri olan, yani iş yapan yordamlarda bulunabilir.
Örnek abstractclassCalisan { publicString pozisyon="Calisan" ; publicabstractvoidcalis() ;// soyut yordam publicvoidzamIste() { // soyut olmayan yordam System.out.println("CalisanzamIste"); } }
Niye Soyut Sınıf ve Yordamlara İhtiyaç Duyarız • Cizim sınıfımızın içerisinde bulunan cizgiCiz() yordamı soyut (abstract) değildir fakat noktaCiz() yordamı soyuttur, neden? Sebebi, cizgiCiz() yordamının ekranlara çizgi çizmek için noktaCiz() yordamına ihtiyaç duymasından kaynaklanır.cizgiCiz() yordamının ihtiyaç duyduğu tek şey, ekran üzerinde tek bir noktanın nasıl çizileceğini bilmektir, bu bilgiler cizgiCiz() yordamına verildiği sürece sorun yaşanmayacaktır. Ekrana tek bir noktanın nasıl çizileceğini, Cizim sınıfından türemiş alt sınıflar tarafından verilmektedir. ÖZET.. eğer bir işlem değişik verilere ihtiyaç duyup aynı işi yapıyorsa, bu işlem soyut (abstract) sınıfın içerisinde tanımlanmalıdır.
Yapılandırıcılar İçerisindeki İlginç Durumlar abstractclass Sporcu { publicabstractvoidcalis(); public Sporcu() { // yapılandırıcı yordam System.out.println("calis() cagrilmadan evvel"); calis(); // ! Dikkat ! System.out.println("calis() cagrildiktan sonra"); } } class Futbolcu extends Sporcu { intantraman_sayisi = 4 ; publicvoidcalis() { System.out.println("Futbolcu calis() " + antraman_sayisi ); } publicFutbolcu() { // yapılandırıcı yordam System.out.println("Futbolcu yapilandirici" ); calis(); } } publicclass Spor { publicstaticvoid main( Stringargs[] ) { Futbolcu f = new Futbolcu(); // Sporcu s = new Sporcu(); // ! Hata soyut sınıf ! } }
Kalıtım ve Yukarı Çevirim (Upcasting) • Yukarı çevirim (upcasting) her zaman güvenlidir, sonuçta daha özellikli bir tipten daha genel bir tipe doğru çevirim gerçekleşmiştir
Yukarı çevirim (upcasting) olayında iki taraf vardır, bir tanesi heap alanında nesnenin kendisi diğer tarafta yığın (stack) alanında bulunan referans Xmodelx_model_kumanda = newYmodel(); x_model_kumanda.sesAc(); x_model_kumanda.sesKapa(); x_model_kumanda.kanalDegistir(); //!! hata !!, bu kumandanın böyle bir düğmesi yok :) // x_model_kumanda.teleText() ;
Aşağıya Çevirim (Downcasting) • daha genel bir tipten, daha özellikli bir tipe doğru geçiş demektir ve tehlikelidir. • Java programlama dilinde aşağıya çevirim (downcasting) yaparken, hangi tipe doğru çevirim yapılacağı açık olarak belirtmelidir. Object obj ="merhaba"; Stringstr= (String) obj; // int i = (int) obj; // hatali System.out.println(i+" "+str);
Object[] ob = new Object[2] ; ob[0] = newXmodel() ; ob[1] = newYmodel() ; for (int i = 0 ; i < ob.length ; i++) { Object o = ob[i] ; if (o instanceofYmodel) { // RTTI Ymodely_model_kumanda = (Ymodel) o ; y_model_kumanda.sesAc(); y_model_kumanda.sesKapa(); y_model_kumanda.kanalDegistir(); y_model_kumanda.teleText() ; } else if (o instanceofXmodel) { // RTTI Xmodelx_model_kumanda = (Xmodel) o; …
ARAYÜZLER VE DAHİLİ SINIFLAR • Interfaceand Inner Classes • Java programlama dilinde çoklu kalıtım desteğinden faydalanmak için arayüz(interface) ve dahili sınıflar (innerclasses) kullanılır.
Arayüz (Interface) • soyut (abstract) sınıfların bir üst modeli gibi düşünelebilir. • Arayüzlerin içerisinde ise iş yapan herhangi bir yordam (method) bulunamaz; arayüzün içerisinde tamamen gövdesiz yordamlar (soyut yordamlar) bulunur • birleştirici bir rol oynamaları için tasarlanmıştır • public erişim belirleyicisine sahip olurlar ve sizin bunu değiştirme imkanınız yoktur
Calisanarayüzü (interface), birleştirici bir rol oynamaktadır
interfaceCalisan { // arayuz publicvoidcalis() ; } class Mudur implementsCalisan{ … classGenelMudurextends Mudur {… classProgramciimplementsCalisan{… classAnalizProgramciextendsProgramci{ … Calisan[] c = newCalisan[6]; // c[0]=newCalisan(); ! Hata ! arayüz yaratılamaz c[0]=newProgramci(); // yukaricevirim (upcasting) c[1]=newPazarlamaci();// yukari cevirim (upcasting) c[2]=new Mudur(); //yukari cevirim (upcasting) c[3]=newGenelMudur(); //yukari cevirim (upcasting) c[4]=newAnalizProgramci();//yukari cevirim //(upcasting) c[5]=newSistemProgramci();//yukari cevirim mesaiBasla(c)… publicstaticvoidmesaiBasla(Calisan[] c ) { for (int i = 0 ; i < c.length ; i++) { c[i].calis(); // ! Dikkat ! }
Arayüz (Interface) ve Soyut Sınıflar (AbstractClasses)Arayüz (Interface) ve Soyut Sınıflar (AbstractClasses) • eğer soyut bir sınıf (abstractclass) bir arayüzeulaşırsa, arayüze ait gövdesiz yordamları kesin olarak, kendi içerisinde iptal etmeli mi? interface Hayvan { publicvoid avlan() ; } abstractclass Kedi implements Hayvan { }
interface Hayvan { publicvoid avlan() ; } abstractclass Kedi implements Hayvan { publicabstractvoidtakipEt() ; } class Kaplan extends Kedi { publicvoid avlan() { // iptal etti (override) System.out.println("Kaplan a++vlaniyor..."); } publicvoidtakipEt() { // iptal etti (override) System.out.println("Kaplan takip ediyor..."); } } Kaplan sınıfı Hayvan arayüzünde (interface)tanımlanmış soyut olan (gövdesiz) avlan() yordamını iptal etmek (override) zorunda mı? EVET
Arayüz (Interface) İle Çoklu Kalıtım (MultipleInheritance)Arayüz (Interface) İle Çoklu Kalıtım (MultipleInheritance) classSportmenMehmetextendsBuzPatenci, Basketbolcu { } //JAVA DA OLMAZ Sporcu s = newSportmenMehmet(); // yukaridogrucevirims.calis(); // ??
interfaceBuzUstundeKayabilme { publicvoidbuzUstundeKay(); } interfaceSutAtabilme { publicvoidsutAt(); } classSportmenMehmetimplementsBuzUstundeKayabilme, SutAtabilme { publicvoidbuzUstundeKay() { System.out.println("SportmenMehmet buz ustunde kayiyor"); } publicvoidsutAt() { System.out.println("SportmenMehmet sut atiyor"); } }
Kavramsal Farklar Soyut vs Arabirim • Soyut bir sınıftan türetilme yapıldığı zaman, türetilen sınıf ile soyut sınıf arasında mantıksal bir ilişki olması gerekirdi, • Yarasa birHayvandır vb • Arabirimde bu şekilde bir ilişki olamyabilir.
Arayüzlerin Kalıtım (İnheritance) Yoluyla GenişletilmesiArayüzlerin Kalıtım (İnheritance) Yoluyla Genişletilmesi interface C { //..} interface B { //..} interface A extendsB { //..} class D implements B,C{…} Class E extends D implements A {…}
Çakışmalar • Arayüzlerin içerisinde dönüş tipleri haricinde her şeyleri aynı olan gövdesiz (soyut) yordamlar varsa, bu durum beklenmedik sorunlara yol açabilir.
interface A1 { publicvoid hesapla(); } interface A2 { publicvoid hesapla(int d); } interface A3 { publicint hesapla(); } class S1 implements A1,A2 { // sorunsuz publicvoid hesapla() { //adas yordamlar(overloaded) System.out.println("S1.hesapla"); } publicvoid hesapla(int d) { //adas yordamlar(overloaded) System.out.println("S1.hesapla " + d ); } } class S2 implements A1,A3 { publicvoid hesapla() { System.out.println("S2.hesapla"); } /* ! Hata ! publicint hesapla() { System.out.println("S2.hesapla"); return 123; } */ } Problem adaş (overload) metodların, derleyici tarasındanhengisinindöndüreleceğinin bilinememesi
Arayüzün (Interface) İçerisinde Alan Tanımlama • Arayüzlerin içerisinde gövdesiz (soyut) yordamların dışında alanlarda bulunabilir. Bu alanlar uygulamalarda sabit olarak kullanabilir. interface Aylar { int OCAK = 1, SUBAT = 2, MART = 3, NISAN = 4, MAYIS = 5, HAZIRAN = 6, TEMMUZ = 7, AGUSTOS = 8, EYLUL = 9, EKIM = 10, KASIM = 11, ARALIK = 12; } publicclassAyBul { publicstaticvoid main(Stringargs[]) { int ay = (int)(Math.random()*13) ; System.out.println("Gelen ay = " + ay); switch ( ay ) { caseAylar.OCAK : System.out.println("Ocak");break; caseAylar.SUBAT : System.out.println("Subat");break; caseAylar.MART : System.out.println("Mart");break; caseAylar.NISAN : System.out.println("Nisan");break; caseAylar.MAYIS : System.out.println("Mayis");break; caseAylar.HAZIRAN : System.out.println("Haziran");break; caseAylar.TEMMUZ : System.out.println("Temmuz");break; ….
Arayüzün İçerisinde Tanımlanmış Alanlara İlk Değerlerinin VerilmesiArayüzün İçerisinde Tanımlanmış Alanlara İlk Değerlerinin Verilmesi interface A7 { intdevir_sayisi = (int) ( Math.random() *6 ) ; String isim = "A7" ; double t1 = ( Math.random() * 8 ) ; } publicclass Test { publicstaticvoid main(Stringargs[] ) { System.out.println("devir_sayisi = " + A7.devir_sayisi ); System.out.println("isim = " + A7.isim ); System.out.println("t1 = " + A7.t1 ); } }
Genel Bakış • Arayüzler ve soyut sınıfların bizlere sağlamak istediği fayda nedir? • Aslında ulaşılmak istenen amaç çoklu yukarı çevirimdir (upcasting). • Bir sınıfa ait nesnenin bir çok tipteki sınıf referansına bağlanabilmesi, uygulama içerisinde büyük esneklik sağlar.
interface Arayuz1 { publicvoid a1() ; } interface Arayuz2 { publicvoid a2() ; } abstractclass Soyut1 { publicabstractvoid s1(); } class B extends Soyut1 implements Arayuz1{ publicvoid s1();} class A extends Soyut1 implements Arayuz1, Arayuz2 { publicvoid s1() { // iptal etti (override) System.out.println("A.s1()"); } publicvoid a1() { // iptal etti (override) System.out.println("A.a1()"); } publicvoid a2() { // iptal etti (override) System.out.println("A.a2()"); } } publicclassGenelBakis { publicstaticvoid main(Stringargs[]) { Soyut1 soyut_1 = new A(); Arayuz1 arayuz_1 = (Arayuz1) soyut_1 ; Arayuz2 arayuz_2 = (Arayuz2) soyut_1 ; // Arayuz2 arayuz_2 = (Arayuz2) arayuz_1 ; // dogru soyut_1.s1(); // soyut_1.a1(); // ! Hata ! // soyut_1.a2(); // ! Hata ! arayuz_1.a1(); // arayuz_1.a2(); // ! Hata ! // arayuz_1.s1(); // ! Hata ! arayuz_2.a2(); // arayuz_2.a1(); // ! Hata ! // arayuz_2.s1(); // ! Hata ! } }
Dahili Arayüzler • Bir arayüz, başka bir arayüzün veya sınıfın içerisinde tanımlanabilir. Bir arayüzün içerisinde tanınlanandahili arayüzler, protected, friendly veya private erişim belirleyicisine sahip olamaz.
interfaceArayuzA { //aslindapublicerisim belirleyicisine sahip publicinterface DahiliArayuz1 { publicvoid isYap1() ; } /* // ! Hata ! protectedinterface DahiliArayuz2 { publicvoid isYap2() ; } */ interface DahiliArayuz3 { // aslindapublicerisimbelirleyicisine sahip publicvoid isYap3() ; } /* // ! Hata ! privateinterface DahiliArayuz4 { publicvoid isYap4() ; } */ }// ARAYUZ SONU class Erisim1 implements ArayuzA.DahiliArayuz1 { publicvoid isYap1() { System.out.println("Erisim1.isYap1()"); } } class Erisim2 implements ArayuzA.DahiliArayuz3 { publicvoid isYap3() { System.out.println("Erisim1.isYap3()"); } } publicclassDahiliArayuzTest { publicstaticvoid main(Stringargs[]) { Erisim1 e1 = new Erisim1(); Erisim2 e2 = new Erisim2(); e1.isYap1(); e2.isYap3(); } }
Sınıfların İçerisinde Tanımlanan Dahili Arayüzler • Bir arayüz diğer bir arayüzün içerisinde tanımlandığı gibi, bir sınıfın içerisinde de tanımlanabilir. publicclassSinifA { publicinterface A1 { publicvoidekranaBas(); } //arayüz …………… }//A sınıfı bitiş classSinifBimplements SinifA.A1{ publicvoidekranaBas() { System.out.println("SinifB.ekranaBas()"); }
Dahili Sınıflar (Inner Classes) • Dahili sınıflar JDK 1.1 ile gelen bir özelliktir. Bu özellik sayesinde bir sınıf diğer bir sınıfın içerisinde tanımlanabilir; böylece mantıksal bir bütünü oluşturan bir çok sınıf tek bir çatı alında toplanır. Dahili sınıflar yapısal olarak 3 gruba ayrılabilir. ∙•••••••Dahili üye sınıflar ∙•••••••Yerel sınıflar (Localclasses) ∙•••••••İsimsiz sınıflar (Anonymousclasses)
Dahili Üye Sınıflar • Bir sınıfın içerisinde, başka bir sınıfı tanımlamak mümkündür • Komposizyondan yönteminden farklıdır. classCevreliyiciSinif { classDahiliSinif { //.... } //... } publicclass Hesaplama { publicclass Toplama { //Dahili uyesinif publicinttoplamaYap(int a, int b) { returna+b ; } } // classToplama publicstaticvoid main(Stringargs[]) { Hesaplama.Toplamaht = new Hesaplama().new Toplama() ; intsonuc = ht.toplamaYap(3,5); System.out.println("Sonuc = " + sonuc ); } } // class Hesapla Sonuc = 8
Dahili Üye Sınıflar ve Erişim • public, protectedveya private erişim belirleyicileri atanabilir. • public anlamı • protected anlamı • private anlamı
class Hesaplama2 { publicclass Toplama2 { // Dahili uyesinif - public publicinttoplamaYap(int a, int b) { return a + b ; } } // class Toplama2 protectedclass Cikartma2 { // Dahili uyesinif - protected publicintcikartmaYap(int a, int b) { return a - b ; } } // class Cikartma2 class Carpma2 { // Dahili uyesinif - friendly publicintcarpmaYap(int a, int b) { return a * b ; } } // class Carpma2 privateclass Bolme2 { // Dahili uyesinif - private publicintbolmeYap(int a, int b) { return a / b ; } } // class Bolme2 } // class Hesaplama2 publicclass Hesaplama2Kullan { publicstaticvoid main(Stringargs[]) { Hesaplama2.Toplama2 ht=new Hesaplama2().new Toplama2() ; Hesaplama2.Cikartma2 hck=new Hesaplama2().new Cikartma2() ; Hesaplama2.Carpma2 hcp = new Hesaplama2().new Carpma2() ; // Hesaplama2.Bolme3 hb = new Hesaplama2().new Bolme2() ; // ! Hata ! int sonuc1 = ht.toplamaYap(10,5); int sonuc2 = hck.cikartmaYap(10,5); int sonuc3 = hcp.carpmaYap(10,5); // int sonuc4 = hb.bolmeYap(10,5); // ! Hata ! System.out.println("Toplama Sonuc = " + sonuc1 ); System.out.println("CikartmaSonuc = " + sonuc2 ); System.out.println("CarpmaSonuc = " + sonuc3 ); } }
Dahili Üye Sınıflar ve Bunları Çevreleyen Sınıflar Arasındaki İlişkiDahili Üye Sınıflar ve Bunları Çevreleyen Sınıflar Arasındaki İlişki • Dahili üye sınıflar, içerisinde bulundukları çevreleyici sınıfların tüm alanlarına (statik veya değil- private dahil) ve yordamlarına (statik veya değil-private dahil) erişebilirler.
Statik Dahili Üye Sınıflar • Statik dahili üye sınıfına ait nesne oluşturmak için, onu çevreleyen sınıfa ait bir nesne oluşmak zorunda değilizdir. • Statik dahili üye sınıflar, kendilerini çevreleyen sınıfa ait bağlantıyı (-this-) kaybederler • Statik dahili üye sınıflar, onları çevreleyen üst sınıfa ait global alanlara (statik veya değil) ve yordamlara (statik veya değil) direk ulaşım şansını kaybeder
// ! Hata ! // Hesaplama4.Toplama4 ht=new Hesaplama4().new Toplama4() ; Hesaplama4.Toplama4 tp4 = new Hesaplama4.Toplama4();
Statik Dahili Üye Sınıflar ve Statik Yordamlar • Eğer statik dahili üye sınıfı içerisinde, statik bir yordam oluşturulmuş ise, bu yordamı çağırmak için ne statik dahili üye sınıfına ne de onu çevreleyen sınıfa ait herhangi bir nesne oluşturmak gerekmez publicclass Hesaplama5 { privatestaticint x = 3 ; publicstaticclass Toplama5 { // Statik uye dahili sinif staticint toplam ; // dogru intsonuc ; // dogru publicstaticinttoplamaYap(int a, int b) { // sonuc = a+b + x ; // ! Hata ! toplam = a + b + x ; return toplam ; } } // classToplama5 publicstaticvoid main(Stringargs[]) { intsonuc = Hesaplama5.Toplama5.toplamaYap(16,8); // dikkat System.out.println("Sonuc = 16 + 8 = " + sonuc ); } } // class Hesaplama5
Statik ve Final Alanlar • Statik olmayan dahili üye sınıfların içerisinde, statik alanlar ve yordamlar tanımlanamaz; ama "statik ve final" alanlar tanımlanabilir
Dahili Üye Sınıflar ve Yapılandırıcılar • Dahili üye sınıfını çevreleyen sınıfa ait bir nesne oluşturulduğu zaman, dahili üye sınıfına ait bir nesne otomatik oluşturulmaz. publicclassBuyukA { publicclass B { public B() { // yapilandirici System.out.println("Ben B sinifi "); } } // class B publicBuyukA() { System.out.println("Ben BuyukAsinifi "); } publicstaticvoid main(Stringargs[]) { BuyukAba = newBuyukA(); BuyukA.Bbb= newBuyukA().new B(); } }
İç içe Dahili Üye Sınıflar • Bir sınıfın içerisinde dahili üye sınıf tanımlayabilirsiniz. Tanımlanan bu dahili üye sınıfın içerisinde, yine bir dahili üye sınıf tanımlayabilirsiniz... bu böyle sürüp gidebilir... publicclassAbc { publicAbc() { // Yapilandirici System.out.println("Abc nesnesi olusturuluyor"); } publicclass Def { public Def() { // Yapilandirici System.out.println("Def nesnesi olusturuluyor"); } publicclassGhi { publicGhi() { // Yapilandirici System.out.println("Ghi nesnesi olusturuluyor"); } } // classGhi } //class Def publicstaticvoid main( Stringargs[] ) { Abc.Def.Ghiici_ice = newAbc().new Def().newGhi(); } } // classAbc Abc nesnesi olusturuluyor Def nesnesi olusturuluyor Ghi nesnesi olusturuluyor
Soyut (Abstract) Dahili Üye Sınıflar • Dahili üye sınıflar, soyut (abstract) sınıf olarak tanımlanabilir. Bu soyut dahili üye sınıflardan türeyen sınıflar, soyut dahili üye sınıfların içerisindeki gövdesiz (soyut) yordamları iptal etmeleri gerekmektedir. class Kartal extendsHayvan.Kus {
Türetilebilen Dahili Üye Sınıflar • Dahili üye sınıflar, aynı normal sınıflar gibi başka sınıflardan türetilebilirler.
Yerel Sınıflar (LocalClasses) • Yerel sınıflar, yapılandırıcıların (constructor), sınıf yordamlarının (statik yordam), nesne yordamların, statik alanlara toplu değer vermek için kullandığımız statik bloğun veya statik olmayan alanlara toplu değer vermek için kullandığımız bloğun içerisinde tanımlanabilir. • Yerel sınıflar tanımlandıkları yordamın veya bloğun dışından erişilemezler • Yerel sınıflar başka sınıflardan türetilebilir veya arayüzlere (interface) erişebilir. • Yerel sınıfların yapılandırıcıları olabilir.