C++
C++ est un langage de programmation compilé permettant la programmation sous de multiples paradigmes, dont la programmation procédurale, la programmation orientée objet et la programmation générique. Ses bonnes performances, et sa compatibilité avec le langage C en font un des langages de programmation les plus utilisés dans les applications où la performance est critique. Créé initialement par Bjarne Stroustrup dans les années 1980, le langage C++ est aujourd'hui normalisé par l'ISO. Sa première normalisation date de 1998 (ISO/CEI 14882:1998), ensuite amendée par l'erratum technique de 2003 (ISO/CEI 14882:2003). Une importante mise à jour a été ratifiée et publiée par l'ISO en sous le nom de ISO/IEC 14882:2011, ou C++11[3]. Depuis, des mises à jour sont publiées régulièrement : en (ISO/CEI 14882:2014, ou C++14[4]), en (ISO/CEI 14882:2017, ou C++17[5]) puis en (ISO/IEC 14882:2020, ou C++20[6]) et en (C++23). DénominationEn langage C, HistoireBjarne Stroustrup commence le développement de C with Classes (C avec classes) en 1979[8]. Il travaille alors dans les laboratoires Bell où il est notamment collègue de l'inventeur du C Dennis Ritchie. L'idée de créer un nouveau langage venait de l'expérience en programmation de Stroustrup pour sa thèse de doctorat. Il s'agissait en l'occurrence d'améliorer le langage C. Stroustrup trouvait que Simula avait des fonctionnalités très utiles pour le développement de gros programmes mais qu'il était trop lent pour être utilisé en pratique (cela était dû à un problème d'implémentation du compilateur Simula), tandis que BCPL était rapide mais de trop bas niveau et non adapté au développement de gros logiciels. Quand Stroustrup commença à travailler aux laboratoires Bell, on lui demanda d'analyser le noyau UNIX en vue de faire du calcul distribué. Se rappelant sa thèse, Stroustrup commença à améliorer le langage C avec des fonctionnalités similaires à celle de Simula. C fut choisi parce qu'il est rapide, portable et d'usage général. En outre, il était une bonne base pour le principe original et fondateur de C++ : « vous ne payez pas pour ce que vous n'utilisez pas ». Dès le départ, le langage ajoutait à C la notion de classe (avec encapsulation des données), de classe dérivée, de vérification des types renforcés (typage fort), d'« inlining », et de paramètre par défaut. Alors que Stroustrup développait C with classes, il écrivit CFront, un compilateur qui générait du code source C à partir de code source C with classes. La première commercialisation se fit en . En le nom « C++ » est inventé, et en le nom du langage passa de C with classes à celui de « C++ ». Parmi les nouvelles fonctionnalités qui furent ajoutées au langage, il y avait les fonctions virtuelles, la surcharge des opérateurs et des fonctions, les références, les constantes, le contrôle du typage amélioré et les commentaires en fin de ligne. En 1985 fut publiée la première édition de The C++ Programming Language, apportant ainsi une référence importante au langage qui n'avait pas encore de standard officiel. En , c'est la sortie de la version 2.0 de C++. Parmi les nouvelles fonctionnalités, il y avait l'héritage multiple, les classes abstraites, les fonctions membres statiques, les fonctions membres constantes, et les membres protégés. En , The Annotated C++ Reference Manual (« ARM ») fut publié apportant les bases du futur standard. Les ajouts de fonctionnalités tardifs qu'il comportait couvraient les templates, les exceptions, les espaces de noms, les nouvelles conversions et le type booléen. Pendant l'évolution du langage C++, la bibliothèque standard évoluait de concert. Le premier ajout à la bibliothèque standard du C++ concernait les flux d'entrées/sorties qui apportaient les fonctionnalités nécessaires au remplacement des fonctions C traditionnelles telles que Personne ne possède le langage C++. Il est libre de droits[9] ; cependant, le document de standardisation n'est quant à lui pas disponible gratuitement. Fonctionnalités introduitesOn pouvait considérer que C++ « était du C » avec un ajout de fonctionnalités. Cependant, plusieurs programmes syntaxiquement corrects en C ne le sont pas en C++, à commencer bien sûr par ceux qui font usage d'identificateurs correspondant à des mots-clefs en C++. Il est également à noter que l'architecture d'un programme C++ moderne (C++11) est differente de celle d'un programme en C. Parmi les fonctionnalités ajoutées figurent :
La compilation d'un programme en C++ effectue également un contrôle plus minutieux du typage. Bibliothèque standardLa bibliothèque standard du C++ englobe la Standard Template Library (STL) qui met à la disposition du programmeur des outils puissants comme des collections (conteneurs) et des itérateurs. À l'origine, la STL était une bibliothèque développée par Alexander Stepanov qui travaillait pour Hewlett-Packard. Dans la norme, celle-ci n'est pas appelée STL, car elle est considérée comme faisant partie de la bibliothèque standard de C++. Toutefois, beaucoup de personnes l'appellent encore de cette manière pour distinguer d'une part, les fonctions d'entrées/sorties comprises dans cette bibliothèque et, d'autre part, celles fournies par la bibliothèque C. Comme en C, l'utilisation d'une bibliothèque peut se faire par l'intermédiaire de la directive Programmation orientée objetLe langage C++ utilise les concepts de la programmation orientée objet et permet entre autres :
EncapsulationL'encapsulation permet de faire abstraction du fonctionnement interne (c'est-à-dire la mise en œuvre) d'une classe et ainsi de ne se préoccuper que des services rendus par celle-ci. C++ met en œuvre l'encapsulation en permettant de déclarer les membres d'une classe avec le mot réservé
C++ n'impose pas l'encapsulation des membres dans leurs classes. On pourrait donc déclarer tous les membres publics, mais en perdant une partie des bénéfices apportés par la programmation orientée objet. Il est de bon usage de déclarer toutes les données privées, ou au moins protégées, et de rendre publiques les fonctions membres agissant sur ces données. Ceci permet de cacher les détails de la mise en œuvre de la classe. « Hello, world »Voici l'exemple de Hello world donné dans The C++ Programming Language, Third Edition[12] de Bjarne Stroustrup : #include<iostream>
int main()
{
std::cout << "Hello, new world!\n";
}
Dans l'exemple ci-dessus, le code source Avec l'ajout des modules dans le langage et de la fonction print dans la bibliothèque standard, l'exemple Hello World peut depuis C++23 être écrit : import std;
int main ()
{
std::println("Hello, World!");
}
Espace de nomsEn C++, le mot clef Dans différents espaces de noms, on peut ainsi définir des entités (routines, variables, etc.) ayant le même identificateur. L'ambiguïté est résolue en utilisant le nom de l'espace de nom devant l'opérateur de portée ( Directive « using »Il est possible de spécifier un espace de noms précis à utiliser afin d'éviter d'avoir à recourir à l'opérateur de résolution de portée. Pour cela, le mot-clé using namespace nom_du_namespace;
// ou
using nom_d_un_symbole;
// ou
using enum nom_d_un_enum_class; // C++20
Ainsi, pour utiliser la variable #include <iostream>
using namespace std;
int main()
{
cout << "Hello, new world!\n";
}
Il est aussi possible, et conseillé, d'importer un symbole particulier, ou de placer cette instruction dans une fonction afin de limiter la portée : #include <iostream>
int main()
{
using std::cout;
// std::cout est disponible sans utilisation de std::
cout << "Hello, new world!" << std::endl; // mais pas std::endl (endline)
}
void foo()
{
// std::cout n'est plus disponible sans utilisation de std::
std::cout << "Hello, new world!" << std::endl;
}
Le mot-clé #include <iostream>
// Déclaration de la classe de base A.
class A
{
protected:
void f()
{
std::cout << "A::f()\n";
}
public:
void g()
{
std::cout << "A::g()\n";
}
};
// Déclaration de la classe B héritant de A.
class B : public A
{
public:
using A::f; // rend public A::f()
};
// Déclaration de la classe C héritant de A.
class C: public A
{
public:
void g(int Val) // masque A::g()
{
std::cout << "C::g(int)\n";
}
};
// Déclaration de la classe D héritant de A.
class D: public A
{
public:
void g(int Val) // masque A::g()
{
std::cout << "D::g(int)";
}
using A::g; // démasque A::g()
};
int main()
{
A a;
B b;
C c;
D d;
// a.f(); // impossible car f est protégé dans A
a.g();
b.f(); // possible car A::f est publique dans B.
// c.g(); // impossible car A::g() est masquée par C::g(int) dans C
c.g(6); // possible car C::g(int Val) est masquée par C::g(int) dans C
d.g(); // possible car A::g() est démasquée dans D
d.g(5); // possible car D::g() est démasquée dans D
}
Le programme ci-dessus affiche : A::g()
A::f()
C::g(int)
A::g()
D::g(int)
Il est aussi possible de définir un nouveau nom pour un namespace : namespace fs = std::filesystem;
// on peut alors écrire fs::path au lieu de std::filesystem::path
Déclaration et définition de classeIl est d'usage de séparer prototype (déclaration) et implémentation (définition) de classe dans deux fichiers : la déclaration se fait dans un fichier d'en-tête (dont l'extension varie selon les préférences des développeurs : sans extension dans le standard, .h comme en C, .hh ou .hpp ou .hxx pour différencier le code source C++ du C) alors que la définition se fait dans un fichier source (d'extension également variable : .c comme en C, .cc ou .cpp ou .cxx pour différencier C++ du C). Déclaration de classeExemple de la déclaration d'une classe comportant des attributs privés et des fonctions membres publiques : // messageinternet.hpp
#include <string_view>
class MessageInternet
{
private: // Ici, private: est optionnel car il est par défaut.
std::string_view m_sujet;
std::string_view m_expediteur;
std::string_view m_destinataire; // attributs
public:
// constructeur
MessageInternet(
std::string_view sujet,
std::string_view expediteur,
std::string_view destinataire);
// fonctions membres :
auto sujet();
auto expediteur();
auto destinataire();
};
Définition de classeLe nom d'une fonction membre déclarée par une classe doit nécessairement être précédé du nom de la classe suivi de l'opérateur de résolution de portée Exemple de définition des fonctions membres d'une classe (celle déclarée précédemment) : // messageinternet.cpp
#include "messageinternet.hpp"
MessageInternet::MessageInternet(
std::string_view sujet,
std::string_view expediteur,
std::string_view destinataire)
: m_sujet(sujet),
m_expediteur(expediteur),
m_destinataire(destinataire)
{}
auto MessageInternet::sujet() { return m_sujet; }
auto MessageInternet::expediteur() { return m_expediteur; }
auto MessageInternet::destinataire() { return m_destinataire; }
ModèlesLes Modèles (ou templates) permettent d'écrire des variables, des fonctions et des classes en paramétrant le type de certains de leurs constituants (type des paramètres ou type de retour pour une fonction, type des éléments pour une classe collection par exemple). Les modèles permettent d'écrire du code générique, c'est-à-dire qui peut servir pour une famille de fonctions ou de classes qui ne diffèrent que par le type de leurs constituants. Paramètres des modèlesLes paramètres peuvent être de différentes sortes :
Utilité des modèlesEn programmation, il faut parfois écrire de nombreuses versions d'une même fonction ou classe suivant les types de données manipulées. Par exemple, un tableau de Les avantages des modèles sont :
Exemple de modèlesDans la bibliothèque standard C++, on trouve de nombreux templates. On citera à titre d'exemple, les entrées/sorties, les chaînes de caractères ou les conteneurs. Les classes Les fonctions de recherche et de tri sont aussi des templates écrits et utilisables avec de nombreux types. #include <string>
template<typename T>
T max(T a, T b) // Une fonction similaire est aussi définie dans le header <algorithm>
{
return a < b ? b : a;
}
int main()
{
int i = max(3, 5);
char ch = max('e', 'b');
using namespace std::string_literals;
std::string str = max("hello"s, "world"s);
float fp = max<float>(1, 2.2f); // type paramétré donné explicitement
// (obligatoire avec ces paramètres de types différents)
}
Dans la ligne Spécialisation des templatesUn template donné peut avoir plusieurs instanciations possibles selon les types donnés en paramètres. Si un seul paramètre est spécialisé, on parle de spécialisation partielle. Ceci permet par exemple :
#include <cstring>
template<>
const char* max(const char* a, const char* b)
{
// Normalement, le résultat d'une comparaison directe
// entre deux chaînes de caractères est un comportement non défini;
// utiliser std::strcmp le rend défini.
return std::strcmp(a, b) > 0 ? a : b;
}
template<std::size_t N>
struct Factorielle
{
static constexpr std::size_t value = N * Factorielle<N - 1>::value;
};
template<>
struct Factorielle<0>
{
static constexpr std::size_t value = 1;
};
À partir de C++14 pour arriver aux mêmes fins nous pourrions aussi utiliser les variables templates : template<std::size_t N>
constexpr auto factorielle = N * factorielle<N - 1>;
template<>
constexpr auto factorielle<0> = 1;
Ainsi nous pouvons écrire SFINAELe mécanisme décrit par l'abréviation SFINAE (Substitution Failure Is Not an Error) permet de surcharger un template par plusieurs classes (ou fonctions), même si certaines spécialisations, par exemple, ne peuvent pas être utilisées pour tous les paramètres de templates. Le nom décrit précisément le fonctionnement du mécanisme, littéralement l’acronyme de « Un échec de substitution n'est pas une erreur », le compilateur, lors de la substitution, ignore alors les instanciations inapplicables, au lieu d'émettre une erreur de compilation. Par exemple : #include <iostream>
class A
{
public:
int foo()
{
return 3;
}
};
class B
{
public:
int bar()
{
return 5;
}
};
class C : public A, public B {};
template <typename T>
auto f(const T& f) -> decltype(f.foo())
{
return f.foo();
}
template <typename T>
auto f(const T& f) -> decltype(f.bar())
{
return f.bar();
}
int main()
{
A a{};
B b{};
std::cout << f(a) << '\n'; // affiche 3 en appellant a.foo()
std::cout << f(b) << '\n'; // affiche 5 en appellant b.bar()
// std::cout << f(C{}) << '\n'; // ne compile pas, en effet, C a les deux
// fonctions membres, ainsi la déduction est
// ambigue
// std::cout << f(5) << '\n'; // ne compile pas, en effet, int n'a aucune des
// deux fonctions membre
}
Ici f est définie deux fois, le type de retour est conditionné par le type donné en paramètre, il est du type du retour de f.foo() dans le premier cas et de celui de f.bar() dans le deuxième cas. Ainsi, si on appelle f avec un objet de la classe A, seule la première fonction fonctionne puisque la classe A n'a pas de fonction membre bar() et donc la substitution est possible avec cette première version mais pas pour la deuxième. Ainsi, f(a) appelle la première version de f, f(b) appelle la deuxième avec le même raisonnement, mais cette fois pour la fonction membre bar(). Si lors d'un développement à venir, un développeur venait à écrire une nouvelle classe ayant une fonction membre publique foo ou bien (ou exclusif) bar, il pourrait également utiliser f avec. Polymorphisme et fonctions membres virtuellesLe polymorphisme d'inclusion est mis en œuvre à l'aide du mécanisme des fonctions membres virtuelles en C++. Une fonction membre est rendue virtuelle par le placement du mot-clé Le mot-clé En particulier, il est obligatoire d'utiliser le mot-clé Ce type de polymorphisme (le polymorphisme d'inclusion) est dit dynamique. Le mécanisme de la surcharge de fonction qui est un polymorphisme ad hoc est de type statique. Dans les deux cas il faut appliquer une logique (par exemple : le nombre et le type des paramètres) pour résoudre l'appel. Dans le cas de la surcharge de fonction, la logique est entièrement calculée à la compilation. Ce calcul permet des optimisations rendant le polymorphisme statique plus rapide que sa version dynamique. La liaison dynamique de fonctions membres issues du mécanisme des fonctions membres virtuelles induit souvent une table cachée de résolution des appels, la table virtuelle. Cette table virtuelle augmente le temps nécessaire à l'appel de fonction membre à l'exécution par l'ajout d'une indirection supplémentaire. Le choix entre liaison dynamique et surcharge (polymorphisme dynamique et statique) est typiquement un problème de calculabilité des appels, ayant souvent pour conséquence finale un choix entre expressivité et performance. Malgré ce dynamisme, il est à noter que le compilateur est capable de « dévirtualiser » les appels de fonctions membres qui peuvent être résolus au moment de la compilation. Dans gcc par exemple, l'option Outils de développementUn programme C++ peut être produit avec des outils qui automatisent le processus de construction. Les plus utilisés sont :
Environnements de développement
Compilateurs
BibliothèquesRéférences
AnnexesBibliographieOuvrages en langue anglaise
Ouvrages en langue française
Articles connexesLiens externes
|