Welcome here! | Articles | Main projects | About me
(FR) using namespace
, vraiment une mauvaise pratique ?
Sommaire
Introduction
Tout comme beaucoup de langages, C++ est bien fait. La stdlib est vaste, et tout est bien rangé dans le namespace std
.
Sauf que des fois, on aimerait bien ne pas avoir à tout préfixer à coups de std::
. Avoir une fonction qui découpe une std::string
en std::vector<std::string>
et les affiche via std::cout
, ça commence à faire beaucoup de std::
quand même.
Là où certains puristes menaceraient de vous casser les genoux si vous utilisez using namespace std
(ou autre namespace), je vous propose des solutions alternatives et des cas d’usage où cette instruction est -selon moi- acceptable.
Tout d’abord, une chose simple : JAMAIS de using namespace XXX;
dans un header. Sous aucun prétexte. Si vous importez le namespace std
dans un header, et que le code client décide de créer une classe, par exemple string
, que va-t-il se passer ?
Cela étant dit, j’ai souvent lu et entendu çà et là qu’il était tout aussi dangereux d’en faire de même dans un fichier source. Je ne suis pas tout à fait d’accord avec cette affirmation, et je vais l’expliquer dans ce petit article.
Alternatives et cas d’usages
A mon sens, il n’est absolument pas idiot d’utiliser un namespace dans un fichier source. Car il n’y a aucun risque d’affecter le code client de cette manière.
Néanmoins, je vais tout de même nuancer ce propos. S’il n’y a aucun risque d’affecter le code client, il est possible d’affecter le vôtre. Aussi, il faut être sûr de ce que l’on fait et de ce que l’on va faire.
D’une manière générale, je propose deux manières alternatives pour réduire la verbosité de votre code sans pour autant risquer de générer des conflits.
Lorsque vous avez un namespace dont le nom est long, préférez un alias :
// Dans un fichier .cpp :
namespace fs = std::filesystem;
// (Ou boost::filesystem si vous n'avez pas de compilateur supportant C++17)
L’intérêt est que l’on peut faire en sorte que fs
soit un alias de boost::filesystem
ou de std::filesystem
(en fonction de la norme utilisée) à coups de #if
. C’est un détail, mais c’est toujours bon à savoir.
On peut également importer sélectivement ce dont on a besoin :
// De manière globale :
using std::cout;
using std::endl;
// ... Ou dans une fonction :
void print_words(std::string const& text) {
using std::cout;
using std::endl;
const auto words = split(text); // Admettons qu'une telle fonction existe
cout << "There's " << words.size() << " words here!" << endl;
for (auto const& word : words) {
cout << word << endl;
}
}
Cette méthode propose l’avantage de pouvoir être utilisée dans une fonction définie dans un header, vu que son scope est la fonction elle-même.
Il est par ailleurs tout à fait viable d’importer un namespace dans une fonction :
// Extrait d'un petit programme fait sur mon temps libre :
void mov(Register& dst, Address src) noexcept {
using namespace constants; // MAX_ADDRESS, MIN_ADDRESS
const auto addr = static_cast<Byte>(src);
assert(addr < MAX_ADDRESS);
assert(addr >= MIN_ADDRESS);
mov(dst, RAM[addr]);
}
Bien que cela ne soit pas nécessairement très utile (car après tout, je n’utiliserais que deux fois constants::
), j’utilise des namespaces explicitement pour éviter d’avoir des instructions à rallonge.
Notez que j’anote tout de même en commentaire ce pour quoi j’importe chacun des namespaces concernés. Ainsi, je sais pour quoi j’en ai besoin, et il devient plus facile de savoir si je peux ou non supprimer l’instruction using namespace XXX
en cas de refactoring.
Conclusion
En résumé, using namespace XXX
n’est dans l’absolu pas une mauvaise pratique. Il y a certes des cas où l’usage de cette instruction n’est pas judicieux (comme dans le cas d’un header, par exemple), mais son utilisation peut être acceptable à des échelles plus réduites (comme des fonctions). Il y a des cas où il est obligatoire de le faire (je pense notamment à std::string_literals
pour pouvoir utiliser l’opérateur ""
). Je conseille toutefois d’anoter en commentaire ce qui nous intéresse lorsque nous utilisons un namespace afin de savoir pourquoi il est utilisé.
Nous avons également la possibilité de faire des alias afin d’avoir des noms plus courts (fs
vs std::filesystem
), et il est tout aussi viable d’importer sélectivement ce dont nous avons besoin (std::cout
, std::endl
, …).