44. Предпочитайте функции, которые не являются ни членами, ни друзьями
44. Предпочитайте функции, которые не являются ни членами, ни друзьями
Резюме
Там, где это возможно, предпочтительно делать функции не членами и не друзьями классов.
Обсуждение
Функции, не являющиеся членами или друзьями классов, повышают степень инкапсуляции путем снижения зависимостей: тело такой функции не может зависеть от закрытых и защищенных членов класса (см. рекомендацию 11). Они также разрушают монолитность классов, снижая связность (см. рекомендацию 33), и повышают степень обобщенности, так как сложно писать шаблоны, которые не знают, является ли интересующая нас операция членом данного типа или нет (см. рекомендацию 67).
Для определения того, должна ли функция быть реализована как член и/или друг класса, можно воспользоваться следующим алгоритмом:
// Если у вас нет выбора - делайте функцию членом.
Если функция представляет собой один из операторов =, ->,
[] или (), которые должны быть членами,
то
делайте данную функцию членом класса.
// Если функция может быть не членом и не другом либо
// имеются определенные преимущества от того, чтобы сделать
// ее не членом и другом
иначе если 1. функция требует левый аргумент иного типа
(как, например, в случае операторов >> или <<)
или 2. требует преобразования типов для левого аргумента,
или 3. может быть реализована с использованием только
открытого интерфейса класса
то
сделайте ее не членом класса (и, при необходимости,
в случаях 1 и 2 - другом)
Если функция требует виртуального поведения,
то
добавьте виртуальную функцию-член для обеспечения
виртуального поведения, и реализуйте функцию-не член
с использованием этой виртуальной функции.
иначе
сделайте ее функцией-членом.
Примеры
Пример. basic_string. Стандартный класс basic_string чересчур монолитен и имеет 103 функции-члена, из которых 71 без потери эффективности можно сделать функциями, не являющимися ни членами, ни друзьями класса. Многие из них дублируют функциональность, уже имеющуюся в качестве алгоритма стандартной библиотеки, либо представляют собой алгоритмы, которые могли бы использоваться более широко, если бы не были спрятаны в классе basic_string. (См. рекомендации 5 и 32, а также [Sutter04].)
Ссылки
[Lakos96] §3.6.1, §9.1.2 • [McConnell93] §5.1-4 • [Murray93] §2.6 • [Meyers00] • [Stroustrup00] §10.3.2, §11.3.2, §11.3.5, §11.5.2, §21.2.3.1 • [Sutter00] §20 • [Sutter04] §37-40