Коллекции небольших размеров из C5 Generic Collection Library сравнительно медленные - можно ли что-нибудь сделать?
Недавно я тестировал коллекции C5 на C# и мне нравится их функциональность. Для больших коллекций производительность кажется на уровне общих аналогов. Однако для небольших коллекций они значительно медленнее. Я подозреваю, что резкое ухудшение относительной скорости происходит из-за операций с постоянным временем, выполняемых коллекциями C5. Я знаю одну операцию - запуск событий. Может ли это быть причиной плохой работы небольших коллекций? Можно ли исправить это, отключив некоторые функции? Вот тест производительности:
//Two containers to be tested. 'Test' is a wrapper over decimal.
var arrayList = new C5.ArrayList<Test>();
var genericList = new System.Collections.Generic.List<Test>();
var toBeAdded = new List<Test>();
var watch = new Stopwatch();
//Fill both tested containers
for (int i = 10; i > 0; i--)
{
var test = new Test(i);
genericList.Add(test);
arrayList.Add(test);
}
//Fill the container the range of which will be inserted to the tested containers
for (int i = 5; i > 0; i--)
{
toBeAdded.Add(new Test(i+0.5m));
}
//Test the speed of adding a range of values and sorting for both containers
watch.Start();
genericList.AddRange(toBeAdded);
Console.WriteLine("Adding values for generic list: {0} ticks", watch.ElapsedTicks);
watch.Restart();
genericList.Sort();
Console.WriteLine("Sorting for generic list: {0} ticks", watch.ElapsedTicks);
watch.Restart();
arrayList.AddAll(toBeAdded);
Console.WriteLine("Adding values for C5 ArrayList: {0} ticks", watch.ElapsedTicks);
watch.Restart();
arrayList.Sort();
Console.WriteLine("Sorting for C5 ArrayList: {0} ticks", watch.ElapsedTicks);
и тестовый класс:
class Test : IComparable
{
private decimal _number;
internal Test(decimal aNumber)
{
_number = aNumber;
}
public int CompareTo(object obj)
{
var test = (Test) obj;
return _number.CompareTo(test._number);
}
}
Выход:
Adding values for generic list: 56 ticks
Sorting for generic list: 770 ticks
Adding values for C5 ArrayList: 3575 ticks
Sorting for C5 ArrayList: 4815 ticks
И C5, и тест являются сборками Release. Соотношение скоростей около 60x для вставки и 6x для сортировки согласовано между тестовыми прогонами.
РЕДАКТИРОВАТЬ: вышеуказанный тест был запущен из VS. Результаты для запуска вне VS:
Adding values for generic list: 54 ticks
Sorting for generic list: 2135 ticks
Adding values for C5 ArrayList: 5765 ticks
Sorting for C5 ArrayList: 5198 ticks
Опять же, соотношение скоростей около 100x для вставки и 2x для сортировки согласовано между запусками теста.
Мой проект включает в себя множество манипуляций с небольшими контейнерами, и их производительность имеет первостепенное значение. Функциональность контейнеров C5 великолепна, и я хотел бы использовать их, но в настоящее время не могу по соображениям производительности. Буду признателен за понимание этого вопроса.
РЕДАКТИРОВАТЬ 2: Согласно ответу Iridium, я выполнил тест в цикле (поместив всю логику, включая создание контейнера в цикл, чтобы исключить любые приемы оптимизации компилятора), отбросил первые два результата и усреднил последующие 1000 результатов. Вот они:
Adding values for generic list: 1.09 ticks
Sorting for generic list: 14.07 ticks
Adding values for C5 ArrayList: 1.92 ticks
Sorting for C5 ArrayList: 13.69 ticks
Теперь вставка C5 происходит на 76% медленнее, а сортировка выполняется наравне со списком. Это достаточно хорошо для моей цели. Я принимаю ответ Иридиум. Тем не менее, если у кого-то есть понимание медленной вставки, пожалуйста, поделитесь им. Спасибо всем за помощь.
1 ответ
Мне интересно, если ваши результаты немного вводят в заблуждение, и что различия, которые вы видите для C5, на самом деле (возможно) связаны с накладными расходами при загрузке сборки /JIT-компиляции при первом использовании метода add/sort. Родовые коллекции, не страдающие от этого в результате того, что системные сборки уже были загружены /NGen'ed.
Я повторил ваш тест, используя C5 2.1.4596.30350, но запустил весь тест несколько раз (без перезапуска приложения, чтобы избежать однократных накладных расходов). Результаты, по-видимому, предполагают, что при первом использовании коллекций C5 существует временное наказание (в соответствии с одноразовыми накладными расходами, такими как JIT-компиляция), которое исчезает при последующих использованиях, оставляя производительность C5 практически такой же, как и у общих коллекций.