Pragma bir kez - Pragma once

İçinde C ve C ++ Programlama dilleri, bir kez pragma standart değildir ancak geniş çapta desteklenen Önişlemci direktifi akıma neden olmak için tasarlanmış Kaynak dosyası tek bir derlemeye yalnızca bir kez dahil edilecektir.[1] Böylece, #pragma bir kez ile aynı amaca hizmet eder korumaları dahil et, ancak aşağıdakileri içeren çeşitli avantajlara sahiptir: daha az kod, ad çatışmalarından kaçınma ve bazen derleme hızında iyileşme.[2] Diğer taraftan, #pragma bir kez mutlaka tüm derleyicilerde mevcut değildir ve uygulanması zordur ve her zaman güvenilir olmayabilir.

Misal

"Grandparent.h" dosyası
#pragma bir kezyapı foo {    int üye;};
"Parent.h" dosyası
#Dahil etmek "büyükbaba.h"
"Child.c" dosyası
#Dahil etmek "büyükbaba.h"#Dahil etmek "parent.h"

Bu örnekte, aşağıdakilerin dahil edilmesi büyükbaba.h hem de parent.h ve child.c normalde derleme hatasına neden olur, çünkü yapı belirli bir adla, belirli bir derlemede yalnızca bir kez tanımlanabilir. #pragma bir kez direktif, sonraki eklemeleri görmezden gelerek bundan kaçınmaya yarar büyükbaba.h.

Avantajlar

Kullanma #pragma bir kez izin verir C ön işlemcisi gerektiğinde bir başlık dosyası eklemek ve bir #Dahil etmek aksi yönde direktif. Bu, cihazın davranışını değiştirme etkisine sahiptir. C ön işlemcisi ve programcıların dosya bağımlılıklarını basit bir şekilde ifade etmelerine izin vererek manuel yönetim ihtiyacını ortadan kaldırır.

En yaygın alternatif #pragma bir kez kullanmak #tanımlamak ayarlamak için #include guard makrosu, programcı tarafından adı o dosyaya özgü olacak şekilde seçilir. Örneğin,

#ifndef GRANDPARENT_H#define GRANDPARENT_H... içerik nın-nin büyükbaba.h#endif / *! GRANDPARENT_H * /

Bu yaklaşım, içerme dosyasının içeriğinin bir kereden fazla görülmemesini minimum düzeyde sağlar. Bu daha ayrıntılıdır, daha fazla manuel müdahale gerektirir ve aynı makro adının birden fazla dosyada yanlışlıkla kullanılmasını önlemek için derleyicide kullanılabilecek bir mekanizma olmadığından programcı hatasına yatkındır, bu da dosyalardan yalnızca biriyle sonuçlanır dahil ediliyor. Bu tür hataların saptanmadan kalması olası değildir, ancak bir derleyici hata raporunun yorumlanmasını karmaşıklaştırabilir. Ön işlemcinin kendisi işlemden sorumlu olduğundan #pragma bir kezprogramcı isim çatışmalarına neden olan hatalar yapamaz.

Yokluğunda #include guards etrafında #Dahil etmek direktifler, kullanımı #pragma bir kez üst düzey bir mekanizma olduğu için bazı derleyiciler için derleme hızını artıracaktır; derleyicinin kendisi dosya adlarını karşılaştırabilir veya düğümler çağırmak zorunda kalmadan C ön işlemcisi başlığı taramak için #ifndef ve #endif. Yine de, içerme korumaları çok sık göründüğünden ve dosyaları açma ek yükü önemli olduğundan, derleyicilerin dahil korumalarının işlenmesini optimize etmesi yaygındır, bu da onları #pragma bir kez.[3][4][5]

Uyarılar

Bir dosya sisteminde aynı dosyanın belirlenmesi önemsiz bir iş değildir.[6] Sembolik bağlantılar ve özellikle sabit bağlantılar aynı dosyanın farklı dizinlerde farklı isimler altında bulunmasına neden olabilir. Derleyiciler, dosya boyutunu, değişiklik zamanını ve içeriğini karşılaştıran bir buluşsal yöntem kullanabilir.[7] Bunlara ek olarak, #pragma bir kez aynı dosya kasıtlı olarak bir projenin birkaç parçasına kopyalanmışsa, yanlış bir şey yapabilir, ör. yapıyı hazırlarken. Buna karşılık korumaları dahil et yine de çift tanımlardan korurdu, #pragma bir kez bunları derleyiciye bağlı bir şekilde aynı dosya olarak ele alabilir veya etmeyebilir. Bu zorluklar, sabit bağlantıların, ağa bağlı dosya sistemlerinin vb. Varlığında aynı dosyayı neyin oluşturduğunun tanımlanmasına ilişkin zorluklarla birlikte, şimdiye kadar #pragma bir kez.[kaynak belirtilmeli ]

Kullanımı #include guard makrolar, bağımlı kodun semantik veya rakip alternatiflerin arayüzlerindeki küçük farklılıkları tanımasına ve bunlara yanıt vermesine izin verir. Örneğin,

#Dahil etmek TLS_API_MACRO / * komut satırında tanımlı * /...# tanımlıysa TLS_A_H... kullanım bir bilinen API#elif tanımlı TLS_B_H... kullanım bir diğeri bilinen API#Başka#error "tanınmayan TLS API"#endif

Bu durumda, hangi API için kullanılabilir olduğu doğrudan belirleme, dahil etme dosyasının kendi #include guard makro.

#Dahil etmek yönerge, bir programcının yönergenin noktasına bir dosyanın metnini gerçekten dahil etme niyetini temsil etmek için tanımlanır. Bu, tek bir derleme biriminde birkaç kez meydana gelebilir ve makro içeren içerikleri, makronun değişen tanımlarına karşı birden çok kez değerlendirmek için kullanışlıdır.

Kullanımı #pragma bir kez, kullanımı gibi #include guard İçerme dosyasındaki makrolar, istenmeyen çoklu eklemeye karşı koruma sağlamak için sorumluluğu yazarlarına yükler. Doğrudan, korumasız kullanım yoluyla programcılar tarafında her iki mekanizmaya aşırı güvenme #Dahil etmek kendilerine ait olmayan direktifler #include guard her iki mekanizma ile de kendini korumayan bir içerme dosyası kullanıldığında başarısızlığa yol açacaktır.

Taşınabilirlik

Derleyici#pragma bir kez
ClangDestekleniyor[8]
Comeau C / C ++Destekleniyor[9]
Cray C ve C ++Destekleniyor[10] (9.0'dan beri)
C ++ Oluşturucu XE3Destekleniyor[11]
Dijital Mars C ++Destekleniyor[12]
GCCDestekleniyor[13] (resmi olarak 3.4[6][14])
HP C / aC ++Destekleniyor[15] (en az A.06.12'den beri)
IBM XL C / C ++Destekleniyor[16] (13.1.1'den beri)
Intel C ++ DerleyiciDestekleniyor[17][başarısız doğrulama ]
Microsoft Visual C ++Destekleniyor[18][19] (4.2'den beri)
NVIDIA CUDA DerleyiciDesteklenir (temeldeki ana bilgisayar derleyicisine bağlı olarak)
Pelles CDestekleniyor[20]
ARM DS-5Destekleniyor[21]
IAR C / C ++Destekleniyor[22]
Keil CC 5Destekleniyor[23]
Oracle Developer Studio C / C ++Destekleniyor[24] (12.5'ten beri)
Portland Grubu C / C ++Destekleniyor[25] (en az 17.4'ten beri)
TinyCCDestekleniyor[26] (Nisan 2015'ten beri)
TriCore için TASKING VX-araç seti: C derleyicisiDestekleniyor[27] (v6.2r2'den beri)

Referanslar

  1. ^ "bir Zamanlar". Microsoft Docs. 3 Kasım 2016. Alındı 25 Temmuz 2019.
  2. ^ "İçeriden Oyunlar: Dahil Olan Daha Fazla Deney". Web.archive.org. 2005-01-25. Arşivlenen orijinal 30 Eylül 2008. Alındı 2013-08-19.
  3. ^ "C Ön İşlemcisi: 1. C Ön İşlemcisi". Gcc.gnu.org. 1996-02-01. Alındı 2013-08-19.
  4. ^ ""Clang "CFE Internals Manual - Clang 3.4 belgeleri". Clang.llvm.org. Alındı 2013-08-19.
  5. ^ "clang: Dosya işleme rutinleri". Clang.llvm.org. Alındı 2013-08-19.
  6. ^ a b "GCC 3.4 Sürüm Serisi - Değişiklikler, Yeni Özellikler ve Düzeltmeler". Gcc.gnu.org. Alındı 2013-08-19.
  7. ^ "GCC kaynak kodunda should_stack_file () işlevi".
  8. ^ "clang: clang: Pragma.cpp Kaynak Dosyası". Clang.llvm.org. Arşivlenen orijinal 2014-04-04 tarihinde. Alındı 2013-08-19.
  9. ^ "Comeau C ++ Ön Sürüm Kullanıcı Belgeleri: Pragmas". Comeaucomputing.com. Alındı 2013-08-19.[ölü bağlantı ]
  10. ^ "CCE 9.0.0 Sürüme Genel Bakış Giriş S-5212". Cray Inc. 2019-06-01. Alındı 2019-09-23.
  11. ^ "#pragma bir kez - RAD Studio XE3". Docwiki.embarcadero.com. 2010-12-02. Alındı 2013-08-19.
  12. ^ "Pragmalar". Dijital Mars. Alındı 2013-08-19.
  13. ^ "Sarmalayıcıya Alternatifler #ifndef". Gcc.gnu.org. Alındı 2013-08-20.
  14. ^ "GCC Bug 11569 - #pragma'nın yerini bir kez alamazsınız". 2003-07-18. Alındı 2020-10-21.
  15. ^ "HP aC ++ / HP C A.06.29 Programcı Kılavuzu; Mart 2016 (AR1603)".
  16. ^ "Desteklenen GCC pragmaları". IBM. Alındı 2015-02-20.
  17. ^ "Teşhis 1782: #pragma bir kez eski. Onun yerine #ifndef korumasını kullanın". Intel Geliştirici Bölgeleri. Alındı 4 Aralık 2013.[ölü bağlantı ]
  18. ^ "bir kez (C / C ++)". Msdn.microsoft.com. Arşivlenen orijinal 2016-08-10 tarihinde. Alındı 2013-08-19.
  19. ^ https://msdn.microsoft.com/en-us/library/4141z1cx.aspx
  20. ^ IDE yardımı / belgeleri
  21. ^ "ARM Bilgi Merkezi". KOL. Alındı 2013-12-17.
  22. ^ "IAR C / C ++ Geliştirme Kılavuzu" (PDF). IAR Sistemleri. Arşivlenen orijinal (PDF) 16 Mayıs 2017 tarihinde. Alındı 4 Aralık 2013.
  23. ^ "Derleyici tarafından tanınan Pragmalar". Keil.
  24. ^ "Oracle® Developer Studio 12.5: GCC Uyumluluk Kılavuzu". Oracle. Alındı 2016-07-26.
  25. ^ "Portland Grubu". Alındı 31 Temmuz 2016.
  26. ^ "TinyCC pragma bir kez uygulama". Alındı 19 Haziran 2018.
  27. ^ "MA160-800 (v6.2r2) 13 Mart 2018, sayfa 92" (PDF).

Dış bağlantılar