Герберт Шилдт - C# 4.0: полное руководство
- Название:C# 4.0: полное руководство
- Автор:
- Жанр:
- Издательство:ООО И.Д. Вильямс
- Год:2011
- Город:Москва -- Киев
- ISBN:978-5-8459-1684-6
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Герберт Шилдт - C# 4.0: полное руководство краткое содержание
В этом полном руководстве по C# 4.0 - языку программирования, разработанному специально для среды .NET, - детально рассмотрены все основные средства языка: типы данных, операторы, управляющие операторы, классы, интерфейсы, методы, делегаты, индексаторы, события, указатели, обобщения, коллекции, основные библиотеки классов, средства многопоточного программирования и директивы препроцессора. Подробно описаны новые возможности C#, в том числе PLINQ, библиотека TPL, динамический тип данных, а также именованные и необязательные аргументы. Это справочное пособие снабжено массой полезных советов авторитетного автора и сотнями примеров программ с комментариями, благодаря которым они становятся понятными любому читателю независимо от уровня его подготовки.
Книга рассчитана на широкий круг читателей, интересующихся программированием на C#.Введите сюда краткую аннотацию
C# 4.0: полное руководство - читать онлайн бесплатно ознакомительный отрывок
Интервал:
Закладка:
И во-вторых, ограничение на базовый класс гарантирует использование только тех аргументов типа, которые поддерживают указанный базовый класс. Это означает, что для любого ограничения, накладываемого на базовый класс, аргумент типа должен обозначать сам базовый класс или производный от него класс. Если же попытаться использовать аргумент типа, не соответствующий указанному базовому классу или не наследующий его, то в результате возникнет ошибка во время компиляции.
Ниже приведена общая форма наложения ограничения на базовый класс, в которой используется оператор where:
where Т : имя_базового_класса
где T обозначает имя параметра типа, а имя_базового_класса — конкретное имя ограничиваемого базового класса. Одновременно в этой форме ограничения может быть указан только один базовый класс.
В приведенном ниже простом примере демонстрируется механизм наложения ограничения на базовый класс.
// Простой пример, демонстрирующий механизм наложения
// ограничения на базовый класс.
using System;
class A {
public void Hello() {
Console.WriteLine("Hello");
}
}
// Класс В наследует класс А.
class B : A { }
// Класс С не наследует класс А.
class C { }
//В силу ограничения на базовый класс во всех аргументах типа,
// передаваемых классу Test, должен присутствовать базовый класс А.
class Test where T : A {
T obj;
public Test(T o) {
obj = o;
}
public void SayHello() {
// Метод Hello() вызывается, поскольку
// он объявлен в базовом классе А.
obj.Hello();
}
}
class BaseClassConstraintDemo {
static void Main() {
A a = new A();
B b = new B();
C с = new C();
// Следующий код вполне допустим, поскольку
// класс А указан как базовый.
Test tl = new Test(a);
tl.SayHello();
// Следующий код вполне допустим, поскольку
// класс В наследует от класса А.
Test t2 = new Test(b);
t2.SayHello();
// Следующий код недопустим, поскольку
// класс С не наследует от класса А.
// Test t3 = new Test(c); // Ошибка!
// t3.SayHello(); // Ошибка!
}
}
В данном примере кода класс А наследуется классом В, но не наследуется классом С. Обратите также внимание на то, что в классе А объявляется метод Hello(), а класс Test
объявляется как обобщенный следующим образом.
class Test where Т : А {
Оператор where в этом объявлении накладывает следующее ограничение: любой аргумент, указываемый для типа Т, должен иметь класс А в качестве базового.
А теперь обратите внимание на то, что в классе Test
объявляется метод SayHello()
, как показано ниже.
public void SayHelloO {
// Метод Hello() вызывается, поскольку он объявлен в базовом классе А. obj.Hello();
}
Этот метод вызывает в свою очередь метод Hello()
для объекта obj типа Т. Любопытно, что единственным основанием для вызова метода Hello()
служит следующее требование ограничения на базовый класс: любой аргумент типа, привязанный к типу Т, должен относиться к классу А или наследовать от класса А, в котором объявлен метод Hello()
. Следовательно, любой допустимый тип Т будет также определять метод Hello()
. Если бы данное ограничение на базовый класс не было наложено, то компилятору ничего не было бы известно о том, что метод Hello()
может быть вызван для объекта типа Т. Убедитесь в этом сами, удалив оператор where из объявления обобщенного класса Test. В этом случае программа не подлежит компиляции, поскольку теперь метод Hello()
неизвестен.
Помимо разрешения доступа к членам базового класса, ограничение на базовый класс гарантирует, что в качестве аргументов типа могут быть переданы только те типы данных, которые наследуют базовый класс. Именно поэтому приведенные ниже строки кода закомментированы.
// Test t3 = new Test(c); // Ошибка!
// t3.SayHello(); // Ошибка!
Класс С не наследует от класса А, и поэтому он не может использоваться в качестве аргумента типа при создании объекта типа Test. Убедитесь в этом сами, удалив символы комментария и попытавшись перекомпилировать этот код.
Прежде чем продолжить изложение дальше, рассмотрим вкратце два последствия наложения ограничения на базовый класс. Во-первых, это ограничение разрешает доступ к членам базового класса из обобщенного класса. И во-вторых, оно гарантирует допустимость только тех аргументов типа, которые удовлетворяют данному ограничению, обеспечивая тем самым типовую безопасность.
В предыдущем примере показано, как накладывается ограничение на базовый класс, но из него не совсем ясно, зачем это вообще нужно. Для того чтобы особое значение ограничения на базовый класс стало понятнее, рассмотрим еще один, более практический пример. Допустим, что требуется реализовать механизм управления списками телефонных номеров, чтобы пользоваться разными категориями таких списков, в частности отдельными списками для друзей, поставщиков, клиентов и т.д. Для этой цели можно сначала создать класс P honeNumber
, в котором будут храниться имя абонента и номер его телефона. Такой класс может иметь следующий вид.
// Базовый класс, в котором хранятся имя абонента и номер его телефона,
class PhoneNumber {
public PhoneNumber(string n, string num) {
Name = n;
Number = num;
}
// Автоматически реализуемые свойства, в которых
// хранятся имя абонента и номер его телефона,
public string Number { get; set; }
public string Name { get; set; }
}
Далее создадим классы, наследующие класс PhoneNumber: Friend и Supplier
. Эти классы приведены ниже.
// Класс для телефонных номеров друзей,
class Friend : PhoneNumber {
public Friend(string n, string num, bool wk) : base(n, num)
{
IsWorkNumber = wk;
}
public bool IsWorkNumber { get; private set; }
// ...
}
// Класс для телефонных номеров поставщиков,
class Supplier : PhoneNumber {
Интервал:
Закладка: