Я создавал библиотеку расширений, и я использовал отличный метод расширения, найденный в http://www.extensionmethod.net для включения. В моем модульном тесте (с использованием NUnit 1.5.2) я столкнулся с интересной проблемой. Во-первых, давайте посмотрим на код:KeyNotFoundException, но не при отладке
/// <summary>
/// Groups and aggregates the sequence of elements.
/// </summary>
/// <typeparam name="TSource">The source type in the sequence.</typeparam>
/// <typeparam name="TFirstKey">The first key type to group by.</typeparam>
/// <typeparam name="TSecondKey">The second key type to rotate by.</typeparam>
/// <typeparam name="TValue">The type of value that will be aggregated.</typeparam>
/// <param name="source">The source sequence.</param>
/// <param name="firstKeySelector">The first key selector.</param>
/// <param name="secondKeySelector">The second key selector.</param>
/// <param name="aggregator">The aggregating function.</param>
/// <returns>A <see cref="Dictionary{TKey,TValue}" /> representing the pivoted data.</returns>
public static Dictionary<TFirstKey, Dictionary<TSecondKey, TValue>> Pivot<TSource, TFirstKey, TSecondKey, TValue>
(this IEnumerable<TSource> source,
Func<TSource, TFirstKey> firstKeySelector,
Func<TSource, TSecondKey> secondKeySelector,
Func<IEnumerable<TSource>, TValue> aggregator)
{
return source.GroupBy(firstKeySelector).Select(
x => new
{
X = x.Key,
Y = x.GroupBy(secondKeySelector).Select(
z => new { Z = z.Key, V = aggregator(z) }).ToDictionary(e => e.Z, o => o.V)
}).ToDictionary(e => e.X, o => o.Y);
}
Что делает функция, это занимает в IEnumerable типа TSource, и вращает элементы в словарь, и агрегирует элементы, используя любые функции можно определить. Мой набор данных - это массив людей (в типе Person).
private static readonly Person[] people =
new[]
{
new Person { Forename = "Matt", Surname = "Someone", Email = "[email protected]", Age = 25, IsMale = true },
new Person { Forename = "Chris", Surname = "Someone", Email = "[email protected]", Age = 28, IsMale = false },
new Person { Forename = "Andy", Surname = "Someone", Email = "[email protected]", Age = 30, IsMale = true },
new Person { Forename = "Joel", Surname = "Someone", Email = "[email protected]", Age = 30, IsMale = true },
new Person { Forename = "Paul", Surname = "Someone", Email = "[email protected]", Age = 30, IsMale = true }
};
И, наконец, мы делаем наш тест:
/// <summary>
/// Performs a pivot function on the sample array.
/// </summary>
[Test]
public void Pivot()
{
/* Our sample data is an array of Person instances.
* Let's organise it first by gender (IsMale), and then by Age.
* Finally, we'll return a count. */
var organised = people.Pivot(p => p.IsMale, p => p.Age, l => l.Count());
Assert.IsTrue(organised.Count == 2, "More than two genders were returned.");
Assert.IsTrue(organised[true].Count == 2, "More than two ages were returned for males.");
Assert.IsTrue(organised[false].Count == 1, "More than 1 age was returned for females.");
int count = organised[true][30];
Assert.IsTrue(count == 3, "There are more than 3 male 30 year olds in our data.");
}
Что возвращается в этом тесте, является словарь> экземпляр. Логическое значение является результатом группы IsMale, и в наших примерных данных правильно возвращает 2 элемента: true и false. Внутренний словарь имеет ключ от возраста и значение счетчика. В наших тестовых данных организованный [true] [30] отражает всех мужчин в возрасте 30 лет в наборе.
Проблема заключается не в самой функции поворота, а по какой-то причине, когда мы запускаем ее как с помощью NUnit Test Runner, так и с помощью тестировщика тестов Resharper, тест завершается с ошибкой, сообщая об исключении KeyNotFoundException для строки «int count = организованный [истинно] [30];». Когда мы отлаживаем этот тест, он правильно возвращает значение 3 (как в наших данных образца, у нас 3 мужчины в возрасте 30 лет).
Любые мысли?
Я провел NUnit извне, но использовал Resharper Unit Test Runner внутренне от VS. Оба не работают без отладки, но работают, но работают очень хорошо. –