Доступ к членам вложенного класса

В классе Employee не объявлены специальные методы доступа к переменным- членам класса String. Если объект Edie класса Employee попытается напрямую обратиться к переменной-члену itsLen, содержащейся в ero собственной переменной- члене itsFirstName, это приведет к возникновению ошибки компиляции. Однако в таком обращении нет необходимости. Методы доступа класса Employee просто создают интерфейс для класса String, и классу Employee нет нужды беспокоиться о деталях выполнения класса String, а также о том, каким образом собственная целочисленная переменная-член itsSalary хранит свое значение.

Фильтрация доступа к вложенным классам

Вы, наверное, уже заметили, что в классе String перегружается operator+. В классе Employee доступ к перегруженной функции operator+ заблокирован. Дело в том, что в объявлениях методов доступа класса Employee указано, что все эти методы, такие как GetFirstName(), возвращают константные ссылки. Поскольку функция operator+ не является (и не может быть) константой (она изменяет объект, для которого вызывается), попытка написать следующую строку приведет к сообщению об ошибке компиляции:

String buffer = Edie.GetFirstName() + Edie.GetLastName();

Функция GetFirstName() возвращает константную строку и вы не можете использовать operator+ с константным объектом.

Чтобы устранить эту проблему, следует перегрузить функцию GetFirstName() таким образом, чтобы она стала не константной:

const String & GetFirstName() const { return itsFirstName; }

String & GetFirstName() { return itsFirstName; }

Как видите, возвращаемое значение больше не является константой, также как и сама функция-член. Изменения возвращаемого значения недостаточно для перегрузки имени функции. Необходимо изменить константность самой функции.

Цена вложений

Важно отметить, что пользователю придется "расплачиваться" за каждый объект внешнего класса всякий раз при создании или копировании объекта Employee.

Снимите символы комментариев с операторов cout листинга 15.1 (строки 38, 51, 63, 75, 84 и 100), и вы увидите, как часто они вызываются. В листинге 15.3 представлена та же программа, что и в листинге 15.2, только в этом примере добавлены операторы печати, которые будут показывать сообщения на экране всякий раз при выполнении конструктора класса Employee. Это позволит наглядно увидеть весь процесс создания объектов в программе.

Примечание: До компиляции этого листинга разблокируйте строки 38, 51, 63, 75, 84 и 100 в листинге 15.1.

Листинг 15.3. Конструкторы вложенных классов

1: #include "String.hpp"

2:

3: class Employee

4: {

5:

6: public:

7: Employee();

8: Employee(char *, char *, char *, long);

9: ~Employee();

10: Employee(const Employee&);

11: Employee & operator= (const Employee &);

12:

13: const String & GetFirstName() const

14: { return itsFirstName; }

15: const String & GetLastName() const { return itsLastName; }

16: const String & GetAddress() const { return itsAddress; }

17: long GetSalary() const { return itsSalary; }

18:

19: void SetFirstName(const String & fName)

20: { itsFirstName = fName; }

21: void SetLastName(const String & lName)

22: { itsLastName = lName; }

23: void SetAddress(const String & address)

24: { itsAddress = address; }

25: void SetSalary(long salary) { itsSalary = salary; }

26: private:

27: String itsFirstName;

28: String itsLastName;

29: String itsAddress;

30: long itsSalary;

31: };

32:

33: Employee::Employee();

34: itsFirstName(""),

35: itsLastName(""),

36: itsAddress(""),

37: itsSalary(0)

38: { }

39:

40: Employee::Employee(char * firstName, char * lastName,

41: char * address, long salary):

42: itsFirstName(firstName),

43: itsLastName(lastName),

44: itsAddrsss(address),

45: itsSalary(salary)

46: { }

47:

48: Employee:;Employee(const Employee & rhs):

49: itsFirstName(rhs,GetFirstName()),

50: itsLastName(rhs,GetLastName()),

51: itsAddress(rhs.GetAddress()),

52: itsSalary(rhs.GetSalary())

53: { }

54:

55: Employee::~Employee() { }

56:

57: Employee & Employee::operator= (const Employee & rhs)

58: {

59: if (this == &rhs)

60: return *this;

61:

62: itsFirstName = rhs.GetFirstName();

63: itsLastName = rhs.GetLastName();

64: itsAddress = rhs.GetAddress();

65: itsSalary = rhs.GetSalary();

66:

67: return *this;

68: }

69:

70: int main()

71: {

72: cout << "Creating Edie...\n";

73: Employee Edie("Jane","Doe","1461 Shore Parkway", 20000);

74: Edie,SetSalary(20000);

75: cout << "Calling SetFirstName with char *...\n";

76: Edie,SetFirstName("Edythe");

77: cout << "Creating temporary string LastName...\n";

78: String LastName("Levine");

79: Edis,SetLastName(LastName);

80:

81: cout << "Name: ";

82: cout << Edle.QetFirstName().GetString();

83: cout << " " << Edie,GstLastName().GitString();

84: cout << "\nAddress; ";

85: cout << Edi6.GetAddress(),GetString();

86: cout << "\nSalary; " ;

87: cout << Edie.GstSalary();

88: cout << endl;

89: return 0;

90: }

Результат:

Creating Edie...

String(char*) constructor

String(char*) constructor

String(char*) constructor

Calling SetFirstName with char *...

String(char*) constructor

String destructor

Creating temporary string LastName...

String(char*) constructor

Name: Edythe Levine

Address: 1461 Shore Parkway

Salary: 20000

String destructor

String destructor

String destructor

String destructor

Анализ: В листинге 15.3 используются классы, объявленные ранее в листингах 15.1 и 15.2.

Единственное отличие состоит в том, что операторы cout разблокированы. Чтобы упростить обсуждение, строки, выводимые программой на экран, были пронумерованы.

В строке 72 листинга 15.3 выводится сообщение Creating Edie..., которому соответствует первая строка вывода. Для создания объекта Edie класса Employee задаются четыре параметра. Для инициализации трех из них задействуются конструкторы класса String, о чем свидетельствуют три следующие строки вывода.

Строка 75 информирует о вызове функции SetFirstName. Следующая строка программы, Edie.SetFirstName("Edythe"), создает временный объект класса String из строковой константы "Edythe", для чего вновь задействуются соответствующие конструкторы класса String (см. 6-ю строку вывода). Обратите внимание, что этот временный объект уничтожается сразу же после присвоения его значения переменной-члену, о чем свидетельствует вызов деструктора класса String (см. 7-ю строку вывода).

Присвоив имя, программа приступает к присвоению фамилии служащего. Это можно было бы выполнить так же, как и в случае с присвоением имени с помощью автоматически создаваемого временного объекта класса String. Но чтобы показать все возможности, в строке 78 явно создается объект класса String. Конструктор, создающий этот объект, дал о себе знать 9-й строкой вывода. Деструктор не вызывается, поскольку этот объект не удаляется до тех пор, пока не выйдет за границы своей области видимости в конце функции.

Наконец программа выводит на экран персональные сведения о служащем и выходит за область видимости объекта Employee, в результате чего вызываются четыре деструктора класса для удаления объектов этого класса, вложенных в объект Employee, и созданного ранее временного объекта LastName.

Наши рекомендации