Typedef - Typedef

typedef bir ayrılmış anahtar kelime içinde Programlama dilleri C ve C ++. Ek bir ad oluşturmak için kullanılır (takma ad) bir diğeri için veri tipi, ancak yeni bir tür oluşturmaz[1]belirsiz bir durum dışında nitelikli typedef niteleyicilerinin dizi öğesi türüne aktarıldığı bir dizi türünün typedef'i[2]. Bu nedenle, genellikle karmaşık bildirmenin sözdizimini basitleştirmek için kullanılır. veri yapıları oluşan yapı ve sendika türleri, ancak belirli tanımlayıcı tür adlarının sağlanmasında olduğu kadar yaygındır. tamsayı veri türleri değişen uzunluklarda.

Sözdizimi

Typedef bildiriminin sözdizimi şöyledir:[3]

typedef tip beyanı;

Yeni tür diğer adının adı, diğer herhangi bir C tanımlayıcısını bildirmekle aynı sözdizimini izler, bu nedenle daha ayrıntılı biçimde:

typedef tür tanımı tanımlayıcı

İçinde C standart kitaplığı ve POSIX belirtimler typedef tanımının tanımlayıcısı genellikle _tolduğu gibi size_t ve time_t. Bu, diğer kodlama sistemlerinde uygulanmaktadır, ancak POSIX bu uygulamayı açıkça POSIX veri tipleri.

Örnekler

 typedef int uzunluk;

Bu, türü yaratır uzunluk türün eş anlamlısı olarak int.

Dokümantasyon kullanımı

Bir typedef bildirimi, programlama bağlamında bir değişkenin anlamını göstererek dokümantasyon olarak kullanılabilir, örneğin, bir ölçü birimi veya sayımların ifadesini içerebilir. Jenerik beyanlar,

int Geçerli hız;int yüksek skor;geçersiz tebrik etmek(int puanın) {    Eğer (puanın > yüksek skor) {        // ...    }}

bağlama özgü türler bildirilerek ifade edilebilir:

typedef int km_per_hour;typedef int puan;// "km_per_saat" burada "int" ile eş anlamlıdır ve bu nedenle derleyici davranır// yeni değişkenlerimiz tamsayı olarak.km_per_hour Geçerli hız;puan yüksek skor;geçersiz tebrik etmek(puan puanın) {    Eğer (puanın > yüksek skor) {        // ...    }}

Kodun her iki bölümü de aynı şekilde yürütülür. Bununla birlikte, ikinci kod bloğunda typedef bildirimlerinin kullanılması, iki değişkenin aynı veri türünü temsil ederken int, farklı veya uyumsuz verileri depolayın. Tanımı tebrik et () nın-nin puanın programcıya şunu belirtir: Geçerli hız (veya bir puan) bir argüman olarak aktarılmamalıdır. Her ikisi de değişkenler olarak bildirilseydi, bu o kadar belirgin olmazdı. int veri tipi. Ancak gösterge şudur: sadece programcı için; C / C ++ derleyicisi her iki değişkenin de türde olduğunu düşünür int ve "yanlış" bağımsız değişken türleri için tür uyuşmazlığı uyarılarını veya hatalarını işaretlemez tebrik edin (puanınızı puanlar) aşağıdaki kod parçacığında:

geçersiz foo() {    km_per_hour km100 = 100;    tebrik etmek(km100);}

Tip basitleştirme

Typedef, bir bileşik türün bildirimini basitleştirmek için kullanılabilir (yapı, Birlik ) veya Işaretçi yazın.[4] Örneğin,

yapı MyStruct {    int veri1;    kömür veri2;};

Bu veri türünü tanımlar struct MyStruct. C'de bu türden bir değişken bildirimi ayrıca anahtar sözcüğünü gerektirir yapı, ancak C ++ 'da ihmal edilebilir:

 struct MyStruct a;

Typedef bildirimi, belirtme gerekliliğini ortadan kaldırır yapı C. Örneğin, beyan

typedef struct MyStruct yeni türü;

indirgenir:

yeni tür a;


Yapı bildirimi ve typedef ayrıca tek bir ifadede birleştirilebilir:

typedef yapı MyStruct {    int veri1;    kömür veri2;} yeni tip;

Veya aşağıdaki şekilde kullanılabilir:

typedef yapı {    int veri1;    kömür veri2;} yeni tip;

İçinde C ++, C'nin tersine, anahtar kelimeler yapı, sınıf, ve Sıralama başka bir tanımlayıcı için belirsizlik olmadığı sürece tanımlardan ayrı değişken bildirimlerinde isteğe bağlıdır:

yapı MyStruct x;MyStruct y;

Gibi, MyStruct her yerde kullanılabilir yeni tip kullanılabilir. Ancak bunun tersi doğru değildir; örneğin, yapıcı yöntemleri MyStruct isimlendirilemez yeni tip.

Kötü şöhretli bir örnek C ++ ihtiyacı var yapı anahtar kelime POSIX stat sistem çağrısı argümanlarında aynı isimde bir yapı kullanan:

int stat(sabit kömür *dosya adı, yapı stat *buf){    // ...}

Burada ikisi de C Hem de C ++ ihtiyaç duymak yapı parametre tanımında anahtar kelime.

İşaretçiler

Typedef, yeni bir işaretçi tipini tanımlamak için kullanılabilir.

typedef int *intptr;intptr ptr;// İle aynı:// int * ptr;

intptr işaretçi tipine sahip yeni bir takma addır int *. Tanım, intptr ptr;, bir değişken tanımlar ptr tip ile int *. Yani, ptr türdeki bir değişkene işaret edebilen bir göstericidir int.

Yeni bir işaretçi türü tanımlamak için typedef kullanmak bazen kafa karışıklığına neden olabilir. Örneğin:

typedef int *intptr;// Hem 'uçurum' hem de 'allen' int * türündedir.intptr uçurum, Allen;// 'cliff2' int * türündedir, ancak 'allen2' int ** türündedir.intptr uçurum2, *Allen2;// İle aynı:// intptr cliff2;// intptr * allen2;

Yukarıda intptr uçurum, allen; 2 değişkeni tanımlamak anlamına gelir int * her ikisi için yazın. Bunun nedeni typedef tarafından tanımlanan bir türün genişletme değil, tür olmasıdır. Diğer bir deyişle, intptr, hangisi int * tip, ikisini de uçurum ve Allen. İçin intptr cliff2, * allen2;, intptr yazı süslüyor uçurum2 ve * allen2. Yani, intptr cliff2, * allen2; 2 ayrı tanıma eşdeğerdir, intptr cliff2; ve intptr * allen2. intptr * allen2 anlamına gelir Allen2 ile bir anıyı gösteren bir işaretçidir int * yazın. Kısaca Allen2 türü var, int **.

Yapılar ve yapı işaretçileri

Yazı tipleri ayrıca tanımları veya bildirimleri basitleştirebilir. yapı Işaretçi türleri. Bunu düşün:

yapı Düğüm {    int veri;    yapı Düğüm *nextptr;};

Typedef kullanılarak yukarıdaki kod şu şekilde yeniden yazılabilir:

typedef yapı Düğüm Düğüm;yapı Düğüm {    int veri;    Düğüm *nextptr;};

C'de, tek bir ifadede aynı türden birden çok değişken bildirilebilir, hatta yapıyı işaretçi veya işaretçi olmayanlarla karıştırabilir. Bununla birlikte, bir işaretçi olarak belirlemek için her değişkene bir yıldız işareti koymak gerekir. Aşağıda, bir programcı şunu varsayabilir: errptr gerçekten bir Düğüm *, ancak bir yazım hatası, errptr bir Düğüm. Bu, ince sözdizimi hatalarına yol açabilir.

yapı Düğüm *startptr, *endptr, *curptr, *prevptr, errptr, *refptr;

Typedef'i tanımlayarak Düğüm *, tüm değişkenlerin yapı işaretçi türleri olduğu veya her değişkenin bir işaretçi türü bir yapı tipi.

typedef yapı Düğüm* NodePtr;NodePtr startptr, endptr, curptr, prevptr, errptr, refptr;

İşlev işaretçileri

int Matematik yapmak(yüzer arg1, int arg2) {    dönüş arg2;}int call_a_func(int (*call_this)(yüzer, int)) {    int çıktı = call_this(5.5, 7);    dönüş çıktı;}int son sonuç = call_a_func(&Matematik yapmak);

Önceki kod typedef özellikleriyle yeniden yazılabilir:

typedef int (*MathFunc)(yüzer, int);int Matematik yapmak(yüzer arg1, int arg2) {    dönüş arg2;}int call_a_func(MathFunc call_this) {    int çıktı = call_this(5.5, 7);    dönüş çıktı;}int son sonuç = call_a_func(&Matematik yapmak);

Buraya, MathFunc tür için yeni takma addır. Bir MathFunc bir tamsayı döndüren ve argüman olarak bir float ve ardından bir tamsayı alan bir işleve göstericidir.

Bir işlev bir işlev işaretçisi döndürdüğünde, typedef olmadan daha da kafa karıştırıcı olabilir. Aşağıdaki fonksiyon prototipidir sinyal (3) itibaren FreeBSD:

geçersiz (*sinyal(int sig, geçersiz (*işlev)(int)))(int);

Yukarıdaki işlev bildirimi, işlevin bağımsız değişken olarak neyi kabul ettiğini ya da döndürdüğü türü açıkça göstermediğinden şifreli niteliktedir. Acemi bir programcı, işlevin tek bir işlevi kabul ettiğini bile varsayabilir. int argüman olarak ve hiçbir şey döndürmez, ancak gerçekte bir işlev işaretçisine ihtiyaç duyar ve başka bir işlev işaretçisi döndürür. Daha temiz yazılabilir:

typedef geçersiz (*sighandler_t)(int);sighandler_t sinyal(int sig, sighandler_t işlev);

Diziler

Bir typedef, dizi türlerinin tanımını basitleştirmek için de kullanılabilir. Örneğin,

typedef kömür arrType[6];arrType arr = {1, 2, 3, 4, 5, 6};arrType *pArr;// İle aynı:// karakter dizisi [6] = {1, 2, 3, 4, 5, 6};// karakter (* pArr) [6];

Buraya, arrType için yeni takma addır karakter [6] type, 6 öğeli bir dizi türüdür. İçin arrType * pArr;, pArr hafızasını gösteren bir işaretçidir karakter [6] yazın.

Tip dökümler

Type kullanılarak bir typedef oluşturulur tanım sözdizimi, ancak tür kullanılarak oluşturulmuş gibi kullanılabilir oyuncular sözdizimi. (Tip döküm bir veri türünü değiştirir.) Örneğin, ilk satırdan sonraki her satırda:

// "funcptr", "double" alan ve "int" döndüren bir işleve göstericidir.typedef int (*funcptr)(çift);// C veya C ++ 'da geçerlidir.funcptr x = (funcptr) BOŞ;// Yalnızca C ++ ile geçerlidir.funcptr y = funcptr(BOŞ);funcptr z = static_cast<funcptr>(BOŞ);

funcptr sol tarafta bir değişken bildirmek için ve sağ tarafta bir değer atamak için kullanılır. Bu nedenle typedef, tanım sözdizimini tür cast sözdizimine nasıl dönüştüreceğini anlamak istemeyen programcılar tarafından kullanılabilir.

Typedef olmadan, genellikle tanım sözdizimini kullanmak ve sözdizimini dönüşümlü olarak kullanmak mümkün değildir. Örneğin:

geçersiz *p = BOŞ;// Bu yasaldır.int (*x)(çift) = (int (*)(çift)) p;// Sol taraf yasal değil.int (*)(çift) y = (int (*)(çift)) p;// Sağ taraf yasal değil.int (*z)(çift) = (int (*p)(çift));

C ++ 'da kullanım

C ++ 'da tür adları karmaşık olabilir ve typedef, türe basit bir ad atamak için bir mekanizma sağlar.

std::vektör<std::çift<std::dizi, int>> değerler;için (std::vektör<std::çift<std::dizi, int>>::const_iterator ben = değerler.başla(); ben != değerler.son(); ++ben){    std::çift<std::dizi, int> sabit & t = *ben;    // ...}

ve

typedef std::çift<std::dizi, int> değer_t;typedef std::vektör<değer_t> değerler_t;değerler_t değerler;için (değerler_t::const_iterator ben = değerler.başla(); ben != değerler.son(); ++ben){    değer_t sabit & t = *ben;    // ...}

C ++ 11 yazı tiplerini ifade etme olanağını sundu kullanma onun yerine typedef. Örneğin, yukarıdaki iki typedef aynı şekilde şöyle yazılabilir:

kullanma değer_t = std::çift<std::dizi, int>;kullanma değerler_t = std::vektör<değer_t>;

Şablonlarla kullanın

C ++ 03 sağlamaz şablonlu typedef'ler. Örneğin, sahip olmak stringpair temsil etmek std :: pair her tür için T bir olumsuz kullanım:

şablon<typename T>typedef std::çift<std::dizi, T> ip çifti<T>; // çalışmıyor

Ancak, kabul etmek isteyen varsa stringpair :: type yerine stringpair aksi takdirde kullanılmayan şablonlu sınıf veya yapı içindeki typedef aracılığıyla istenen sonucu elde etmek mümkündür:

şablon<typename T>sınıf ip çifti{özel:    // `stringpair  '' nin somutlaştırılmasını önle.    ip çifti();halka açık:    // `stringpair  :: type`,` std :: pair  `şeklinde olsun.    typedef std::çift<std::dizi, T> tip;};// "std :: pair " türünde bir değişken bildirin.ip çifti<int>::tip my_pair_of_string_and_int;

İçinde C ++ 11 şablonlu tipler, aşağıdaki sözdizimiyle eklenir ve kullanma yerine anahtar kelime typedef anahtar kelime. (Görmek şablon takma adları.)[5]

şablon <typename T>kullanma ip çifti = std::çift<std::dizi, T>;// "std :: pair " türünde bir değişken bildirin.ip çifti<int> my_pair_of_string_and_int;

Diğer diller

İçinde SystemVerilog typedef, tam olarak C ve C ++ 'da olduğu gibi davranır.[6]

Statik olarak yazılmış birçok işlevsel dilde, örneğin Haskell, Miranda, OCaml vb. tanımlanabilir eş anlamlıları yazın, C'deki typedef'ler ile aynıdır. Haskell'de bir örnek:

tip PairOfInts = (Int, Int)

Bu örnek bir tür eşanlamlısı tanımladı PairOfInts bir tamsayı türü olarak.

İçinde Tohum7 sabit bir türün tanımı, bir tür için eşanlamlılık sağlamak için kullanılır:

const türü: myVector, dizi tamsayıdır;

İçinde Swift, biri kullanır tip takma ad typedef oluşturmak için anahtar kelime:

typealias PairOfInts = (Int, Int)

C # typedef veya the typedef'e benzer bir özellik içerir. kullanma C ++ sözdizimi.[7][5]

kullanma yeni tip = küresel::Sistemi.Çalışma süresi.Birlikte çalışma.Mareşal;kullanma otherType = Numaralandırmalar.MyEnumType;kullanma StringListMap = Sistemi.Koleksiyonlar.Genel.Sözlük<dizi, Sistemi.Koleksiyonlar.Genel.Liste<dizi>>;

İçinde D anahtar kelime takma ad[8] tür veya kısmi tür eş anlamlıları oluşturmaya izin verir.

yapı Foo(T){}takma ad FooInt = Foo!int;takma ad Eğlence = int temsilci(int);

Kullanım endişeleri

Kernighan ve Ritchie typedef kullanmak için iki neden belirtmişlerdir.[1] Birincisi, bir programı daha taşınabilir veya bakımı daha kolay hale getirmek için bir yol sağlar. Programın kaynak dosyalarındaki her görünümde bir türü değiştirmek zorunda kalmak yerine, yalnızca tek bir typedef ifadesinin değiştirilmesi gerekir. size_t ve ptrdiff_t içinde bu tür typedef isimleri vardır. İkincisi, typedef, karmaşık bir tanımı veya bildirimi anlaşılmasını kolaylaştırabilir.

Bazı programcılar yazı tiplerinin yoğun kullanımına karşıdır. Çoğu argüman, typedef'lerin bir değişkenin gerçek veri türünü basitçe gizlediği fikrine odaklanır. Örneğin, Greg Kroah-Hartman, bir Linux çekirdeği hacker ve documenter, işlev prototip bildirimleri dışında herhangi bir şey için kullanılmalarını engelliyor. Bu uygulamanın sadece gereksiz yere kodu gizlemediğini, aynı zamanda programcıların yanlışlıkla büyük yapıları basit tipler olarak düşünerek yanlış kullanmalarına neden olabileceğini savunuyor.[9]

Ayrıca bakınız

Referanslar

  1. ^ a b Kernighan, Brain W.; Ritchie, Dennis M. (1988). C Programlama Dili (2. baskı). Englewood Kayalıkları, New Jersey.: Prentice Hall. s.147. ISBN  0-13-110362-8. Alındı 18 Haziran 2016. C, yeni veri türü adları oluşturmak için typedef adlı bir olanak sağlar. … Typedef bildiriminin hiçbir şekilde yeni bir tür yaratmadığı vurgulanmalıdır; yalnızca bazı mevcut türler için yeni bir ad ekler.
  2. ^ "const türü niteleyici". cppreference.com. Alındı 2020-10-20.
  3. ^ "typedef belirteci". cppreference.com. Alındı 18 Haziran 2016.
  4. ^ Deitel, Paul J .; Deitel, H.M. (2007). C nasıl programlanır (5. baskı). Upper Saddle River, NJ: Pearson Prentice Hall. ISBN  9780132404167. Alındı 12 Eylül 2012. Yapı türleri için isimler genellikle şu şekilde tanımlanır: typedef daha kısa tip isimler oluşturmak için.
  5. ^ a b "Tür takma adı, diğer ad şablonu (C ++ 11'den beri) - cppreference.com". en.cppreference.com. Alındı 2018-09-25.
  6. ^ Tala, Deepak Kumar. "SystemVerilog Veri Tipleri Part-V". www.asic-world.com. ASIC Dünya. Alındı 25 Eylül 2018.
  7. ^ http://msdn.microsoft.com/en-us/library/aa664765(VS.71).aspx
  8. ^ "Beyanlar - D Programlama Dili". dlang.org. Alındı 2017-05-28.
  9. ^ Kroah-Hartman, Greg (2002-07-01). "Uygun Linux Kernel Kodlama Stili". Linux Journal. Alındı 2007-09-23. Typedef kullanmak, değişkenin yalnızca gerçek türünü gizler.