Поделиться через


Пошаговое руководство. Разработка на основе тестов с помощью тестовой Обозреватель

Создавайте модульные тесты, чтобы обеспечить правильную работу кода с помощью добавочных изменений кода. Существует несколько платформ, которые можно использовать для написания модульных тестов, в том числе разработанные третьими сторонами. Некоторые тестовые среды специализируются на тестировании на различных языках или платформах. Обозреватель тестов предоставляет единый интерфейс модульных тестов для любых таких платформ. Дополнительные сведения об использовании обозревателя тестов см. в разделе Выполнение модульных тестов с помощью обозревателя тестов и Вопросы и ответы по обозревателю тестов.

В данном пошаговом руководстве показано, как разработать тестируемый метод в C# с помощью платформы тестирования Microsoft (MSTest). Можно легко адаптировать его для других языков или других тестовых платформ, например NUnit. Дополнительные сведения см. в разделе Установка платформ модульного тестирования сторонних поставщиков.

Создание теста и создание кода

  1. Создайте проект библиотеки классов C# для .NET или .NET Standard. Данный проект будет содержать код, который мы хотим протестировать. Назовите проект MyMath.

  2. В том же решении добавьте новый тестовый проект MSTest для .NET.

    В Visual Studio 2019 версии 16.9 имя шаблона проекта MSTest — проект модульного тестирования.

    Назовите тестовый проект MathTests.

    New code and test projects

    New code and test projects

  3. Напишите простой метод теста, который проверяет результат, полученный для конкретных входных данных. Добавьте в класс 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);
    }
    
  4. Создайте тип на основе кода теста.

    1. Установите курсор на Rooter, а затем в меню лампочки выберите Создать тип "Rooter">Создать новый тип.

      Generate new type quick action

      Generate new type quick action

    2. В диалоговом окне Создать тип установите для параметра Проект значение MyMath, проект библиотеки классов, и нажмите OK.

      Generate Type dialog box in Visual Studio 2019

      Generate Type dialog box in Visual Studio 2019

  5. Создайте метод из кода теста. Установите курсор на SquareRoot, а затем в меню лампочки выберите Создать метод Rooter.SquareRoot.

  6. Выполните модульный тест.

    1. Откройте обозреватель тестов.

      Чтобы открыть тестовые Обозреватель в меню "Тест", выберите "Тест Обозреватель".

      Чтобы открыть Обозреватель теста в меню "Тест", выберите пункт "Тест Windows>" Обозреватель.

    2. В обозревателе тестов выберите Запустить все, чтобы запустить тест.

    Выполняется сборка решения, тест запускается и завершается ошибкой.

  7. Выберите имя теста.

    Дополнительные сведения о тесте появятся на панели Сводка теста.

    Test Detail Summary in Test Explorer

    Test Detail Summary in Test Explorer

  8. Перейдите по верхней ссылке в разделе Трассировка стека, чтобы перейти к расположению, в котором произошел сбой теста.

На данном этапе создан тест и заглушка, которые будут изменены таким образом, что тест будет успешно пройден.

Проверка изменения кода

  1. В файле Class1.cs улучшите код SquareRoot:

    public double SquareRoot(double input)
    {
        return input / 2;
    }
    
  2. В обозревателе тестов выберите Запустить все.

    Выполняется сборка решения, тест запускается и завершается успешно.

    Test Explorer showing a passing test

    Test Explorer showing a passing test

Расширение диапазона входных данных

Для уверенности, что код работает во всех случаях, добавьте тесты, которые используют более широкий диапазон входных значений.

Совет

Избегайте изменения существующих успешно выполненных тестов. Вместо этого добавьте новые тесты. Изменяйте существующие тесты только в тех случаях, когда меняются пользовательские требования. Такой подход позволяет не потерять существующие функциональные возможности при работе с расширенным кодом.

  1. В тестовом классе добавьте следующий тест, который использует диапазон входных значений:

    [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);
    }
    
  2. В обозревателе тестов выберите Запустить все.

    Новый тест завершается неудачей (несмотря на то, что первый тест по-прежнему завершается успешно). Чтобы найти точку сбоя, выберите тест, который не пройден, и просмотрите сведения на панели Сводка теста.

  3. Проверьте тестируемый метод, чтобы узнать, что не так. Измените код 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;
    }
    
  4. В обозревателе тестов выберите Запустить все.

    Теперь оба теста завершаются успешно.

Добавление тестов для исключительных случаев

  1. Добавьте новый тест для отрицательных входных значений:

    [TestMethod]
    public void RooterTestNegativeInput()
    {
        Rooter rooter = new Rooter();
        Assert.ThrowsException<ArgumentOutOfRangeException>(() => rooter.SquareRoot(-1));
    }
    
  2. В обозревателе тестов выберите Запустить все.

    Тестируемый метод зацикливается, его необходимо отменить вручную.

  3. На панели инструментов обозревателя тестов нажмите Отмена.

    Выполнение теста останавливается.

  4. Исправьте код SquareRoot, добавив следующую инструкцию if в начало метода:

    public double SquareRoot(double input)
    {
        if (input <= 0.0)
        {
            throw new ArgumentOutOfRangeException();
        }
        ...
    
  5. В обозревателе тестов выберите Запустить все.

    Все тесты завершаются успешно.

Рефакторинг тестируемого кода

Выполните рефакторинг кода, но не изменяйте тесты.

Совет

Рефакторинг — это изменение, которое делает код более производительным или более понятным. Это действие не предназначено для изменения поведения кода, поэтому тесты не изменяются.

Рекомендуется выполнять рефакторинг отдельно от расширения функциональности. Неизменяемость тестов уменьшает шансы случайных ошибок во время рефакторинга.

  1. Измените строку, которая вычисляет 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;
    }
    
  2. Выберите Выполнить все и убедитесь, что все тесты по-прежнему завершаются успехом.

    Test Explorer showing 3 passed tests

    Test Explorer showing 3 passed tests