Özlem aydin trakya Üniversitesi bilgisayar mühendisliği bölümü

36
NESNEYE YÖNELİK PROGRAMLAMA KOPYA YAPICI FONKSİYON, STATİK ELEMANLAR, ARKADAŞ SINIF VE FONKSİYONLAR,NESNE DİZİLERİ Özlem AYDIN Trakya Üniversitesi Bilgisayar Mühendisliği Bölümü Not: Bu sunumda Doç. Dr. Yılmaz KILIÇASLAN’ın Nesneye Yönelik Programlama dersi sunumlarından faydalanılmıştır.

Upload: brandi

Post on 21-Jan-2016

86 views

Category:

Documents


0 download

DESCRIPTION

NESNEYE YÖNELİK PROGRAMLAMA KOPYA YAPICI FONKSİYON, STATİK ELEMANLAR, ARKADAŞ SINIF VE FONKSİYONLAR,NESNE DİZİLERİ. Özlem AYDIN Trakya Üniversitesi Bilgisayar Mühendisliği Bölümü. - PowerPoint PPT Presentation

TRANSCRIPT

NESNEYE YÖNELİK PROGRAMLAMAKOPYA YAPICI FONKSİYON, STATİK ELEMANLAR,

ARKADAŞ SINIF VE FONKSİYONLAR,NESNE DİZİLERİ

Özlem AYDIN

Trakya ÜniversitesiBilgisayar Mühendisliği BölümüNot: Bu sunumda Doç. Dr. Yılmaz KILIÇASLAN’ın Nesneye Yönelik Programlama dersi sunumlarından faydalanılmıştır.

Sunum Planı

•Kopya yapıcı fonksiyonu

•Nesnelerin parametre olarak geçirilmesi ve döndürülmesi

•Referansların parametre olarak geçirilmesi ve döndürülmesi

•Statik veri elemanları ve statik fonksiyonlar

•Arkadaş sınıflar ve fonksiyonlar

•Nesne dizileri yaratma

2

ATAMA VE İLK DEĞER ALMA - 1

• C’de atama ve ilk değer alma işlemleri etkileri itibariyle aynıdır. Fakat, C++’da bu işlemlerin farklı etkileri olabilir. İki işlem arasındaki farklar şunlardır:▫Atama mevcut bir nesnenin değerinin

değişmesi esnasında gerçekleşir; bir nesneye birçok yeni değer atanabilir.

▫İlk değer alma bir nesnenin bildirimi yapılırken bir başlangıç değeri alması esnasında gerçekleşir; bir nesne yalnızca bir kez ilk değer alır.

3

ATAMA VE İLK DEĞER ALMA - 2• Atama ve ilk değer alma işlemleri arasındaki fark,

nesnelerle işlem yaparken önemli olabilir. Atama deyimi derleyicinin sınıf için tanımlanmış operator= fonksiyonunu çalıştırmasına neden olurken:String string1(“birinci karakter katari”);String string2;string2 = string1;

ilk değer alma aynı fonksiyonu çalıştırmaz:

String string1(“birinci karakter katari”);String string2 = string1;

operator= fonksiyonu yalnızca daha önce yaratılmış nesneler için çağrılabilir.

4

KOPYA YAPICI FONKSİYON - 1

• Bir nesnenin yaratıldığı esnada bir başka nesnenin değerini almasını sağlamak için özel bir yapıcı fonksiyon, kopya yapıcı fonksiyon, çağrılır.

• Varolan bir nesnedeki verileri yeni oluşturulan nesnenin içine kopyalarlar.

• Giriş parametresi olarak aynı sınıftan bir nesneye referans alırlar.

• Eğer programda bir kopya yapıcı fonksiyon tanımlı değilse, derleyici standart bir yapıcıyı sınıfa yerleştirir. Standart kopya yapıcı fonksiyon bir nesnenin elemanlarını bire bir yeni nesnenin veri alanlarına kopyalar. İçinde işaretçi olmayan nesneler için yeterlidir.

5

KOPYA YAPICI FONKSİYON - 2

• Kopya yapıcı bir fonksiyon bir nesnenin ilk değer olarak bir başka nesneyi aldığı durumlarda çalıştırılır. = işareti ile ya da fonksiyon-çağırma sentaksı ile çağrılır:

String string2 = string1;

String string2( string1 );

• Bir kopya yapıcı fonksiyon olmaması halinde yukarıdaki ifade neticesinde string2 her eleman sahasının ilk değerini string1’in elemanlarından alır. Bu da sınıfın elemanları içinde işaretçilerin bulunması durumunda arzu edilmeyen sonuçlara yol açabilir.

6

KOPYA YAPICI FONKSİYON - 3• String sınıfı için kopya yapıcı fonksiyonu şu şekilde yazılabilir:

#include <iostream>

#include <cstring>

Using namespace std;

class String

{ public:

String();

String( const char *s );

String( char c, int n );

String( const String &digeri ); // kopya yapıcı fonksiyon

// ... };

String::String( const String &digeri )

{

uzunluk = digeri.uzunluk;

buf = new char[uzunluk + 1];

strcpy( buf, digeri.buf ); }

7

KOPYA YAPICI FONKSİYON - 4

• Kopya başlangıç fonksiyonu derleyici tarafından 3 durumda çağırılır:

1. Bir sınıf nesnesi fonksiyona parametre olarak geçirildiğinde.

2. Bir fonksiyondan bir nesne döndürüldüğünde.

3. Var olan bir nesnenin kopyası olarak yeni bir nesneye başlangıç ataması yapıldığında .

NESNELERİN PARAMETRE OLARAK GEÇİRİLMESİ

• Aşağıdaki fonksiyon bir nesneyi parametre olarak almaktadır:

void tuket( String parm ){

// parm nesnesinin kullanımı}

void main(){ String string1(“birinci karakter katari”); tuket(string1); }

tuket fonksiyonu değer olarak geçirilmiş bir String nesnesi almıştır. Yani, fonksiyon nesnenin kendisine ait bir kopyasına sahip olmuştur. Aslında, derleyici örtük olarak fonksiyonun parametresine nesneyi ilk değer olarak atamak üzere, kopya yapıcı fonksiyonu çağırmıştır; yaptığı aşağıdakiyle aynıdır:

String parm( string1 );

9

NESNE DÖNDÜREN FONKSİYONLAR

• Aşağıdaki fonksiyon değer olarak bir nesne döndürmektedir:

String f(){ String retDeger(“bir return değeri”);

return retDeger; }

void main(){ String string2; string2 = f(); }

f fonksiyonu değer olarak bir String nesnesi döndürmektedir. Derleyici, fonksiyonu çağıranın kapsam alanındaki geçici bir gizli nesneye fonksiyonun return ifadesinde belirtilmiş nesneyi kullanarak ilk değer atamak üzere kopya yapıcı fonksiyonu çağırır. Yani, derleyici aşağıdakilere denk işlemler gerçekleştirir:

String gecici( retDeger );string2 = gecici;

10

NESNE REFERANSLARININ PARAMETRE OLARAK GEÇİRİLMESİ• Bir nesnenin değer olarak bir fonksiyona her gönderilişi, bir

miktar “overhead” içerir. Sabit bir nesneye referans geçirerek aynı etki, yapıcı fonksiyon çağırmaktan kaynaklı maliyetten de kurtularak elde edilebilir:void tuket( const String &parm ){ // parm nesnesinin kullanımı

}

void main(){ String string1(“birinci karakter katari”); tuket( string1 ); }

Parametre bu şekilde geçirildiğinde, yeni bir nesne yaratılmadığı için, kopya yapıcı fonksiyon çağrılmaz. Derleyici aşağıdaki işlemin aynısını gerçekleştirir:const String &parm = string1;Sonuç olarak, fonksiyon kendisini çağıranla aynı nesneyi kullanmış olur.

11

NESNE REFERANSLARININ DEĞER OLARAK DÖNDÜRÜLMESİ

• Nesnenin referansını değer olarak döndürmek de, nesnenin kendisini döndürmekten daha etkin bir yöntem olabilir. operator= örneğini hatırlayın:

String &String::operator=(const String &digeri){ //... return *this; }void main(){ String string1(“birinci karakter katari”); String string2, string3; string3 = string2 = string1; }

Geçici bir nesne yaratılmadığı için fonksiyon değer döndürürken kopya yapıcı fonksiyon çağrılmaz: yalnızca geçici bir referans yaratılır.

string2 = string1 ifadesinin değerini string3 nesnesine atarken derleyici aşağıdaki işlemleri yapar:

String &geciciRef = string2;string3 = geciciRef;

12

STATİK VERİ ELEMANLARI - 1

•Bir veri elemanının static olarak tanımlanması mümkündür.

•Bir sınıftan üretilen tüm nesnelerin ortak bir veriyi paylaşmasının gerektiği durumlar olabilir.

•Bellekte sadece tek kopyasının yaratılması istenen veriler static olarak tanımlanmalıdırlar

•Static olarak tanımlanan veri elemanının sadece bir kopyası vardır ve kendi sınıfının tüm nesneleri o değişkeni paylaşır.

•Statik üyeler nesne tanımlanmadan önce bellekte yaratılırlar.

STATİK VERİ ELEMANLARI - 2

•Bir bankadaki tasarruf hesapları için bir TasarrufHesabi sınıfı tanımlayalım.

•Sınıfa ait her nesne belirli bir müşterinin hesap ve isim bilgisini taşıyacak olsun.

•Ayrıca, sınıf her gün kazanılan faizleri hesaba ekleyecek bir eleman fonksiyona sahip olsun.

14

STATİK VERİ ELEMANLARI - 3• Böyle bir sınıf için günlük faiz oranını nasıl

gösterebiliriz?

▫ Sürekli değiştiği için sabit değil, değişken olmalıdır.▫ Sınıfın sıradan bir eleman sahası olması halinde, her

nesne kendi kopyasına sahip olacaktır; bu da yalnızca bellek israfına yol açmakla kalmayacak aynı zamanda faiz oranı değiştiğinde her nesneyi güncellemeyi gerektirecektir.

▫ Faiz oranını global bir değişken yapmak isteyebilirsiniz; fakat bu her fonksiyonun değişkeninizin değerini değiştirebileceği anlamına gelecektir.

▫ İhtiyacımız olan belirli bir sınıf için global bir değişkendir.

15

STATİK VERİ ELEMANLARI - 4• C++, belirli bir sınıf için global değişken tanımlama

imkanını statik eleman sahalar aracılığıyla sağlar. Sıradan bir eleman saha gibi işleme sokulmasına rağmen, bildirimi static yapılmış eleman sahaların, sınıfa ait kaç nesne yaratılırsa yaratılsın, tek bir kopyası oluşturulur.

class TasarrufHesabi{ public:

TasarrufHesabi();void faizEkle()

{ toplam += faizOrani * toplam; }//...

private:char isim[30];float toplam;static float faizOrani;//... };

16

STATİK VERİ ELEMANLARI - 5• Statik bir eleman sahayı, bildirimini public yapmak

suretiyle, program içerisinde erişilebilir kılabilirsiniz:

// Eğer faizOrani public isevoid main(){ TasarrufHesabi hesap1; hesap1.faizOrani = 0.0005; }

• Yukarıdaki programda kullanılan sözdizim meşru fakat aldatıcıdır; çünkü, bütün TasarrufHesabi nesnelerinin faiz oranının değiştiriliyor olmasına rağmen, yalnızca hesap1 nesnesinin faiz oranının değiştirildiği izlenimini doğurmaktadır. Daha iyi bir gösterim yolu aşağıdaki olacaktır:

TasarrufHesabi::faizOrani = 0.0005;

Bu sözdizim değişimin bütün sınıfa uygulanacağı gerçeğini yansıtmaktadır ve hiçbir TasarrufHesabi nesnesi yaratılmamış olsa bile kullanılabilir.

17

STATİK VERİ ELEMANLARI - 6• Statik bir eleman sahanın ilk değerini sınıfının yapıcı

fonksiyonu içinden alması mümkün değildir; çünkü ilk değer bir kez atanır fakat yapıcı fonksiyon birçok kez çağrılabilir.

• Statik bir eleman saha, tıpkı global bir değişken gibi, dosya kapsam alanında ilk değerini almalıdır. Erişim belirteci statik eleman sahalar için ilk değer alma esnasında etkin değildir; dolayısıyla private eleman sahalar ilk değerlerini public eleman sahalar gibi alırlar:

float TasarrufHesabi::faizOrani = 0.0005;

TasarrufHesabi::TasarrufHesabi(){ //... }//...

18

STATİK ELEMAN FONKSİYONLAR - 1• Eğer yalnızca statik eleman sahalara erişim yapan bir

fonksiyonunuz varsa, fonksiyonun bildirimini static yapabilirsiniz:

class TasarrufHesabi{ public:

TasarrufHesabi();void faizEkle()

{ toplam += faizOrani * toplam; }static void faizBelirle( float yeniDeger )

{ faizOrani = yeniDeger; }//...

private:char isim[30];float toplam;static float faizOrani;//... };

19

STATİK ELEMAN FONKSİYONLAR - 2

• Statik eleman fonksiyonlar statik eleman sahalara erişim için kullanılan sözdizim ile çağrılabilirler:

// Statik eleman fonksiyon çağrımıvoid main(){

TasarrufHesabi hesap1;hesap1.faizBelirle( 0.0005 );TasarrufHesabi::faizBelirle( 0.0005 );

}

• Statik eleman fonksiyonlar belirli bir nesne üzerinde işlem yapmadıkları için this işaretçisine sahip değillerdir.

20

STATİK ELEMANLAR• Statik eleman sahalar bütün nesnelerin ihtiyacı olan

ortak kaynakları yönetmek ya da nesnelere ilişkin durum bilgisi tutmak için kullanılabilir. Örneğin, herhangi bir anda bir sınıfın kaç nesnesinin bulunduğu bilgisini statik bir eleman saha üzerinde tutabiliriz:

class Ucak{ public:

Ucak() { sayac++; }static int miktarSoyle() { return sayac; }~Ucak(){ sayac--; }

private:static int sayac; }

int Ucak::sayac = 0;

21

ARKADAŞ SINIFLAR - 1• Bazen iki yada daha fazla sınıfın o kadar işbirliği içerisinde

çalışması gerekir ki birbirlerinin erişim fonksiyonlarını kullanmak yeterince verimli olmayabilir ve dolayısıyla birbirlerinin özel korumalı (private) eleman sahalarına doğrudan erişmeleri gerekebilir. Bu tür bir imkan friend anahtar sözcüğünü kullanarak elde edilir:

class Sinif1{ friend class Sinif2; // Arkadaş sınıf private:

int cokGizli; };

class Sinif2{ public:

void degistir(Sinif1 nesne1) };

void Sinif2::degistir(Sinif1 nesne1){ nesne1.cokGizli++; }

22

ARKADAŞ SINIFLAR - 2

•Örnek1 (Lafore 2002):

#include <iostream>using namespace std;

class alpha{ private:

int data1;public: alpha() : data1(99) { } //constructor friend class beta; //beta is a friend class

};

23

ARKADAŞ SINIFLAR - 3

// all member functions can access private// alpha data

class beta{ public: //access private alpha data

void func1(alpha a) { cout << “\ndata1=” << a.data1; }

void func2(alpha a) { cout << “\ndata1=” << a.data1; }

};

24

ARKADAŞ SINIFLAR - 4

int main(){alpha a;beta b;b.func1(a);b.func2(a);cout << endl;return 0;

}

25

ARKADAŞ SINIFLAR - 5

•friend bildirimi public ve private anahtar sözcüklerinden etkilenmez; sınıf bildiriminin herhangi bir yerinde bulunabilir.

•Tanımlanan sınıf kendi özel korumalı eleman sahalarına erişebilecek arkadaş sınıflar belirleyebilir; fakat, kendisini bir başka sınıfın arkadaşı ilan edemez.

•friend arkadaş sözcüğü tek yönlü erişim imkanı sağlar.

26

ARKADAŞ SINIFLAR - 6

•Arkadaş mekanizmasını kullanmanız halinde yalıtılmış sınıflarla değil, bir arada düşünülmesi gereken iki yada daha fazla sınıfla uğraşıyorsunuz demektir. Bu nedenle, bu mekanizma elden geldiğince seyrek kullanılmalıdır.

27

ARKADAŞ FONKSİYONLAR - 1• Bütün bir sınıfın yerine, tek bir fonksiyon da

arkadaş (friend) olarak bildirilebilir.

class myclass {int a, b;

public:friend int sum(myclass x); // Arkadaş fonksiyon

• Arkadaş fonksiyonları sınıfa ait değildir, ama bu fonksiyonların o sınıfa ait private elemanlara erişim hakkı vardır.

28

ARKADAŞ FONKSİYONLAR - 1• Örnek1 (Osborne 1998):

#include <iostream>using namespace std;class myclass {int a, b;

public:friend int sum(myclass x); // Arkadaş fonksiyonvoid set_ab(int i, int j); };

void myclass::set_ab(int i, int j){ a = i;b = j; }

29

ARKADAŞ FONKSİYONLAR - 2// sum() is not a member function of any class.int sum(myclass x){/* Because sum() is a friend of myclass, it can directly access a and b. */

return x.a + x.b;}int main(){myclass n;n.set_ab(3, 4);cout << sum(n);return 0;

}

30

ARKADAŞ FONKSİYONLAR - 3• Örnek2 (Osborne 1998):

#include <iostream>using namespace std;const int IDLE = 0;const int INUSE = 1;

class C2; // forward declaration

class C1 {int status; // IDLE if off, INUSE if on screen// ...

public:void set_status(int state);friend int idle(C1 a, C2 b);

};

31

ARKADAŞ FONKSİYONLAR - 4class C2 {int status; // IDLE if off, INUSE if on screen// ...

public:void set_status(int state);friend int idle(C1 a, C2 b);

};

void C1::set_status(int state){ status = state; }

void C2::set_status(int state){ status = state; }

32

ARKADAŞ FONKSİYONLAR - 5int idle(C1 a, C2 b){ if(a.status || b.status) return 0;else return 1; }

int main(){ C1 x; C2 y;x.set_status(IDLE);y.set_status(IDLE);if(idle(x, y)) cout << "Screen can be used.\n";else cout << "In use.\n";x.set_status(INUSE);if(idle(x, y)) cout << "Screen can be used.\n";else cout << "In use.\n";return 0; }

33

NESNE DİZİLERİ - 1•Herhangi bir veri tipinde dizi tanımlar gibi

nesne dizileri tanımlayabilirsiniz:Tarih dogumGunleri[10];

•Bir nesne dizisi bildirimi yaptığınızda, yapıcı fonksiyon dizideki her eleman için çağrılır.

•Eğer ilk değer belirtmeden dizilerinizin bildirimini yapmak istiyorsanız varsayılan yapıcı fonksiyona sahip olmanız gerekir. Yukarıdaki örnekte, dizinin her elemanı Ocak 1, 1 değerini ilk değer olarak alır.

34

NESNE DİZİLERİ - 2

•Ayrıca, argüman alan yapıcı fonksiyonunuzu açıkça çağırarak da dizideki her elemana ilk değer atayabilirsiniz. Eğer dizinin bütünü için yeterince ilk değer belirtmezseniz, geriye kalan elemanlar için varsayılan yapıcı fonksiyon çağrılır:Tarih dogumGunleri[10] = {Tarih(2,10,1950),

Tarih(9,16,1960), Tarih(7,31,1953),

Tarih(1,3,1970), Tarih(12,2,1963)};

35

NESNE DİZİLERİ - 3

•Eğer sınıf tek bir argüman alan yapıcı fonksiyona sahipse, ilk değer olarak yalnızca bu argümanı belirtebilirisiniz.

•Ayrıca, farklı ilk değer atama tarzlarını karışık da kullanabilirsiniz:

String mesaj[10]={“Mesajin ilk satiri\n”, “ikinci satiri\n”,

String(“ucuncu satiri\n”), String(‘-’,25), String()};

36