【发布时间】:2021-01-19 20:45:00
【问题描述】:
有一种情况,我想检查一个内部类方法被调用了多少次。我有一个敏感的云任务,必须根据某些情况进行计数。我想通过一个单元测试来增强我的应用程序,以断言特定函数被调用的次数。
要在更简单的场景中这样做,我想在以下脚本中进行测试:
class HMT:
def __init__(self):
self.buildedList = []
def handle(self, string_to_explode: str):
for exploded_part in string_to_explode.split(","):
self.doCoolThings(exploded_part)
def doCoolThings(self, fetched_raw_string: str):
self.buildedList.append("Cool thing done: " + fetched_raw_string)
根据我传递给handle 函数的字符串,doCoolThings 将被调用 N 次(在这种简单的情况下,仅取决于字符串中逗号的数量)。
我可以通过计算builderList 中结果元素的数量来进行测试:
import unittest
from HMT import HMT
class test_HMT(unittest.TestCase):
def setUp(self):
self.hmt = HMT()
def test_hmt_3(self):
string_to_test = "alpha,beta,gamma"
self.hmt.handle(string_to_test)
self.assertEqual(3, len(self.hmt.buildedList))
def test_hmt_2(self):
string_to_test = "delta,epsilon"
self.hmt.handle(string_to_test)
self.assertEqual(2, len(self.hmt.buildedList))
但在实际场景中,不会有一个可用的公共类列表始终将其元素数量与函数doCoolThings 被调用的时间相匹配。
那么,我如何检查doCoolThings 被调用了多少次而不需要检查列表元素计数?
我知道我可以在每次调用 doCoolThings 时增加的类中放置一个计数器,然后将其暴露在外部以供以后检查。但我不想弄乱与我的业务规则不直接相关的代码放置行。
@jarmod 评论后,我进入了这段代码的版本:
def mydecorator(func):
def wrapped(*args, **kwargs):
wrapped.calls += 1
return func(*args, **kwargs)
wrapped.calls = 0
return wrapped
class HMT:
def __init__(self):
self.buildedList = []
def handle(self, string_to_explode: str):
for exploded_part in string_to_explode.split(","):
self.doCoolThings(exploded_part)
@mydecorator
def doCoolThings(self, fetched_raw_string: str, *args, **kwargs):
self.buildedList.append("Cool thing done: " + fetched_raw_string)
还有测试:
import unittest
from HMT import HMT
class test_HMT(unittest.TestCase):
def test_hmt_3_dec(self):
hmt = HMT()
string_to_test = "epsilon,ota,eta"
hmt.handle(string_to_test)
self.assertEqual(3, hmt.doCoolThings.calls)
def test_hmt_3(self):
hmt = HMT()
string_to_test = "alpha,beta,gamma"
hmt.handle(string_to_test)
self.assertEqual(3, len(hmt.buildedList))
但仍然无法正常工作。当我运行测试时,我会收到:
.F
======================================================================
FAIL: test_hmt_3_dec (myTest.test_HMT)
----------------------------------------------------------------------
Traceback (most recent call last):
File "D:\Users\danil\tmp\myDec\myTest.py", line 10, in test_hmt_3_dec
self.assertEqual(3, hmt.doCoolThings.calls)
AssertionError: 3 != 6
----------------------------------------------------------------------
Ran 2 tests in 0.001s
FAILED (failures=1)
更多测试表明测试运行了两次,但计数器在新的实例化中也没有重置。
无论如何,最初的想法是在内部类方法中动态放置一个观察者,并在每次触发这些方法时从外部获取(不过,似乎不是使用装饰器解决的问题)。
非常感谢您的回答(如果有人知道如何重置装饰器中的计数器,我也将不胜感激)。
【问题讨论】:
-
也许您可以使用“计数”装饰器来限制对基本代码的修改。示例:stackoverflow.com/questions/21716940/…
标签: python unit-testing class methods assert