- •А.А. Волосевич
- •Содержание
- •1. Работа с числами
- •2. Представление даты и времени
- •3. Работа со строками и текстом
- •4. Преобразование информации
- •5. Сравнение для выяснения равенства
- •6. Сравнение для выяснения порядка
- •7. Жизненный цикл объектов
- •7.1. Алгоритм сборки мусора
- •7.2. Финализаторы и интерфейс iDisposable
- •7.3. Слабые ссылки
- •8. Перечислители и итераторы
- •9. Стандартные интерфейсы коллекций
- •10. Массивы и класс System.Array
- •11. Типы для работы с коллекциями-списками
- •12. Типы для работы с коллекциями-множествами
- •13. Типы для работы с коллекциями-словарями
- •14. Типы для создания пользовательских коллекций
- •15. Технология linq to Objects
- •1. Оператор условия Where().
- •2. Операторы проекций.
- •3. Операторы упорядочивания.
- •4. Оператор группировки GroupBy().
- •5. Операторы соединения.
- •6. Операторы работы с множествами.
- •7. Операторы агрегирования.
- •8. Операторы генерирования.
- •9. Операторы кванторов и сравнения.
- •10. Операторы разбиения.
- •11. Операторы элемента.
- •12. Операторы преобразования.
- •16. Работа с объектами файловой системы
- •17. Ввод и вывод информации
- •17.1. Потоки данных и декораторы потоков
- •2. Классы для работы с потоками, связанными с хранилищами.
- •3. Декораторы потоков.
- •4. Адаптеры потоков.
- •17.2. Адаптеры потоков
- •18. Основы xml
- •19. Технология linq to xml
- •20. Дополнительные возможности обработки xml
- •21. Сериализация времени выполнения
- •22. Контракты данных и xml-сериализация
- •23. Состав и взаимодействие сборок
- •24. Метаданные и получение информации о типах
- •25. Позднее связывание и кодогенерация
- •26. Атрибуты
- •27. Динамическое связывание
- •28. Файлы конфигурации
- •29. Диагностика и мониторинг
- •30. Процессы и домены
- •31. Основы многопоточного программирования
- •32. Синхронизация потоков
- •32.1. Критические секции
- •32.2. Синхронизация на основе подачи сигналов
- •32.3. Неблокирующие средства синхронизации
- •32.4. Разделение данных между потоками
- •33. Библиотека параллельных задач
- •33.1. Параллелизм на уровне задач
- •33.2. Параллелизм при императивной обработке данных
- •33.3. Параллелизм при декларативной обработке данных
- •33.4. Обработка исключений и отмена выполнения задач
- •33.5. Коллекции, поддерживающие параллелизм
- •34. Асинхронный вызов методов
- •Литература
6. Операторы работы с множествами.
В LINQ to Objects имеется набор операторов для работы с множествами.
IEnumerable<T> Distinct<T>(this IEnumerable<T> source);
IEnumerable<T> Union<T>(this IEnumerable<T> first,
IEnumerable<T> second);
IEnumerable<T> Intersect<T>(this IEnumerable<T> first,
IEnumerable<T> second);
IEnumerable<T> Except<T>(this IEnumerable<T> first,
IEnumerable<T> second);
Оператор Distinct() удаляет из коллекции повторяющиеся элементы. Операторы Union(), Intersect() и Except() представляют объединение, пересечение и разность двух множеств.
7. Операторы агрегирования.
К операторам агрегирования относятся операторы, результатом работы которых является скалярное значение. Следующие операторы возвращают количество элементов коллекции. При этом может быть использована перегруженная версия, принимающая в качестве второго аргумента предикат фильтрации.
int Count<T>(this IEnumerable<T> source);
long LongCount<T>(this IEnumerable<T> source);
Следующие операторы подсчитывают сумму и среднее значение в коллекции. При этом Num должен быть типом int, long, float, double, decimal или вариантом этих типов с поддержкой null (например, long?).
Num Sum(this IEnumerable<Num> source);
Num Sum<T>(this IEnumerable<T> source, Func<T, Num> selector);
Num Average(this IEnumerable<Num> source);
Num Average<T>(this IEnumerable<T> source, Func<T, Num> selector);
Существует также несколько перегруженных версий операторов для нахождения минимального и максимального значений. Первые две версии применяются для коллекций с числовым элементом. Последние две предполагают, что элемент коллекции реализует интерфейс IComparable<T>.
Num Min/Max(this IEnumerable<Num> source);
Num Min<T>/Max<T>(this IEnumerable<T> src, Func<T, Num> selector);
T Min<T>/Max<T>(this IEnumerable<T> source);
S Min<T, S>/Max<T, S>(this IEnumerable<T> src, Func<T, S> selector);
Оператор Aggregate() позволяет выполнить для коллекции собственный алгоритм агрегирования. Его простейшая форма:
T Aggregate<T>(this IEnumerable<T> source, Func<T, T, T> func);
Функция func принимает два аргумента: значение-аккумулятор и текущее значение коллекции. Результат функции перезаписывается в аккумулятор. В следующем примере оператор Aggregate() применяется для обращения строки:
var text = "The quick brown fox jumps over the lazy dog";
string[] words = text.Split(' ');
var reversed = words.Aggregate((acc, next) => next + " " + acc);
8. Операторы генерирования.
Эта группа операторов позволяет создать набор данных. Первый оператор группы – оператор Range(). Он просто выдаёт указанное количество подряд идущих целых чисел, начиная с заданного значения.
IEnumerable<int> Range(int start, int count);
Продемонстрируем использование Range() в задаче поиска простых чисел, которую решим при помощи LINQ:
var primes = Enumerable.Range(2, 999)
.Where(x => !Enumerable.Range(2, (int) Math.Sqrt(x))
.Any(y => x != y && x%y == 0));
foreach (var prime in primes)
{
Console.WriteLine(prime);
}
Следующий оператор генерирования – оператор Repeat(). Он создаёт коллекцию, в которой указанный элемент повторяется требуемое число раз. Для ссылочных типов дублируются ссылки, а не содержимое.
IEnumerable<T> Repeat<T>(T element, int count);
Покажем не совсем стандартное применение Repeat() для генерирования последовательности случайных чисел:
Random rnd = new Random();
var r1 = Enumerable.Repeat(0, 20).Select(i => rnd.Next(0, 40));
Последним генерирующим оператором является оператор Empty(), который порождает пустое перечисление определённого типа.
IEnumerable<T> Empty<T>();