Эндрю Троелсен - ЯЗЫК ПРОГРАММИРОВАНИЯ С# 2005 И ПЛАТФОРМА .NET 2.0. 3-е издание
- Название:ЯЗЫК ПРОГРАММИРОВАНИЯ С# 2005 И ПЛАТФОРМА .NET 2.0. 3-е издание
- Автор:
- Жанр:
- Издательство:Издательский дом Вильямс
- Год:2007
- Город:Москва • Санкт-Петербург • Киев
- ISBN:ISBN 5-8459-1124-9
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Эндрю Троелсен - ЯЗЫК ПРОГРАММИРОВАНИЯ С# 2005 И ПЛАТФОРМА .NET 2.0. 3-е издание краткое содержание
В этой книге содержится описание базовых принципов функционирования платформы .NET, системы типов .NET и различных инструментальных средств разработки, используемых при создании приложений .NET. Представлены базовые возможности языка программирования C# 2005, включая новые синтаксические конструкции, появившиеся с выходом .NET 2.0, а также синтаксис и семантика языка CIL. В книге рассматривается формат сборок .NET, библиотеки базовых классов .NET. файловый ввод-вывод, возможности удаленного доступа, конструкция приложений Windows Forms, доступ к базам данных с помощью ADO.NET, создание Web-приложений ASP.NET и Web-служб XML. Книга содержит множество примеров программного кода, призванного помочь читателю в освоении предлагаемого материала. Программный код примеров можно загрузить с Web-сайта издательства.
ЯЗЫК ПРОГРАММИРОВАНИЯ С# 2005 И ПЛАТФОРМА .NET 2.0. 3-е издание - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
Здесь возникает вопрос: "Как сконфигурировать класс (или структуру), чтобы обеспечить поддержку соответствующих функциональных возможностей?" Индексатор в C# представляет собой несколько "искаженное" свойство. Для создания индексатора в самой простой форме используется синтаксис this[]. Вот как может выглядеть подходящая модификации типа Garage.
// Добавление индексатора в определение класса.
public class Garage: IEnumerable { // для каждого элемента
…
// Использование ArrayList для типов Car.
private ArrayList carArray = new ArrayList();
// Индексатор возвращает тип Car, соответствующий
// Числовому индексу.
public Car this[int pos] {
// ArrayList тоже имеет индексатор!
get { return (Car)carArray[pos]; }
set { carArray.Add(value); }
}
}
Если не обращать внимания на ключевое слово this, то объявление индексатора очень похоже на объявление свойства в C#. Но следует подчеркнуть, что индексаторы не обеспечивают иных функциональных возможностей массива, кроме возможности использования операции индексирования, Другими словами, пользователь объекта не может применить программный код, подобный следующему.
// Используется свойство ArrayList.Count? Нет!
Console.WriteLine("Машин в наличии: {0} ", carLot. Count);
Для поддержки этой функциональной возможности вы должны добавить свое свойство Count в тип Garage и, соответственно, делегат.
public class Garage: IEnumerable {
…
// Локализация/делегирование в действии снова.
public int Count { get { return carArray.Count; } }
}
Итак, индексаторы – это еще одна синтаксическая "конфетка", поскольку соответствующих функциональных возможностей можно достичь и с помощью "обычных" методов. Например, если бы тип Garage не поддерживал индексатор, все равно можно было бы позволить "внешнему миру" взаимодействовать с внутренним массивом, используя для этого именованное свойство или традиционные методы чтения и модификации данных (accessor/mutator). Но при использовании индексаторов пользовательские типы коллекции лучше согласуются со структурой библиотек базовых классов .NET.
Исходный код.Проект SimpleIndexer размещен в подкаталоге, соответствующем главе 9.
Вариации индексатора для типа Garage
В своем текущем виде тип Gаrage определяет индексатор, который позволяет вызывающей стороне идентифицировать внутренние элементы, используя число-вое значение. Но это не является непременным требованием метода индексатора. Предположим, что объекты Car содержатся в System.Collections.Specialized. ListDictionary, а не в ArrayList. Поскольку типы ListDictionary позволяют доступ к содержащимся типам с помощью ключевых маркеров (таких как, например, строки), можно создать новый индексатор Garage, подобный показанному ниже.
public class Garage: IEnumerable {
private ListDictionary carDictionary = new ListDictionarу();
// Этот индексатор возвращает соответствующий тип Car
// на основе строкового индекса.
public Car this[string name] {
get { return (Car)carDictionary[name]; }
set { carDictionary[name] = value; }
}
public int Length { get { return carDictionary.Count; } }
public IEnumerator GetEnumerator() { return carDictionary.GetEnumerator(); }
}
Вызывающая сторона теперь может взаимодействовать с машинами внутри так, как показано ниже,
public class Program {
static void Main(string[] args) {
Console:WriteLine("***** Забавы с индексаторами *****\n");
Garage carLot = new Garage();
// Добавление именованных машин в гараж.
carLot["FeeFee"] = new Car("FeeFee", 200, 0);
carLot["Clunker"] = new Car("Clunker", 90, 0);
carLot["Zippy"] = new Car("Zippy", 30, 0);
// Доступ к Zippy.
Car zippy = carLot["Zippy"];
Console.WriteLine("{0} едет со скоростью {1} км/ч", zippy.PetName, zippy.CurrSpeed);
Console.ReadLine();
}
}
Индексаторы могут быть и перегруженными. Так, чтобы позволить вызывающей стороне доступ к внутренним элементам посредством числового индекса или строковых значений, вы можете определить множество индексаторов для одного типа.
Исходный код.Проект StringIndexer размещен в подкаталоге, соответствующем главе 9.
Внутреннее представление индексаторов типов
Мы рассмотрели примеры метода индексатора в C#, и пришло время выяснить, как представляются индексаторы в терминах CIL. Если открыть числовой индексатор типа Garage, то будет видно, что компилятор C# создает свойство Item, которое сводится к подходящей паре методов get/set.
property instance class SimpleIndexer.Car Item(int32) {
.get instance class SimpleIndexer.Car SimpleIndexer.Garage::get_Item(int32)
.set instance void SimpleIndexer.Garage::set_Item(int32, class SimpleIndexer.Car)
} // end of property Garage::Item
Методы get_Item() и set_Item() будут реализованы аналогично любому другому свойству .NET, например:
method public hidebysig specialname instance сlass SimpleIndexer.Car get_Item(int32 pos) cil managed {
Code size 22 (0x16)
.maxstack 2
.locals init ([0] class SimpleIndexer.Car CSS1$0000)
IL_0000: ldarg.0
IL_0001: ldfld class [mscorlib] System.Collections.ArrayList SimpleIndexer.Garage::carArray
IL_0006: ldarg.1
IL_0007: callvirt instance object [mscorlib] Sysftem.Collections.ArrayList::get_Item(int32)
IL_000c: castclass SimpleIndexer.Car
IL_0011: stloc.0
IL_0012: br.s IL_0014
IL_0014: ldloc.0
IL_0015: ret
} // end of method Garage::get_Item
Заключительные замечания об индексаторах
Чтобы получить настоящую экзотику, вы можете создать индексатор, который имеет множество параметров. Предположим, что у нас есть пользовательская кол-лекция, которая хранит элементы в двумерном массиве. В этом случае вы можете создать метод индексатора, доказанный ниже.
public class SameContainer {
private int[,] my2DinArray = new int[10, 10];
public int this[int row, int column] {/* прочитать или установить значение 2D-массива * /}
}
В заключение следует заметить, что индексаторы могут определяться и для типа интерфейса .NET, что обеспечивает реализующим интерфейс типам возможность его настройки. Вот пример такого интерфейса.
public interface IEstablishSubObjects {
// Этот интерфейс определяет индексатор , возвращающий
// строки на основе числового индекса.
string this[int index] {get; set;}
}
Пожалуй, об индексаторах C# уже сказано достаточно. Перейдем к рассмотрению еще одного подхода, используемого в некоторых (но не во всех) языках программирования .NET: это перегрузка операций.
Перегрузка операций
В C#, как и в любом другом языке программирования, есть свой ограниченный набор лексем, используемых для выполнения базовых операций со встроенными типами. Так, вы знаете, что операция + применима к двум целым числам и в результате дает их сумму.
// Операция + с целыми числами.
Интервал:
Закладка: