Как упоминалось в главе 9, если перегружается оператор ==, то обычно приходит ся переопределять методы Equals(object) и GetHashCode, поскольку чаще всего требуется, чтобы метод Equals(object) и оператор == функционировали одинаково. Когда же переопределяется метод Equals(object), то следует переопределить и ме тод GetHashCode, чтобы оба метода оказались совместимыми.
Метод ToString возвращает символьную строку, содержащую описание того объекта, для которого он вызывается. Кроме того, метод ToString автоматически вызывается при выводе содержимого объекта с помощью метода WriteLine. Этот метод переопределяется во многих классах, что позволяет приспосабливать описание к конкретным типам объектов, создаваемых в этих классах. Ниже приведен пример применения данного метода. // Продемонстрировать применение метода ToString using System; class MyClass { static int count = 0; int id; public MyClass { id = count; count++; } public override string ToString { return "Объект #" + id + " типа MyClass"; } } class Test { static void Main { MyClass ob1 = new MyClass; MyClass ob2 = new MyClass; MyClass ob3 = new MyClass; Console.WriteLine(obi); Console.WriteLine(ob2); Console.WriteLine(ob3); } }
При выполнении этого кода получается следующий результат. Объект #0 типа MyClass Объект #1 типа MyClass Объект #2 типа MyClass Упаковка и распаковка
Как пояснялось выше, все типы в С#, включая и простые типы значений, являются производными от класса object. Следовательно, ссылкой типа object можно вос пользоваться для обращения к любому другому типу, в том числе и к типам значений. Когда ссылка на объект класса object используется для обращения к типу значения, то такой процесс называется упаковкой. Упаковка приводит к тому, что значение про стого типа сохраняется в экземпляре объекта, т.е. "упаковывается" в объекте, который затем используется как и любой другой объект. Но в любом случае упаковка происхо дит автоматически. Для этого достаточно присвоить значение переменной ссылочного типа object, а об остальном позаботится компилятор С#.
Распаковка представляет собой процесс извлечения упакованного значения из объекта. Это делается с помощью явного приведения типа ссылки на объект класса object к соответствующему типу значения. Попытка распаковать объект в другой тип может привести к ошибке во время выполнения.
Ниже приведен простой пример, демонстрирующий упаковку и распаковку. // Простой пример упаковки и распаковки. using System; class BoxingDemo { static void Main { int x; object obj; х = 10; obj = x; // упаковать значение переменной х в объект int у = (int)obj; // распаковать значение из объекта, доступного по // ссылке obj, в переменную типа int Console.WriteLine(у); } }
В этом примере кода выводится значение 10. Обратите внимание на то, что значе ние переменной х упаковывается в объект простым его присваиванием переменной obj, ссылающейся на этот объект. А затем это значение извлекается из объекта, до ступного по его ссылке obj, и далее приводится к типу int.
Ниже приведен еще один, более интересный пример упаковки. В данном случае значение типа int передается в качестве аргумента методу Sqr, который, в свою очередь, принимает параметр типа object. // Пример упаковки при передаче значения методу. using System; class BoxingDemo { static void Main { int x; x = 10; Console.WriteLine("Значение x равно: " + x); // значение переменной x автоматически упаковывается // когда оно передается методу Sqr. х = BoxingDemo.Sqr(х); Console.WriteLine("Значение x в квадрате равно: " + х); } static int Sqr(object о) { return (int)о * (int)о; } }
Вот к какому результату приводит выполнение этого кода. Значение х равно: 10 Значение х в квадрате равно: 100
В данном примере значение переменной х автоматически упаковывается при пере даче методу Sqr.
Упаковка и распаковка позволяют полностью унифицировать систему типов в С#. Благодаря тому что все типы являются производными от класса object, ссылка на значение любого типа может быть просто присвоена переменной ссылочного типа object, а все остальное возьмут на себя упаковка и распаковка. Более того, методы класса object оказываются доступными всем типам, поскольку они являются про изводными от этого класса. В качестве примера рассмотрим довольно любопытную программу. // Благодаря упаковке становится возможным вызов методов по значению! using System; class MethOnValue { static void Main { Console.WriteLine(10.ToString); } }