Как сократить длинные (квалифицированные) идентификаторы в C++?
В Java я считаю очень простым использование пространств имен. Для каждого квалифицированного идентификатора Ident
Я использую в исходном файле, я ставлю import ns1.ns2.ns2.ns3.ns4.Ident;
в верхней части файла. Тогда я могу использовать (короткие) неквалифицированные имена везде в моем исходном коде. import
Оператор не может вызвать каких-либо проблем, потому что он применяется только к файлу, в котором он записан.
Тем не менее, я не совсем уверен, как избавиться от классификаторов пространства имен в C++ лучшим способом.
Наиболее очевидным решением, вероятно, будет using
а также using namespace
заявление. Однако, это кажется довольно плохим решением, по крайней мере, в случае заголовочных файлов, потому что операторы using не ограничиваются одним файлом, в который они записаны. Так using
исключается в случае, например, тонких библиотек, состоящих только из заголовочных файлов с реализациями непосредственно внутри или в случае заголовочных файлов в целом.
Другой вариант, который я использую до сих пор, заключается в добавлении для каждого квалифицированного имени, которое я использую в классе, соответствующего typedef
в приватной секции класса. Поэтому, сравнивая этот подход с Java, я в основном беру весь список операторов импорта, заменяя import
с typedef
и поместите его в объявление класса.
Однако мне не очень нравится этот подход, потому что пользователи моих классов, строго говоря, не знают типы возвращаемых значений и значений параметров, потому что типы в объявлениях методов являются частными типами соответствующих классов.
Хорошо, теперь мы можем сделать все это typedef
вещи публичные. Но это, вероятно, не очень хорошая идея, так как мы будем переопределять каждый тип много-много раз. Просто подумай о структуре ns1::ns2::ns3::MyStructure
и два класса MyClassA
а также MyClassB
, Оба класса имеют метод, который на самом деле должен принимать в качестве параметра экземпляр ns1::ns2::ns3::MyStructure
, Но поскольку каждый класс переопределяет типы, которые он использует, чтобы избавиться от длинных квалифицированных имен, эти два метода теперь принимают параметры "разных" типов, скажем, MyClassA::MyStructure
а также MyClassB::MyStructure
, Это становится еще более броским, когда у нас есть третий класс MyClassC
который работает с экземпляром MyStructure
и нужно вызвать оба метода с ним. Должен ли этот класс объявить этот экземпляр с типом MyClassA::MyStructure
, MyClassB::MyStructure
или же MyClassC::MyStructure
?
Ну, я просто хочу знать: как лучше избавиться от классификаторов пространства имен?
1 ответ
Все создаваемые мной пространства имен вложены в пространство имен моего имени пользователя, которое имеет короткое имя.
Вы можете создать семантически значимое имя пространства имен внутри пространства имен вашего имени пользователя, которое затем будет служить псевдонимом для какого-то действительно длинного пути, тем самым помогая изолировать его от конфликтов переменных.
Для этого вам не нужно ссылаться на опасности определений типов. Вместо этого вы можете просто использовать семантически значимый псевдоним пространства имен (https://en.cppreference.com/w/cpp/language/namespace_alias):
- предположим, что мое корневое пространство имен — mr_user, а имя моего проекта — color_balloons.
пространство имен mr_user_cb01_psp = mr_user::colored_balloons::my_package_namespace::my_subpackage_namespace; mr_user_cb01_psp::MyClass myClass;
В зависимости от того, как вы это используете, чтобы избежать конфликтов в долгосрочной перспективе, вам, вероятно, следует отслеживать псевдонимы вашего пространства имен в безопасном месте, отдельно от ваших проектов. Большую часть времени меня это не беспокоит: я просто использую исходное имя пространства имен в его полной форме.