Kısa devre değerlendirmesi - Short-circuit evaluation

Kısa devre değerlendirmesi, minimum değerlendirmeveya McCarthy değerlendirmesi (sonra John McCarthy ) bazılarının anlambilimidir Boole operatörleri bazılarında Programlama dilleri burada ikinci bağımsız değişkenin yalnızca ilk bağımsız değişken ifadenin değerini belirlemek için yeterli olmadığında çalıştırılır veya değerlendirilir: VE işlev değerlendirir yanlışgenel değer olmalıdır yanlış; ve ilk argüman ne zaman VEYA işlev değerlendirir doğrugenel değer olmalıdır doğru.

İle programlama dillerinde tembel değerlendirme (Lisp, Perl, Haskell ), olağan Boole operatörleri kısa devredir. Diğerlerinde (Ada, Java, Delphi ), hem kısa devre hem de standart Boole operatörleri mevcuttur. Bazı Boole işlemleri için özel veya (XOR), kısa devre yapmak mümkün değildir, çünkü sonucu belirlemek için her iki işlenen her zaman gereklidir.

Kısa devre operatörleri, aslında, Kontrol Yapıları basit aritmetik operatörler yerine katı. İçinde zorunlu dil terimler (özellikle C ve C ++ ), yan etkilerin önemli olduğu durumlarda, kısa devre operatörleri sıra noktası - herhangi biri dahil olmak üzere ilk argümanı tamamen değerlendirirler yan etkiler, ikinci bağımsız değişkeni işlemeden önce (isteğe bağlı olarak). ALGOL 68 Kullanılmış prosedür başarmak Kullanıcı tanımlı kısa devre operatörleri ve prosedürleri.

Kısa devre operatörlerinin kullanımı sorunlu olduğu için eleştirildi:

Koşullu bağlayıcılar - "cand" ve "cor"kısaca - ... ilk bakışta göründüklerinden daha az masumdur. Örneğin, cor dağıtmaz cand: karşılaştırmak

(Bir cand B) cor C ile (Bir cor C) cand (B cor C);

¬A ∧ C durumunda, ikinci ifade B'nin tanımlanmasını gerektirir, birincisi tanımlamaz. Koşullu bağlaçlar, programlarla ilgili resmi mantığı karmaşıklaştırdığı için, bunlardan kaçınılması daha iyidir.

Tanım

Kısa devre değerlendirmesini uygulayan herhangi bir programlama dilinde, ifade x ve y eşdeğerdir koşullu ifade Eğer x sonra y Başka xve ifade x veya y eşdeğerdir Eğer x sonra x Başka y. Her iki durumda da, x yalnızca bir kez değerlendirilir.

Yukarıdaki genelleştirilmiş tanım, ikisinden daha fazlasına sahip, gevşek biçimde yazılmış dilleri barındırır. doğruluk değerleri Doğru ve Yanlış, kısa devre operatörleri son değerlendirilen alt ifadeyi döndürebilir. Bu, aşağıdaki tabloda "son değer" olarak adlandırılır. Kesin olarak yazılmış bir dil için, ifade şu şekilde basitleştirilmiştir: Eğer x sonra y Başka yanlış ve Eğer x sonra doğru Başka y sırasıyla boole durumu için.

Öncelik

olmasına rağmen VE alır öncelik bitmiş VEYA Birçok dilde bu, kısa devre değerlendirmesinin evrensel bir özelliği değildir. Aynı önceliği alan iki operatörün bir örneği sol çağrışımlı birbirimizle POSIX kabuğu komut listesi sözdizimi.[2](§2.9.3)

Aşağıdaki basit soldan sağa değerlendirici, bir önceliğe zorlar VE bitmiş VEYA tarafından devam et:

işlevi kısa devre değerlendirme (operatörler, değerler)    İzin Vermek sonuç : = Doğru her biri için (op, val) içinde (operatörler, değerler):        Eğer op = "VE" && sonuç = Yanlış devam et        Aksi takdirde op = "VEYA" && sonuç = Doğru dönüş sonuç        Başka            sonuç := val    dönüş sonuç

Resmileştirme

Yan etkilere sahip olan veya olmayan kısa devre mantığı, Hoare koşullu. Bunun bir sonucu, kısa devre yapmayan operatörlerin aynı değerlendirme sırasına sahip olmak için kısa devre mantığından tanımlanabilmesidir.[3]

Yaygın programlama ve komut dosyası dillerinde destek

Çeşitli dillerde Boole operatörleri
Dilİstekli operatörlerKısa devre operatörleriSonuç türü
Gelişmiş İş Uygulama Programlama (ABAP )Yokve, veyaBoole1
Adave, veyave daha sonra, ya da başkaBoole
ALGOL 68ve, &, ∧; veya, ∨andf, orf (her ikisi de kullanıcı tanımlı)Boole
APL, , (nand), (ne) vb.:Ve eğer, :Ya da eğerBoole1
awkYok&&, ||Boole
BashYok&&, ||Boole
C, Amaç-CYok&&, ||, ?[4]int (&&,||), opnd-bağımlı (?)
C ++2Yok&&, ||, ?[5]Boole (&&,||), opnd-bağımlı (?)
C #&, |&&, ||, ?, ??Boole (&&,||), opnd-bağımlı (?, ??)
ColdFusion İşaretleme Dili (CFML)YokVE, VEYA, &&, ||Boole
D3&, |&&, ||, ?Boole (&&,||), opnd-bağımlı (?)
Eyfelve, veyave daha sonra, ya da başkaBoole
Erlangve, veyave ayrıca, OrelseBoole
Fortran4.ve., .veya..ve., .veya.Boole
Git, Haskell, OCamlYok&&, ||Boole
Java, MATLAB, R, Swift&, |&&, ||Boole
JavaScript, Julia&, |&&, ||Son değer
KementYokve, veya, &&, ||Son değer
Kotlinve, veya&&, ||Boole
Lisp, Lua, ŞemaYokve, veyaSon değer
KABAKULAK (M)&, !YokSayısal
Modula-2YokVE, VEYABoole
OberonYok&, VEYABoole
OCamlYok&&, ||Boole
Pascalve, veya5,9ve daha sonra, or_else6,9Boole
Perl&, |&&, ve, ||, veyaSon değer
Yakutve, veya&&, ||Son değer
PHP&, |&&, ve, ||, veyaBoole
POSIX kabuğu (komut listesi)Yok&&, ||Son değer (çıkış)
PythonYok[6]ve, veyaSon değer
Pas, paslanma&, |&&, ||[7]Boole
Smalltalk&, |ve:, veya:7Boole
Standart MLBilinmeyenve ayrıca, OrelseBoole
TTCN-3Yokve, veya[8]Boole
Visual Basic .NETVe, VeyaVe ayrıca, OrElseBoole
Visual Basic, Uygulamalar için Visual Basic (VBA)Ve, VeyaVakayı Seçin8Sayısal
Wolfram DiliVe @@ {...}, Veya @@ {...}Ve, Veya, &&, ||Boole
ZTT&, |YokBoole

1 ABAP ve APL'nin farklı bir boole türü yoktur.
2 Operatörler aşırı yüklendiğinde && ve || hevesliler ve her türden geri dönebilirler.
3 Bu yalnızca çalışma zamanında değerlendirilen ifadeler için geçerlidir, statik eğer ve statik iddia. Statik başlatıcılardaki veya bildirim sabitlerindeki ifadeler istekli değerlendirmeyi kullanır.
4 Fortran operatörleri ne kısa devre yapar ne de istekli: dil spesifikasyonu, derleyicinin optimizasyon yöntemini seçmesine izin verir.
5 ISO / IEC 10206: 1990 Genişletilmiş Pascal kısa devreye izin verir, ancak bunu gerektirmez.
6 ISO / IEC 10206: 1990 Extended Pascal destekler ve daha sonra ve or_else.[9]
7 Smalltalk, bağımsız değişken olduğu sürece kısa devre semantiğini kullanır. ve: bir bloktur (ör. false ve: [Transkript gösterisi: "Beni görmeyeceğim"]).
8 TEMEL CASE deyimlerini destekleyen diller bunu, sabit etiketlerle sınırlı atlama tabloları yerine koşullu değerlendirme sistemini kullanarak yaptı.
9 Delphi ve Ücretsiz Pascal kısa devre değerlendirmesine varsayılan. Bu, derleyici seçenekleri tarafından değiştirilebilir, ancak yaygın olarak kullanılmıyor gibi görünüyor.

Genel kullanım

İkinci argümanın istenmeyen yan etkilerinden kaçınmak

Olağan örnek, bir C tabanlı dil:

int mezhep = 0;Eğer (mezhep != 0 && num / mezhep){    ... // num / denom hesaplamanın hiçbir zaman sıfıra bölme hatasıyla sonuçlanmamasını sağlar }

Aşağıdaki örneği düşünün:

int a = 0;Eğer (a != 0 && benim fonksiyonum(b)){    bir şey yap();}

Bu örnekte, kısa devre değerlendirmesi şunları garanti eder: işlevim (b) asla aranmaz. Bunun nedeni ise a! = 0 değerlendirir yanlış. Bu özellik, iki kullanışlı programlama yapısına izin verir.

  1. İlk alt ifade, pahalı bir hesaplamaya gerek olup olmadığını kontrol ederse ve kontrol, yanlışikinci argümanda pahalı hesaplama ortadan kaldırılabilir.
  2. İlk ifadenin bir koşulu garanti ettiği bir yapıya izin verir; çalışma hatası.

Her ikisi de, minimum değerlendirmenin hem boş işaretçi çözümlemesini hem de fazla bellek getirmelerini engellediği aşağıdaki C snippet'inde gösterilmektedir:

bool is_first_char_valid_alpha_unsafe(sabit kömür *p){    dönüş isalpha(p[0]); // SEGFAULT, p == NULL ile son derece mümkün}bool is_first_char_valid_alpha(sabit kömür *p){    dönüş p != BOŞ && isalpha(p[0]); // 1) p == NULL ile gereksiz isalpha () yürütme yok, 2) SEGFAULT riski yok}

İdiomatik koşullu yapı

Minimum değerlendirme bir operatörün anlamsal tanımının bir parçası olduğundan ve (isteğe bağlı) bir optimizasyon olmadığından, birçok kodlama modeli[hangi? ] kısa ve öz (eğer deyimsel ise) koşullu bir yapı olarak buna güvenmeye başladılar. Örnekler şunları içerir:

Perl deyimler:

bazı_şartlar veya ölmek;    # Some_condition yanlışsa yürütmeyi iptal edinbazı_şartlar ve ölmek;   # Some_condition doğruysa yürütmeyi iptal edin

POSIX kabuğu deyimler:[10]

modprobe -q bazı_modüller && Eko "bazı_modüller yüklendi" || Eko "bazı_modüller yüklü değil"

Bu deyim varsaymaktadır ki Eko başarısız olamaz.

Olası sorunlar

Test edilmemiş ikinci durum, yerine getirilmemiş yan etkilere yol açar

Bu faydalara rağmen, minimum değerlendirme, bunun gerçekleştiğini anlamayan (veya unutmayan) programcılar için sorunlara neden olabilir. Örneğin, kodda

Eğer (ifadeA && işlevim(b)) {    bir şey yap();}

Eğer işlevim (b) ne olursa olsun bazı gerekli işlemleri yapması gerekiyor bir şey yap() sistem kaynaklarını ayırma gibi yürütülür ve ifadeA yanlış olarak değerlendirilirse işlevim (b) çalıştırılmayacak, bu da sorunlara neden olabilir. Gibi bazı programlama dilleri Java, bu sorunu önlemek için biri minimum değerlendirme kullanan diğeri kullanmayan iki operatöre sahip olun.

Gerçekleştirilmemiş yan etki ifadeleriyle ilgili problemler, uygun programlama stiliyle, yani, değerlendirmelerde yan etkilere sahip değerlerin kullanılması genellikle kodu opak ve hataya açık hale getirme eğiliminde olduğundan, boole ifadelerinde yan etkilerin kullanılmaması ile kolayca çözülebilir.[11]

Optimizasyonların kısıtlanması nedeniyle düşük verimlilik

Kısa devre, aşağıdaki durumlarda hatalara neden olabilir: şube tahmini modernde merkezi işlem birimleri (CPU'lar) ve performansı önemli ölçüde azaltır. Dikkate değer bir örnek, eksen hizalı kutu kesişim koduna sahip son derece optimize edilmiş ışındır. Işın izleme.[açıklama gerekli ] Bazı derleyiciler bu tür durumları algılayabilir ve daha hızlı kod yayabilir, ancak programlama dili anlambilim bu tür optimizasyonları kısıtlayabilir.[kaynak belirtilmeli ]

Böyle bir durum için optimize edemeyen bir derleyici örneği Java 2012 itibariyle Hotspot VM.[12]

Ayrıca bakınız

Referanslar

  1. ^ Edsger W. Dijkstra "Biraz hayal kırıklığı yaratan bir yazışmada", EWD1009-0, 25 Mayıs 1987 tam metin
  2. ^ "Kabuk Komut Dili". pubs.opengroup.org.
  3. ^ Jan A. Bergstra, A. Ponse, D.J.C. Staudt (2010). "Kısa devre mantığı". arXiv:1010.3674 [cs.LO ].CS1 Maint: yazar parametresini kullanır (bağlantı)
  4. ^ ISO / IEC 9899 standardı, bölüm 6.5.13
  5. ^ ISO / IEC IS 14882 taslağı.
  6. ^ https://wiki.python.org/moin/BitwiseOperators
  7. ^ "std :: ops - Pas". doc.rust-lang.org. Alındı 2019-02-12.
  8. ^ ETSI ES 201 873-1 V4.10.1, bölüm 7.1.4
  9. ^ "and_then - GNU Pascal Kılavuzu". Gnu-pascal.de. Alındı 2013-08-24.
  10. ^ "|| bash'da ne anlama geliyor?". stackexchange.com. Alındı 2019-01-09.
  11. ^ "Referans Şeffaflık, Kesinlik ve Katlanamazlık" (PDF). Itu.dk. Alındı 2013-08-24.
  12. ^ Wasserman, Louis. "java - Koşulsuz VE (&& yerine &) kullanmanın daha iyi olduğu durumlar nelerdir?". Yığın Taşması.