Категории
Самые читаемые
PochitayKnigi » Разная литература » Прочее » C# 4.0 полное руководство - 2011 - Герберт Шилдт

C# 4.0 полное руководство - 2011 - Герберт Шилдт

Читать онлайн C# 4.0 полное руководство - 2011 - Герберт Шилдт

Шрифт:

-
+

Интервал:

-
+

Закладка:

Сделать
1 ... 134 135 136 137 138 139 140 141 142 ... 268
Перейти на страницу:

// который является базовым по накладываемому ограничению, if(phList[i].Number == number) return phList[i];

}

// Номер телефона отсутствует в списке, throw new NotFoundException();

}

// ...

}

Ограничение на базовый класс разрешает коду в классе PhoneList доступ к свойствам Name и Number для управления любым видом списка телефонных номеров. Оно гарантирует также, что для построения объекта класса PhoneList будут использоваться только доступные типы. Обратите внимание на то, что в классе PhoneList генерируется исключение NotFoundException, если имя или номер телефона не найдены. Это специальное исключение, объявляемое ниже.

class NotFoundException : Exception {

/* Реализовать все конструкторы класса Exception.

Эти конструкторы выполняют вызов конструктора базового класса.

Класс NotFoundException ничем не дополняет класс Exception и поэтому не требует никаких дополнительных действий. */

public NotFoundException()    :    base()    {    }

public NotFoundException(string str) : base (str) { }

public NotFoundException(

string str, Exception inner) : base(str, inner) { } protected NotFoundException(

System.Runtime.Serialization.Serializationlnfo si,

System.Runtime.Serialization.StreamingContext sc) : base(si, sc) { }

В данном примере используется только конструктор, вызываемый по умолчанию, но ради наглядности этого примера в классе исключения NotFoundException реализуются все конструкторы, определенные в классе Exception. Обратите внимание на то, что эти конструкторы вызывают эквивалентный конструктор базового класса, определенный в классе.Exception. А поскольку класс исключения NotFoundException ничем не дополняет базовый класс Exception, то для любых дополнительных действий нет никаких оснований.

В приведенной ниже программе все рассмотренные выше фрагменты кода объединяются вместе, а затем демонстрируется применение класса PhoneList. Кроме того, в ней создается класс Email Friend. Этот класс не наследует от класса PhoneNumber, а следовательно, он не может использоваться для создания объектов класса PhoneList.

// Более практический пример, демонстрирующий применение // ограничения на базовый класс.

using System;

// Специальное исключение, генерируемое в том случае,

// если имя или номер телефона не найдены, class NotFoundException : Exception {

/* Реализовать все конструкторы класса Exception.

Эти конструкторы выполняют вызов конструктора базового класса.

Класс NotFoundException ничем не дополняет класс Exception и поэтому не требует никаких дополнительных действий. */

public NotFoundException () base() { }

public NotFoundException(string str) : base(str) { }

public NotFoundException (

string str,Exception inner) : base(str, inner) { } protected NotFoundException(

System.Runtime.Serialization.Serializationlnfо si,

System.Runtime.Serialization.StreamingContext sc) : base (si, sc) { }

}

// Базовый класс, в котором хранятся имя абонента и номер его телефона, class PhoneNumber {

public PhoneNumber(string n, string num) {

Name = n;

Number = num;

}

public string Number { get; set; } public string Name { get; set; }

}

// Класс для телефонных номеров друзей, 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 {

public Supplier(string n, string num) : base(n, num) { }

// ...

}

// Этот класс не наследует от класса PhoneNumber. class EmailFriend {

// ...

}

// Класс PhoneList способен управлять любым видом списка телефонных номеров, // при условии, что он является производным от класса PhoneNumber. class PhoneList<T> where T : PhoneNumber {

T[] phList; int end;

public PhoneList() { phList = new T[10]; end = 0;

}

// Добавить элемент в список, public bool Add(T newEntry) { if(end == 10) return false;

phList[end] = newEntry; end++; return true;

}

// Найти и возвратить сведения о телефоне по заданному имени, public Т FindByName(string name) {

for(int i=0; i<end; i++) {

// Имя может использоваться, потому что его свойство Name // относится к членам класса PhoneNumber, который является // базовым по накладываемому ограничению, if(phList[i].Name == name) return phList[i];

}

// Имя отсутствует в списке, throw new NotFoundException() ;

// Найти и возвратить сведения о телефоне по заданному номеру, public Т FindByNumber(string number) { for(int i=0; i<end; i++)    {

// Номер" телефона также может использоваться, поскольку // его свойство Number относится к членам класса PhoneNumber, // который является базовым по накладываемому ограничению, if(phList[i].Number == number) return phList[i] ;

}

// Номер телефона отсутствует в списке. • throw new NotFoundException ();

}

// ...

}

// Продемонстрировать наложение ограничений на базовый класс, class UseBaseClassConstraint { static void Main() {

// Следующий код вполне допустим, поскольку // класс Friend наследует от класса PhoneNumber.

PhoneList<Friend> plist = new PhoneList<Friend>(); plist.Add(new Friend("Том", "555-1234", true)); plist.Add(new Friend("Гари", "555-6756", true)); plist.Add(new Friend("Матт", "555-9254", false));

try {

// Найти номер телефона по заданному имени друга.

Friend frnd = plist.FindByName("Гари") ;

Console.Write(frnd.Name + ": " + frnd.Number);

if(frnd.IsWorkNumber)

Console.WriteLine(" (рабочий)"); else

Console.WriteLine ();

} catch(NotFoundException) {

Console.WriteLine("He найдено");

}

Console.WriteLine();

• // Следующий код также допустим, поскольку // класс Supplier наследует от класса PhoneNumber.

PhoneList<Supplier> plist2 = new PhoneList<Supplier>(); plist2.Add(new Supplier("Фирма Global Hardware", "555-8834")); plist2.Add(new Supplier ("Агентство Computer Warehouse", "555-9256")); plist2.Add(new Supplier("Компания NetworkCity", "555-2564")); try {

// Найти наименование поставщика по заданному номеру телефона. Supplier sp = plist2.FindByNumber("555-2564");

Console.WriteLine(sp.Name + ": " + sp.Number);

} catch(NotFoundException) {

Console.WriteLine("He найдено");

}

// Следующее объявление недопустимо, поскольку

// класс EmailFriend НЕ наследует от класса PhoneNumber.

// PhoneList<EmailFriend> plist3 =

// new PhoneList<EmailFriend>(); // Ошибка!

}

}

Ниже приведен результат выполнения этой программы.

Гари: 555-6756 (рабочий)

Компания NetworkCity: 555-2564

Поэкспериментируйте с этой программой. В частности, попробуйте составить разные виды списков телефонных номеров или воспользоваться свойством IsWorkNumber в классе PhoneList. Вы сразу же обнаружите, что компилятор не позволит вам этого сделать, потому что свойство IsWorkNumber определено в классе Friend, а не в классе PhoneNumber, а следовательно, оно неизвестно в классе PhoneList.

Применение ограничения на интерфейс

Ограничение на интерфейс позволяет указывать интерфейс, который должен быть реализован аргументом типа. Это ограничение служит тем же основным целям, что и ограничение на базовый класс. Во-первых, оно позволяет использовать члены интерфейса в обобщенном классе. И во-вторых, оно гарантирует использование только тех аргументов типа, которые реализуют указанный интерфейс. Это означает, что для любого ограничения, накладываемого на интерфейс, аргумент типа должен обозначать сам интерфейс или же тип, реализующий этот интерфейс.

Ниже приведена общая форма наложения ограничения на интерфейс, в которой используется оператор where:

where Т : имя_интерфейса

где Г — это имя параметра типа, а имя_интерфейса — конкретное имя ограничиваемого интерфейса. В этой форме ограничения может быть указан список интерфейсов через запятую. Если ограничение накладывается одновременно на базовый класс и интерфейс, то первым в списке должен быть указан базовый класс.

Ниже приведена программа, демонстрирующая наложение ограничения на интерфейс и представляющая собой переработанный вариант предыдущего примера программы, управляющей списками телефонных номеров. В этом варианте класс PhoneNumber преобразован в интерфейс I PhoneNumber, который реализуется в классах Friend и Supplier.

// Применить ограничение на интерфейс, using System;

// Специальное исключение, генерируемое в том случае,

// если имя или номер телефона не найдены, class NotFoundException : Exception {

1 ... 134 135 136 137 138 139 140 141 142 ... 268
Перейти на страницу:
Тут вы можете бесплатно читать книгу C# 4.0 полное руководство - 2011 - Герберт Шилдт.
Комментарии