0

请注意:这是在W10上。这可能很重要。

Python:3.9.4 pytest:6.2.5 pytest-qt:4.0.2

我已经使用 pytest-qt 大约一个星期来开始开发 PyQt5 应用程序。有一些莫名其妙的问题,但没有一个比这个莫名其妙。

我的应用程序代码:

class LogTableView(QtWidgets.QTableView):    
    def __init__(self, parent, *args, **kwargs):
        super().__init__(parent, *args, **kwargs)

    def resizeEvent(self, resize_event):
        super().resizeEvent(resize_event)
        # self.resizeRowsToContents()

上面的最后一行需要添加。因此,我使用 TDD 方法开始编写测试:

def test_resize_event_should_result_in_resize_rows(request, qtbot):
    t_logger.info(f'\n>>>>>> test name: {request.node.originalname}')
    table_view = logger_table.LogTableView(QtWidgets.QSplitter())
    # with unittest.mock.patch.object(table_view, 'resizeRowsToContents') as mock_resize:
    # with unittest.mock.patch('logger_table.LogTableView.resizeRowsToContents') as mock_resize:
    table_view.resizeEvent(QtGui.QResizeEvent(QtCore.QSize(10, 10), QtCore.QSize(20, 20)))

注意注释掉的行显示了我一直在尝试的那种事情。但是您可以看到,即使只是创建 type 的对象LogTableView,然后调用该方法,根本没有任何模拟,也会导致错误。

在运行这个:

>pytest -s -v -k test_logger_table.py

我明白了:

...
self = <logger_table.LogTableView object at 0x000002B672697670>
resize_event = <PyQt5.QtGui.QResizeEvent object at 0x000002B672743940>

    def resizeEvent(self, resize_event):
>       super().resizeEvent(resize_event)
E       RuntimeError: wrapped C/C++ object of type LogTableView has been deleted
...

有没有人知道这是关于什么的?

PS FWIW,出于绝望,我什至试过这个:

super(LogTableView, self).resizeEvent(resize_event)

...同样的错误。

4

1 回答 1

1

子构造函数中创建父代不是一个好主意。

请记住,PyQt 是一个绑定,Python 中使用的每个引用都是 Qt 对象的包装器:如果在 C++ 端删除该对象,python 引用仍然存在,但任何尝试使用它的函数都会导致上面的 RuntimeError。

在你的情况下,在 python 端没有对父级的持久引用,只有 Qt 端的指针,这不足以避免垃圾收集:只有对象在 Qt 中拥有所有权(这就是为什么你可以避免持久引用子 Qt 对象),情况并非如此。问题是孩子相信它有一个父级(就像它在创建时有一个),但与此同时,一旦返回子级构造函数,这个父级就被删除了。

只需为父级创建一个局部变量。

def test_resize_event_should_result_in_resize_rows(request, qtbot):
    t_logger.info(f'\n>>>>>> test name: {request.node.originalname}')
    parent = QtWidgets.QSplitter()
    table_view = logger_table.LogTableView(parent)
    # ...

除了主题的问题,从技术上讲,使用 QSplitter 等非常具体的小部件作为父级是没有意义的(特别是考虑到为了正确使用,小部件应该添加addWidget(),因为单独的父级对于分离器);如果您需要父母,只需使用基本的 QWidget。

于 2021-11-05T19:27:07.287 回答