// Передать конструктору ссылку на объект типа object, public NonGen(object о) { ob = о;
}
// Возвратить объект типа object, public object GetOb() {
return ob;
}
// Показать тип переменной ob. public void ShowTypeO {
Console.WriteLine('Тип переменной ob: ' + ob.GetType());
}
}
// Продемонстрировать применение необобщенного класса, class NonGenDemo { static void Main() {
NonGen iOb;
// Создать•объект класса NonGen. iOb = new NonGen(102);
// Показать тип данных, хранящихся в переменной iOb. iOb.ShowType();
// Получить значение переменной iOb.
//На этот раз потребуется приведение типов, int v = (int) iOb.GetObO;
Console.WriteLine('Значение: ' + v);
Console.WriteLine();
// Создать еще один объект класса NonGen и // сохранить строку в переменной it.
NonGen strOb = new NonGen('Тест на необобщенность');
// Показать тип данных, хранящихся в переменной strOb. strOb.ShowType();
// Получить значение переменной strOb.
//Ив этом случае требуется приведение типов.
String str = (string) strOb.GetOb();
Console.WriteLine('Значение: ' + str);
// Этот код компилируется, но он принципиально неверный! iOb = strOb;
// Следующая строка кода приводит к исключительной // ситуации во время выполнения.
// v = (int) iOb.GetObO; // Ошибка при выполнении!
}
}
При выполнении этой программы получается следующий результат.
Тип переменной ob: System.Int32 Значение: 102
Тип переменной ob: System.String Значение: Тест на необобщенность
Как видите, результат выполнения этой программы такой же, как и у предыдущей программы.
В этой программе обращает на себя внимание ряд любопытных моментов. Прежде всего, тип Т заменен везде, где он встречается в классе Non Gen. Благодаря этому в классе Non Gen может храниться объект любого типа, как и в обобщенном варианте этого класса. Но такой подход оказывается непригодным по двум причинам. Во-первых, для извлечения хранящихся данных требуется явное приведение типов. И во-вторых, многие ошибки несоответствия типов не могут быть обнаружены вплоть до момента выполнения программы. Рассмотрим каждую из этих причин более подробно.
Начнем со следующей строки кода.
int v = (int) iOb.GetObO;
Теперь возвращаемым типом метода GetOb () является object, а следовательно, для распаковки значения, возвращаемого методом GetOb (), и его последующего сохранения в переменной v требуется явное приведение к типу int. Если исключить приведение типов, программа не будет скомпилирована. В обобщенной версии этой программы приведение типов не требовалось, поскольку тип int указывался в качестве аргумента типа при создании объекта iOb. А в необобщенной версии этой программы потребовалось явное приведение типов. Но это не только неудобно, но и чревато ошибками.
А теперь рассмотрим следующую последовательность кода в конце анализируемой здесь программы.
// Этот код компилируется, но он принципиально неверный! iOb = strOb;
