Decltype - Decltype

V C ++ programovací jazyk, decltype je klíčové slovo slouží k dotazu na typ z výraz. Představeno v C ++ 11, jeho primární zamýšlené použití je v generické programování, kde je často obtížné nebo dokonce nemožné vyjádřit typy, na kterých závisí šablona parametry.

Tak jako generické programování Během 90. let byly techniky stále populárnější, byla uznána potřeba mechanismu dedukce typu. Mnoho dodavatelů překladačů implementovalo své vlastní verze operátoru, obvykle nazývané Typa byly vyvinuty některé přenosné implementace s omezenými funkcemi založené na stávajících jazykových funkcích. V roce 2002 Bjarne Stroustrup navrhl, aby byla do jazyka C ++ přidána standardizovaná verze operátoru, a navrhla název „decltype“, aby odrážela, že operátor získá „deklarovaný typ“ výrazu.

decltypeSémantika byla navržena tak, aby vyhovovala spisovatelům obecných knihoven i začínajícím programátorům. Obecně se odvozený typ shoduje s typem objektu nebo funkce přesně tak, jak je deklarováno ve zdrojovém kódu. Jako velikost[1] operátor, decltypeoperand není vyhodnocen.

Motivace

Se zavedením šablony do programovacího jazyka C ++ a nástup generické programování techniky propagované Standardní knihovna šablon potřeba mechanismu pro získání typu výraz, běžně označované jako Typ, byl uznán. V obecném programování je často obtížné nebo nemožné vyjádřit typy, které závisí na parametrech šablony,[2][3] zejména návratový typ instancí šablony funkcí.[2]

Mnoho prodejců poskytuje Typ operátor jako rozšíření kompilátoru.[4] Již v roce 1997, předtím, než byl C ++ plně standardizován, navrhl Brian Parker přenosné řešení založené na velikost operátor.[4] Jeho práce byla rozšířena o Bill Gibbons, který dospěl k závěru, že tato technika měla několik omezení a byla obecně méně silná než skutečná Typ mechanismus.[4] V článku z října 2000 Dr. Dobb's Journal, Andrei Alexandrescu poznamenal, že „mít typ písma by značně usnadnilo psaní a pochopení kódu šablony“.[5] Poznamenal také, že „typ a velikost sdílejí stejný back-end, protože sizeof stejně musí vypočítat typ.“[5] Andrew Koenig a Barbara E. Moo také uznala užitečnost vestavěného Typ zařízení s upozorněním, že „jeho používání často vyvolává jemné programovací chyby a existují problémy, které nedokáže vyřešit.“[6] Charakterizovali použití konvencí typu, jako je typedefs poskytuje Standardní knihovna šablon, jako silnější a obecnější technika.[6] Nicméně, Steve Dewhurst tvrdil, že takové konvence jsou „nákladné na návrh a zveřejnění“ a že by bylo „mnohem snazší ... jednoduše extrahovat typ výrazu“.[7] V článku z roku 2011 o C ++ 0x „Koenig a Moo předpovídali, že„ typ decltype bude široce používán k usnadnění psaní každodenních programů. “[8]

V roce 2002 Bjarne Stroustrup navrhl rozšíření jazyka C ++ o mechanismy pro dotazování na typ výrazu a inicializaci objektů bez zadání typu.[2] Stroustrup poznamenal, že sémantika klesajícího odkazu, kterou nabízí Typ provozovatel poskytovaný GCC a EDG překladače mohou být problematické.[2] Naopak operátor vracející referenční typ založený na lhodnota - výraznost výrazu byla považována za příliš matoucí. Původní návrh výboru pro standardy C ++ nastínil kombinaci obou variant; operátor by vrátil referenční typ pouze v případě, že deklarovaný typ výrazu zahrnoval odkaz. Aby se zdůraznilo, že odvozený typ bude odrážet „deklarovaný typ“ výrazu, bylo navrženo pojmenování operátoru decltype.[2]

Jednou z uvedených hlavních motivací pro decltype návrhem byla schopnost psát perfektně funkce přesměrování šablony.[9] Někdy je žádoucí napsat obecnou funkci předávání, která vrací stejný typ jako zabalená funkce, bez ohledu na typ, se kterým je vytvořena instance. Bez decltype, to není obecně možné dosáhnout.[9] Příklad, který také využívá koncový-návratový typ:[9]

int& foo(int& i);plovák foo(plovák& F);šablona <třída T> auto transparentní předávající(T& t) > decltype(foo(t)) {  vrátit se foo(t);}

decltype je zde zásadní, protože zachovává informace o tom, zda zabalená funkce vrací referenční typ.[10]

Sémantika

Podobně jako velikost operátor, operand decltype je nevyhodnocen.[11] Neformálně typ vrácený uživatelem decltype (e) se odvodí takto:[2]

  1. Pokud výraz E odkazuje na proměnnou v lokálním rozsahu nebo oboru názvů, statickou členskou proměnnou nebo parametr funkce, pak je výsledkem proměnná nebo parametr deklarovaný typ
  2. Jinak, pokud E je lhodnota, decltype (e) je T &, kde T je typ e; pokud e je xvalue, výsledek je T &&; jinak e je a prvočíslo a výsledek je T.

Tato sémantika byla navržena tak, aby splňovala potřeby spisovatelů obecných knihoven a zároveň byla pro začínající programátory intuitivní, protože návratový typ decltype vždy odpovídá typu objektu nebo funkce přesně tak, jak je deklarováno ve zdrojovém kódu.[2] Formálně se pravidlo 1 vztahuje na neparentézované id-výrazsa výrazy přístupu členů třídy a.[12][13] Příklad:[12]Poznámka pro přidané řádky pro bar (). Pod typem odvozeným pro „bar ()“ je prostý int, ne const int, protože první hodnoty ne-třídních typů mají vždy typy bez cv bez ohledu na staticky deklarovaný odlišný typ.

konst int&& foo();konst int bar();int i;struktur A { dvojnásobek X; };konst A* A = Nový A();decltype(foo()) x1; // typ je const int &&decltype(bar()) x2; // typ je intdecltype(i) x3; // typ je intdecltype(A->X) x4; // typ je dvojitýdecltype((A->X)) x5; // typ je const double &

Důvod rozdílu mezi posledně uvedenými dvěma vyvoláními decltype je, že výraz v závorkách (a-> x) není ani id-výraz ani výraz přístupu člena, a proto neoznačuje pojmenovaný objekt.[14] Protože výraz je lvalue, jeho odvozeným typem je „odkaz na typ výrazu“, nebo const double &.[11]

V prosinci 2008 vznesl Jaakko Järvi vůči výboru znepokojení nad nemožností použití decltype vytvořit a kvalifikované id,[1] což je v rozporu se záměrem decltype (e) by se mělo zacházet "jako by to bylo a typedef-name".[15] Při komentování formálního návrhu výboru pro C ++ 0x, Japonci ISO členské tělo poznamenalo, že „operátor oboru (: :) nelze použít na decltype, ale měl by být. Bylo by užitečné v případě získat typ člena (vnořený typ) z instance následovně“:[16]

vektor<int> proti;decltype(proti)::typ hodnoty i = 0; // int i = 0;

Toto a podobné problémy týkající se formulace, která brání použití decltype v prohlášení a odvozená třída a v destruktor David Vandevoorde a hlasovali v pracovním dokumentu v březnu 2010.[17][18]

Dostupnost

decltype je od roku součástí jazykového standardu C ++ C ++ 11.[12] Poskytuje jej řada překladačů jako rozšíření. Microsoft je Visual C ++ 2010 a novější překladače poskytují a decltype specifikátor typu, který úzce napodobuje sémantiku, jak je popsáno v návrhu výboru pro normy. Může být použit s oběma podařilo se a nativní kód.[10] Dokumentace uvádí, že je „užitečná především pro vývojáře, kteří píší knihovny šablon.“[10] decltype byl přidán do hlavní řady GCC Překladač C ++ ve verzi 4.3,[19] vydáno 5. března 2008.[20] decltype je také přítomen v Codegear je C ++ Builder 2009,[21] the Překladač Intel C ++,[22] a Zvonit.[23]

Reference

  1. ^ A b Miller, William M. (2009-09-29). „C ++ standardní problémy se základním jazykem, revize 66“. ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2009-10-03.
  2. ^ A b C d E F G Gregor, Douglas; Järvi, Jaakko; Siek, Jeremy; Stroustrup, Bjarne (2003-04-28). „Decltype and auto“ (PDF). ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2015-08-28.
  3. ^ Kalev, Danny (08.05.2008). "Vyčištění syntaxe funkce pomocí decltype". DevX.com. Citováno 2009-09-04.
  4. ^ A b C Gibbons, Bill (01.11.2000). „Přenosný“ typ „operátora“. Dr. Dobb's Journal. Citováno 2009-09-03.
  5. ^ A b Alexandrescu, Andrei (10.10.2000). "Obecné : Mapování mezi typy a hodnotami". Dr. Dobb's Journal. Citováno 2009-09-03.
  6. ^ A b Koenig, Andrew; Barbara E. Moo (01.02.2002). „C ++ Made Easier: Naming Unknown Types“. Dr. Dobb's Journal. Citováno 2009-09-03.
  7. ^ Dewhurst, Steve (01.08.2000). „Obecná znalost: Bitový typ operátora, 1. část“. Dr. Dobb's Journal. Citováno 2009-09-03.
  8. ^ Koenig, Andrew; Barbara E. Moo (2011-07-19). „4 Užitečné nové funkce v C ++ 0x“. Dr. Dobb's Journal. Citováno 2012-01-12.
  9. ^ A b C Dos Reis, Gabriel; Järvi, Jaakko; Stroustrup, Bjarne (10. 10. 2004). „Decltype and auto (revision 4)“ (PDF). ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2009-09-04.
  10. ^ A b C „operátor decltype“. společnost Microsoft. Citováno 2009-09-04.
  11. ^ A b Dos Reis, Gabriel; Järvi, Jaakko; Stroustrup, Bjarne (18.7.2007). „Dekltyp (revize 7): navrhované znění“ (PDF). ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2009-09-04.
  12. ^ A b C Becker, Pete. „Pracovní koncept, standard pro programovací jazyk C ++“ (PDF). ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2009-09-04.
  13. ^ Miller, William M. (2009-08-03). „Zprávy o defektech standardního jádra jazyka C ++, revize 65“. ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2009-09-15.
  14. ^ Miller, William M. (2009-08-03). „Uzavřené problémy se standardním základním jazykem C ++, revize 65“. ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2009-09-04.
  15. ^ Dos Reis, Gabriel; Järvi, Jaakko; Stroustrup, Bjarne (11.11.2006). „Dekltyp (revize 6): navrhované znění“ (PDF). ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2009-10-03.
  16. ^ Miller, William M. (2009-08-03). „Stav komentáře C ++ CD1“. ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2009-10-03.
  17. ^ Miller, William M. (2010-03-29). „Zprávy o defektech standardního jádra jazyka C ++, revize 69“. ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2010-04-10.
  18. ^ Vandevoorde, Daveed (03.02.2010). „Základní problémy 743 a 950: Další využití decltype (...)“ (PDF). ISO / IEC JTC1 / SC22 / WG21 - Výbor pro standardy C ++. Citováno 2010-04-10.
  19. ^ „Podpora C ++ 0x v GCC“. Free Software Foundation. 2009-08-27. Citováno 2009-09-04.
  20. ^ „Série vydání GCC 4.3“. Free Software Foundation. 2009-08-13. Citováno 2009-09-04.
  21. ^ "Type Specifier decltype (C ++ 0x)". Embarcadero Technologies. Archivovány od originál dne 8.7.2011. Citováno 2009-09-04.
  22. ^ "standardní, Qstd". Intel Corporation. Citováno 2009-09-04.
  23. ^ Gregor, Douglas (2011-01-26). "Nová podpora funkcí C ++ 0x v Clangu". Archivovány od originál dne 30.01.2011.

externí odkazy