Главная страница Случайная страница КАТЕГОРИИ: АвтомобилиАстрономияБиологияГеографияДом и садДругие языкиДругоеИнформатикаИсторияКультураЛитератураЛогикаМатематикаМедицинаМеталлургияМеханикаОбразованиеОхрана трудаПедагогикаПолитикаПравоПсихологияРелигияРиторикаСоциологияСпортСтроительствоТехнологияТуризмФизикаФилософияФинансыХимияЧерчениеЭкологияЭкономикаЭлектроника |
Листинг 10.17. Преобразование int в Counter
1: // Листинг 10.17. 2: // Использование конструктора в качестве оператора преобразования типа 3: 4: int 5: #include < iostream.h> 6: 7: class Counter 8: { 9: public: 10: Counter(); 11: Counter(int val); 12: ~Counter(){ } 13: int GetItsVal()const { return itsVal; } 14: void SetItsVal(int x) { itsVal = x; } 15: private: 16: int itsVal; 17: 18: }; 19: 20: Counter:: Counter(): 21: itsVal(0) 22: { } 23: 24: Counter:: Counter(intval): 25: itsVal(val) 26: { } 27: 28: 29: int main() 30: { 31: int theShort = 5; 32: Counter theCtr = theShort; 33: cout < < " theCtr: " < < theCtr.GetItsVal() < < endl; 34: return 0; 35: }
Результат: the Ctr: 5
Анализ: Важные изменения произошли в строке 11, где конструктор перегружен таким образом, чтобы принимать значения типа int, а также в строках 24—26, где данный конструктор применяется. В результате выполнения конструктора переменной-члену класса Counter присваивается значение типа int. Для присвоения значения программа обращается к конструктору, в котором присваиваемое значение передается в качестве аргумента. Процесс осуществляется в несколько шагов. Шаг 1: создание переменной класса Counter с именем theCtr. Это то же самое, что записать: int x = 5, где создается целочисленная переменная x и ей присваивается значение 5. Но в нашем случае создается объект theCtr класса Counter, который инициализируется переменной theShortTHna short int. Шаг 2: присвоение объекту theCtr значения переменной theShort. Но переменная относится к типу short, а не Counter! Первое, что нужно сделать, — это преобразовать ее к типу Counter. Компилятор может делать некоторые преобразования автоматически, но ему нужно точно указать, чего от него хотят. Именно для инструктирования компилятора создается конструктор класса Counter, который содержит единственный параметр, например типа short: class Counter { Counter (short int x); //... }; Данный конструктор создает объект класса Counter, используя временный безымянный объект этого класса, способный принимать значения типа short. Чтобы сделать этот процесс более наглядным, предположим, что для значений типа short создается не безымянный объект, а объект класса Counter с именем wasShort. Шаг 3: присвоение значения объекта wasShort объекту theCtr, что эквивалентно записи " theCtr = wasShort"; На этом шаге временный объект wasShort, созданный при запуске конструктора, замещается на постоянный объект theCtr, принадлежащий классу Counter. Другими словами, значение временного объекта присваивается объекту theCtr. Чтобы понять, как происходит этот процесс, следует четко уяснить принципы работы, справедливые для ВСЕХ перегруженных операторов, определенных с помощью ключевого слова operator. В случае с операторами с двумя операндами (такими как = или +) находящийся справа операнд объявляется как параметр функции оператора, заданной в конструкторе. Так, выражение а = b объявляется как a.operator=(b); Что произойдет, если изменить порядок присвоения, как в следующем примере: 1: Counter theCtr(5); 2: int theShort = theCtr; 3: cout < < " theShort: " < < theShort < < endl; Вновь компилятор покажет сообщение об ошибке. Хотя сейчас компилятор уже знает, как создать временный объект Counter для принятия значения типа int, но он не знает, как осуществить обратный процесс.
|