Тестирование потоковых приложений
В этой главе мы узнаем о тестировании потоковых приложений. Мы также узнаем о важности тестирования.
Зачем тестировать?
Прежде чем мы углубимся в обсуждение важности тестирования, нам нужно знать, что такое тестирование. В общих чертах, тестирование - это метод определения того, насколько хорошо что-то работает. С другой стороны, особенно если мы говорим о компьютерных программах или программном обеспечении, то тестирование - это метод доступа к функциональности программного обеспечения.
В этом разделе мы обсудим важность тестирования программного обеспечения. При разработке программного обеспечения перед выпуском программного обеспечения для клиента необходимо выполнить двойную проверку. Поэтому очень важно протестировать программу опытной командой тестировщиков. Рассмотрим следующие моменты, чтобы понять важность тестирования программного обеспечения:
Повышение качества программного обеспечения
Конечно, ни одна компания не хочет поставлять некачественное программное обеспечение, и ни один клиент не хочет покупать некачественное программное обеспечение. Тестирование улучшает качество программного обеспечения за счет обнаружения и исправления в нем ошибок.
Удовлетворенность клиентов
Самая важная часть любого бизнеса - это удовлетворение потребностей клиентов. Предоставляя программное обеспечение хорошего качества без ошибок, компании могут удовлетворить потребности клиентов.
Уменьшите влияние новых функций
Предположим, мы создали программную систему из 10000 строк, и нам нужно добавить новую функцию, тогда команда разработчиков будет беспокоиться о влиянии этой новой функции на все программное обеспечение. Здесь также тестирование играет жизненно важную роль, потому что, если команда тестирования создала хороший набор тестов, это может спасти нас от любых потенциальных катастрофических сбоев.
Пользовательский опыт
Еще одна наиболее важная часть любого бизнеса - это опыт пользователей этого продукта. Только тестирование может гарантировать, что конечный пользователь сочтет использование продукта простым и легким.
Сокращение расходов
Тестирование может снизить общую стоимость программного обеспечения за счет обнаружения и исправления ошибок на этапе тестирования его разработки, а не исправления после поставки. Если после поставки программного обеспечения обнаружится серьезная ошибка, это увеличит его материальную стоимость, например, с точки зрения затрат и нематериальных затрат, например, с точки зрения неудовлетворенности клиентов, отрицательной репутации компании и т. Д.
Что тестировать?
Всегда рекомендуется иметь соответствующие знания о том, что нужно тестировать. В этом разделе мы сначала поймем, что является основным мотивом тестировщика при тестировании любого программного обеспечения. Следует избегать покрытия кода, т. Е. Того, сколько строк кода попадает в наш набор тестов во время тестирования. Это потому, что во время тестирования сосредоточение внимания только на количестве строк кода не добавляет реальной ценности нашей системе. Могут остаться некоторые ошибки, которые появятся позже, на более позднем этапе, даже после развертывания.
Примите во внимание следующие важные моменты, связанные с тем, что тестировать:
Нам нужно сосредоточиться на тестировании функциональности кода, а не покрытия кода.
Нам нужно сначала протестировать наиболее важные части кода, а затем перейти к менее важным частям кода. Это точно сэкономит время.
У тестера должно быть множество различных тестов, которые могут подтолкнуть программное обеспечение к его предельным возможностям.
Подходы к тестированию параллельных программ
Благодаря способности использовать истинные возможности многоядерной архитектуры, параллельные программные системы заменяют последовательные системы. В последнее время параллельные системные программы используются во всем: от мобильных телефонов до стиральных машин, от автомобилей до самолетов и т.д. уже ошибка, тогда мы получим несколько ошибок.
Методы тестирования параллельных программных продуктов в значительной степени сосредоточены на выборе чередования, которое выявляет потенциально опасные шаблоны, такие как состояния гонки, взаимоблокировки и нарушение атомарности. Ниже приведены два подхода к тестированию параллельных программ.
Систематическое исследование
Этот подход направлен на максимально широкое исследование пространства перемежений. Такие подходы могут использовать технику грубой силы, а другие используют технику редукции частичного порядка или эвристическую технику для исследования пространства перемежений.
На основе собственности
Подходы, основанные на свойствах, основаны на наблюдении, что сбои параллелизма более вероятны при чередовании, которое раскрывает определенные свойства, такие как подозрительный шаблон доступа к памяти. Различные подходы, основанные на свойствах, нацелены на разные ошибки, такие как состояния гонки, взаимоблокировки и нарушение атомарности, что дополнительно зависит от тех или иных конкретных свойств.
Стратегии тестирования
Стратегия тестирования также известна как тестовый подход. Стратегия определяет, как будет проводиться тестирование. Подход к тестированию имеет две техники -
Проактивный
Подход, при котором процесс разработки теста запускается как можно раньше, чтобы найти и исправить дефекты до создания сборки.
Реактивный
Подход, при котором тестирование не начинается до завершения процесса разработки.
Прежде чем применять любую стратегию тестирования или подход к программе на Python, мы должны иметь базовое представление о типах ошибок, которые может иметь программа. Ошибки следующие -
Синтаксические ошибки
Во время разработки программы может быть много мелких ошибок. Ошибки в основном связаны с опечатками. Например, отсутствие двоеточия или неправильное написание ключевого слова и т. Д. Такие ошибки возникают из-за ошибки в синтаксисе программы, а не в логике. Следовательно, эти ошибки называются синтаксическими ошибками.
Семантические ошибки
Семантические ошибки также называются логическими ошибками. Если в программе есть логическая или семантическая ошибка, оператор будет компилироваться и работать правильно, но он не даст желаемого результата, потому что логика неверна.
Модульное тестирование
Это одна из наиболее часто используемых стратегий тестирования программ на Python. Эта стратегия используется для тестирования модулей или компонентов кода. Под модулями или компонентами мы подразумеваем классы или функции кода. Модульное тестирование упрощает тестирование больших программных систем путем тестирования «маленьких» модулей. С помощью вышеупомянутой концепции модульное тестирование можно определить как метод, при котором отдельные единицы исходного кода тестируются, чтобы определить, возвращают ли они желаемый результат.
В наших последующих разделах мы узнаем о различных модулях Python для модульного тестирования.
модуль unittest
Самый первый модуль для модульного тестирования - это модуль unittest. Он основан на JUnit и по умолчанию включен в Python3.6. Он поддерживает автоматизацию тестирования, совместное использование кода настройки и выключения для тестов, объединение тестов в коллекции и независимость тестов от структуры отчетности.
Ниже приведены несколько важных концепций, поддерживаемых модулем unittest.
Текстовое приспособление
Он используется для настройки теста, чтобы его можно было запустить до начала теста и разорвать после его завершения. Это может включать создание временной базы данных, каталогов и т.д., необходимых перед запуском теста.
Прецедент
Тестовый пример проверяет, исходит ли требуемый ответ от определенного набора входных данных или нет. Модуль unittest включает базовый класс TestCase, который можно использовать для создания новых тестовых случаев. По умолчанию он включает два метода -
setUp()- метод крюка для установки испытательного приспособления перед испытанием. Это вызывается перед вызовом реализованных методов тестирования.
tearDown( - метод перехвата для деконструирования фикстуры класса после выполнения всех тестов в классе.
Тестирование
Это набор тестовых наборов, тестовых примеров или того и другого.
Тестовый бегун
Он контролирует выполнение тестовых случаев или костюмов и предоставляет результат пользователю. Он может использовать графический интерфейс или простой текстовый интерфейс для предоставления результата.
Example
Следующая программа Python использует модуль unittest для тестирования модуля с именем Fibonacci. Программа помогает в вычислении ряда Фибоначчи числа. В этом примере мы создали класс с именем Fibo_test для определения тестовых случаев с использованием различных методов. Эти методы унаследованы от unittest.TestCase. По умолчанию мы используем два метода - setUp () и tearDown (). Мы также определяем метод testfibocal. Название теста должно начинаться с буквы test. В последнем блоке unittest.main () предоставляет интерфейс командной строки для тестового сценария.
import unittest
def fibonacci(n):
a, b = 0, 1
for i in range(n):
a, b = b, a + b
return a
class Fibo_Test(unittest.TestCase):
def setUp(self):
print("This is run before our tests would be executed")
def tearDown(self):
print("This is run after the completion of execution of our tests")
def testfibocal(self):
self.assertEqual(fib(0), 0)
self.assertEqual(fib(1), 1)
self.assertEqual(fib(5), 5)
self.assertEqual(fib(10), 55)
self.assertEqual(fib(20), 6765)
if __name__ == "__main__":
unittest.main()
При запуске из командной строки приведенный выше сценарий производит вывод, который выглядит следующим образом:
Выход
This runs before our tests would be executed.
This runs after the completion of execution of our tests.
.
----------------------------------------------------------------------
Ran 1 test in 0.006s
OK
Теперь, чтобы было понятнее, мы изменяем наш код, который помог в определении модуля Фибоначчи.
Рассмотрим следующий блок кода в качестве примера -
def fibonacci(n):
a, b = 0, 1
for i in range(n):
a, b = b, a + b
return a
Несколько изменений в блок кода внесены, как показано ниже -
def fibonacci(n):
a, b = 1, 1
for i in range(n):
a, b = b, a + b
return a
Теперь, после запуска скрипта с измененным кодом, мы получим следующий вывод:
This runs before our tests would be executed.
This runs after the completion of execution of our tests.
F
======================================================================
FAIL: testCalculation (__main__.Fibo_Test)
----------------------------------------------------------------------
Traceback (most recent call last):
File "unitg.py", line 15, in testCalculation
self.assertEqual(fib(0), 0)
AssertionError: 1 != 0
----------------------------------------------------------------------
Ran 1 test in 0.007s
FAILED (failures = 1)
Приведенный выше вывод показывает, что модуль не смог выдать желаемый результат.
Модуль доктеста
Модуль docktest также помогает в модульном тестировании. Он также поставляется с предварительно упакованным питоном. Его проще использовать, чем модуль unittest. Модуль unittest больше подходит для сложных тестов. Для использования модуля doctest нам необходимо его импортировать. Строка документации соответствующей функции должна иметь интерактивный сеанс Python вместе с их выходными данными.
Если в нашем коде все в порядке, то модуль docktest не будет выводить данные; в противном случае он предоставит результат.
пример
В следующем примере Python используется модуль docktest для тестирования модуля с именем Fibonacci, который помогает в вычислении ряда Фибоначчи числа.
import doctest
def fibonacci(n):
"""
Calculates the Fibonacci number
>>> fibonacci(0)
0
>>> fibonacci(1)
1
>>> fibonacci(10)
55
>>> fibonacci(20)
6765
>>>
"""
a, b = 1, 1
for i in range(n):
a, b = b, a + b
return a
if __name__ == "__main__":
doctest.testmod()
Мы видим, что в строке документации соответствующей функции с именем fib был интерактивный сеанс Python вместе с выходными данными. Если наш код в порядке, то модуль doctest не будет выводить данные. Но чтобы увидеть, как это работает, мы можем запустить его с параметром –v.
(base) D:\ProgramData>python dock_test.py -v
Trying:
fibonacci(0)
Expecting:
0
ok
Trying:
fibonacci(1)
Expecting:
1
ok
Trying:
fibonacci(10)
Expecting:
55
ok
Trying:
fibonacci(20)
Expecting:
6765
ok
1 items had no tests:
__main__
1 items passed all tests:
4 tests in __main__.fibonacci
4 tests in 2 items.
4 passed and 0 failed.
Test passed.
Теперь мы изменим код, который помог в определении модуля Фибоначчи.
Рассмотрим следующий блок кода в качестве примера -
def fibonacci(n):
a, b = 0, 1
for i in range(n):
a, b = b, a + b
return a
Следующий блок кода помогает с изменениями -
def fibonacci(n):
a, b = 1, 1
for i in range(n):
a, b = b, a + b
return a
После запуска сценария даже без опции –v с измененным кодом мы получим результат, как показано ниже.
Выход
(base) D:\ProgramData>python dock_test.py
**********************************************************************
File "unitg.py", line 6, in __main__.fibonacci
Failed example:
fibonacci(0)
Expected:
0
Got:
1
**********************************************************************
File "unitg.py", line 10, in __main__.fibonacci
Failed example:
fibonacci(10)
Expected:
55
Got:
89
**********************************************************************
File "unitg.py", line 12, in __main__.fibonacci
Failed example:
fibonacci(20)
Expected:
6765
Got:
10946
**********************************************************************
1 items had failures:
3 of 4 in __main__.fibonacci
***Test Failed*** 3 failures.
Как видно из вышеприведенного вывода, три теста не прошли.