Пошаговое руководство. Разработка на основе тестов с помощью тестовой Обозреватель
Создавайте модульные тесты, чтобы обеспечить правильную работу кода с помощью добавочных изменений кода. Существует несколько платформ, которые можно использовать для написания модульных тестов, в том числе разработанные третьими сторонами. Некоторые тестовые среды специализируются на тестировании на различных языках или платформах. Обозреватель тестов предоставляет единый интерфейс модульных тестов для любых таких платформ. Дополнительные сведения об использовании обозревателя тестов см. в разделе Выполнение модульных тестов с помощью обозревателя тестов и Вопросы и ответы по обозревателю тестов.
В данном пошаговом руководстве показано, как разработать тестируемый метод в C# с помощью платформы тестирования Microsoft (MSTest). Можно легко адаптировать его для других языков или других тестовых платформ, например NUnit. Дополнительные сведения см. в разделе Установка платформ модульного тестирования сторонних поставщиков.
Создание теста и создание кода
Создайте проект библиотеки классов C# для .NET или .NET Standard. Данный проект будет содержать код, который мы хотим протестировать. Назовите проект MyMath.
В том же решении добавьте новый тестовый проект MSTest для .NET.
В Visual Studio 2019 версии 16.9 имя шаблона проекта MSTest — проект модульного тестирования.
Назовите тестовый проект MathTests.
Напишите простой метод теста, который проверяет результат, полученный для конкретных входных данных. Добавьте в класс
UnitTest1
следующий код.[TestMethod] public void BasicRooterTest() { // Create an instance to test: Rooter rooter = new Rooter(); // Define a test input and output value: double expectedResult = 2.0; double input = expectedResult * expectedResult; // Run the method under test: double actualResult = rooter.SquareRoot(input); // Verify the result: Assert.AreEqual(expectedResult, actualResult, delta: expectedResult / 100); }
Создайте тип на основе кода теста.
Установите курсор на
Rooter
, а затем в меню лампочки выберите Создать тип "Rooter">Создать новый тип.В диалоговом окне Создать тип установите для параметра Проект значение MyMath, проект библиотеки классов, и нажмите OK.
Создайте метод из кода теста. Установите курсор на
SquareRoot
, а затем в меню лампочки выберите Создать метод Rooter.SquareRoot.Выполните модульный тест.
Откройте обозреватель тестов.
Чтобы открыть тестовые Обозреватель в меню "Тест", выберите "Тест Обозреватель".
Чтобы открыть Обозреватель теста в меню "Тест", выберите пункт "Тест Windows>" Обозреватель.
В обозревателе тестов выберите Запустить все, чтобы запустить тест.
Выполняется сборка решения, тест запускается и завершается ошибкой.
Выберите имя теста.
Дополнительные сведения о тесте появятся на панели Сводка теста.
Перейдите по верхней ссылке в разделе Трассировка стека, чтобы перейти к расположению, в котором произошел сбой теста.
На данном этапе создан тест и заглушка, которые будут изменены таким образом, что тест будет успешно пройден.
Проверка изменения кода
В файле Class1.cs улучшите код
SquareRoot
:public double SquareRoot(double input) { return input / 2; }
В обозревателе тестов выберите Запустить все.
Выполняется сборка решения, тест запускается и завершается успешно.
Расширение диапазона входных данных
Для уверенности, что код работает во всех случаях, добавьте тесты, которые используют более широкий диапазон входных значений.
Совет
Избегайте изменения существующих успешно выполненных тестов. Вместо этого добавьте новые тесты. Изменяйте существующие тесты только в тех случаях, когда меняются пользовательские требования. Такой подход позволяет не потерять существующие функциональные возможности при работе с расширенным кодом.
В тестовом классе добавьте следующий тест, который использует диапазон входных значений:
[TestMethod] public void RooterValueRange() { // Create an instance to test. Rooter rooter = new Rooter(); // Try a range of values. for (double expected = 1e-8; expected < 1e+8; expected *= 3.2) { RooterOneValue(rooter, expected); } } private void RooterOneValue(Rooter rooter, double expectedResult) { double input = expectedResult * expectedResult; double actualResult = rooter.SquareRoot(input); Assert.AreEqual(expectedResult, actualResult, delta: expectedResult / 1000); }
В обозревателе тестов выберите Запустить все.
Новый тест завершается неудачей (несмотря на то, что первый тест по-прежнему завершается успешно). Чтобы найти точку сбоя, выберите тест, который не пройден, и просмотрите сведения на панели Сводка теста.
Проверьте тестируемый метод, чтобы узнать, что не так. Измените код
SquareRoot
, как показано:public double SquareRoot(double input) { double result = input; double previousResult = -input; while (Math.Abs(previousResult - result) > result / 1000) { previousResult = result; result = result - (result * result - input) / (2 * result); } return result; }
В обозревателе тестов выберите Запустить все.
Теперь оба теста завершаются успешно.
Добавление тестов для исключительных случаев
Добавьте новый тест для отрицательных входных значений:
[TestMethod] public void RooterTestNegativeInput() { Rooter rooter = new Rooter(); Assert.ThrowsException<ArgumentOutOfRangeException>(() => rooter.SquareRoot(-1)); }
В обозревателе тестов выберите Запустить все.
Тестируемый метод зацикливается, его необходимо отменить вручную.
На панели инструментов обозревателя тестов нажмите Отмена.
Выполнение теста останавливается.
Исправьте код
SquareRoot
, добавив следующую инструкциюif
в начало метода:public double SquareRoot(double input) { if (input <= 0.0) { throw new ArgumentOutOfRangeException(); } ...
В обозревателе тестов выберите Запустить все.
Все тесты завершаются успешно.
Рефакторинг тестируемого кода
Выполните рефакторинг кода, но не изменяйте тесты.
Совет
Рефакторинг — это изменение, которое делает код более производительным или более понятным. Это действие не предназначено для изменения поведения кода, поэтому тесты не изменяются.
Рекомендуется выполнять рефакторинг отдельно от расширения функциональности. Неизменяемость тестов уменьшает шансы случайных ошибок во время рефакторинга.
Измените строку, которая вычисляет
result
в методеSquareRoot
, следующим образом:public double SquareRoot(double input) { if (input <= 0.0) { throw new ArgumentOutOfRangeException(); } double result = input; double previousResult = -input; while (Math.Abs(previousResult - result) > result / 1000) { previousResult = result; result = (result + input / result) / 2; //was: result = result - (result * result - input) / (2*result); } return result; }
Выберите Выполнить все и убедитесь, что все тесты по-прежнему завершаются успехом.