In C kann SFINAE (Substitution Failure Is Not An Error) verwendet werden Vorlagenspezialisierungen basierend auf Typmerkmalen bedingt aktivieren oder deaktivieren. Ein häufiges Problem tritt jedoch auf, wenn SFINAE in Vorlagenrückgabetypen und Vorlagenparametern unterschiedlich angewendet wird.
Bedenken Sie die folgende Hauptfunktion:
int main() { foo(5); foo(3.4); }
Wenn SFINAE wie unten gezeigt in Vorlagenparametern platziert wird, schlägt die Kompilierung mit an fehl Fehler:
template<typename T, typename = typename std::enable_if<std::is_integral<T>::value>::type> auto foo(T) -> void { std::cout << "I'm an integrer!\n"; } template<typename T, typename = typename std::enable_if<std::is_floating_point<T>::value>::type> auto foo(T) -> void { std::cout << "I'm a floating point number!\n"; }
Diese Implementierung funktioniert jedoch korrekt:
template<typename T> auto foo(T) -> typename std::enable_if<std::is_integral<T>::value>::type { std::cout << "I'm an integrer!\n"; } template<typename T> auto foo(T) -> typename std::enable_if<std::is_floating_point<T>::value>::type { std::cout << "I'm a floating point number!\n"; }
Der Unterschied liegt in der Art und Weise, wie das Überladen von Funktionsvorlagen im C-Standard (14.5.6.1) definiert ist. Überladene Vorlagen sind gleichwertig, wenn sie denselben Funktionsnamen, dieselbe Anzahl von Parametern und dieselben Parametertypen haben. Standard-Template-Argumente werden in diesem Vergleich jedoch nicht berücksichtigt.
In der fehlerhaften Implementierung ist die SFINAE-Bedingung als Standard-Template-Argument enthalten, was dazu führt, dass zwei identische Funktionstemplates doppelt definiert werden. Dies löst den Kompilierungsfehler aus.
In der korrekten Implementierung wird die SFINAE-Bedingung als Teil des Vorlagenrückgabetyps verwendet. Da es sich dabei um ausdrucksverweisende Vorlagenparameter handelt, haben die beiden Funktionen unterschiedliche Signaturen und ermöglichen somit, dass SFINAE wie vorgesehen funktioniert.
Das obige ist der detaillierte Inhalt vonSFINAE in C: Rückgabetyp vs. Parametertyp für bedingte Vorlagenspezialisierung?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!