UPDATE: (SOLVED)

感谢 eyllanesc 让我编写了我的问题的工作示例代码 . 这让我发现了问题 . 这是一个有固定问题的工作代码:

from PyQt5.QtCore import *
from PyQt5.QtWidgets import *
from PyQt5.QtGui import *
import sys
import os

class ExtendedTreeWidgetItem(QTreeWidgetItem):

    def __init__(self, *args, **kwargs):
        super(ExtendedTreeWidgetItem, self).__init__(*args, **kwargs)
        self.rescueFile = None
        print("created first time")

    def setRescueFile(self, rFile):
        self.rescueFile = rFile
        print("setting rescue file", self.rescueFile, "for item", self.text(0))

    def getRescueFile(self):
        return self.rescueFile

class RescueFile:
    def __init__(self, path):
        self.path = path
        self.ischecked = True

    def isChecked(self):
        return self.ischecked

    def setChecked(self, checked):
        if isinstance(checked, bool):
            self.ischecked = checked
        elif isinstance(checked, Qt.CheckState):
            self.ischecked = True if checked == Qt.Checked else False

    def getPath(self):
        return self.path


class Window(QMainWindow):
    def __init__(self):
        super(Window, self).__init__()
        windowSize = (800,400)
        screenSize = QDesktopWidget().availableGeometry()
        self.setGeometry(screenSize.width()/2 - windowSize[0]/2, screenSize.height()/2 - windowSize[1]//2, windowSize[0], windowSize[1])
        self.setWindowTitle("TreeTest")



        self.buildGui()

        self.show()

    def buildGui(self):
        wgt = QWidget()
        vlayout = QVBoxLayout()
        vlayout.setAlignment(Qt.AlignTop)
        grid_layout = QGridLayout()

        self.setupTree(grid_layout)

        vlayout.addLayout(grid_layout)
        wgt.setLayout(vlayout)
        self.setCentralWidget(wgt)


    def setupTree(self, grid_layout):
        self.treeWidget = QTreeWidget()
        self.treeWidget.setHeaderLabel("Results")
        self.treeWidget.setColumnCount(1)

        #INSERTING DATA
        topItem = ExtendedTreeWidgetItem()
        topItem.setText(0, "top item")
        topItem.setCheckState(0, Qt.Checked)
        child1 = ExtendedTreeWidgetItem(topItem)
        child1.setText(0, "child 1")
        child1.setCheckState(0, Qt.Checked)
        topItem.addChild(child1)
        deeper_child = ExtendedTreeWidgetItem(child1)
        deeper_child.setText(0, "deeper child 1")
        deeper_child.setCheckState(0, Qt.Checked)
        r_file = RescueFile("/home/user1/Desktop")
        deeper_child.setRescueFile(r_file)
        child1.addChild(deeper_child)

        self.treeWidget.addTopLevelItem(topItem)
        self.treeWidget.expandAll()

        self.treeWidget.itemChanged.connect(self.singleClickTreeWidget)
        grid_layout.addWidget(self.treeWidget, 1, 1, 4, 3)

    def singleClickTreeWidget(self, widgetItem, column):
        parent = widgetItem.parent()
        if parent and parent.checkState(0) == Qt.Unchecked:
            widgetItem.setCheckState(0, Qt.Unchecked)
            return

        checkState = widgetItem.checkState(0)

        widgetItem.setCheckState(0, checkState)
        rescue_file = widgetItem.getRescueFile()
        if rescue_file:
            rescue_file.setChecked(checkState)
            print("rescue file found, path:", rescue_file.getPath(), "checked:", rescue_file.isChecked())

        self.iterateThroughChildren(widgetItem, checkState)

    def iterateThroughChildren(self, item, checkState):
        for i in range(item.childCount()):
            child = item.child(i)
            print("child:", child, ",text:", child.text(0))
            child.setCheckState(0, checkState)
            # HERE WAS THE MISTAKE
            # -- >> rescue_file = item.getRescueFile()  << --
            rescue_file = child.getRescueFile()  # CORRECT CODE!! 
            print("rescue file", rescue_file, "child", child.text(0))
            if rescue_file is not None:
                rescue_file.setChecked(checkState)
            else:
                self.iterateThroughChildren(child, checkState)



def main():
    if __name__ == '__main__':
        app = QApplication(sys.argv)
        window = Window()
        app.exec_()

main()

BEFORE UPDATE:

我希望我能够清楚地解释我的问题 .

我试图在python 3.5.3中创建使用扩展QTreeWidgetItems的QTreeWidget . 这是我的项目类的代码:

class ExtendedTreeWidgetItem(QTreeWidgetItem):

    def __init__(self, *args, **kwargs):
        super(ExtendedTreeWidgetItem, self).__init__(*args, **kwargs)
        self.rescueFile = None

    def setRescueFile(self, rFile):
        self.rescueFile = rFile

    def getRescueFile(self):
        return self.rescueFile



class RescueFile:

    def __init__(self, path):
        self.path = path
        self.ischecked = True

    def isChecked(self):
        return self.ischecked

    def setChecked(self, checked):
        if isinstance(checked, bool):
            self.ischecked = checked
        elif isinstance(checked, Qt.CheckState):
            self.ischecked = True if checked == Qt.Checked else False
        print(self.path, self.ischecked)

我使用此代码来实现检查和取消选中救援文件:

****
...
self.treeWidget.itemChanged.connect(self.singleClickTreeWidget)
...
****


def singleClickTreeWidget(self, widgetItem, column):
        parent = widgetItem.parent()
        if parent and parent.checkState(0) == Qt.Unchecked:
            widgetItem.setCheckState(0, Qt.Unchecked)
            return

        checkState = widgetItem.checkState(0)

        widgetItem.setCheckState(0, checkState)
        rescue_file = widgetItem.getRescueFile()
        **# I CAN GET THE RESCUE FILE OBJECT HERE FROM**
        if rescue_file:
            rescue_file.setChecked(checkState)

        self.iterateThroughChildren(widgetItem, checkState)



def iterateThroughChildren(self, item, checkState):
    for i in range(item.childCount()):
        child = item.child(i)
        child.setCheckState(0, checkState)
        **# I CAN'T GET ANY FIND HERE ANYMORE**
        rescue_file = item.getRescueFile()
        if rescue_file:
            rescue_file.setChecked(checkState)
        else:
            self.iterateThroughChildren(child, checkState)

我的代码所做的是它生成一个可检查的树,如图所示:
QTreeWidget

我想要实现的是,当我取消选择/选择具有子项的项目(ExtendedTreeWidgetItem)时,也会选择/取消选择所有子项以及与项目关联的RescueFile对象 . 只有文件与RescueFile对象相关联 . 目录留有 self.rescueFile = None

例如,如果我按下取消选择'icons',则还应取消选择magnifier.png以及与之关联的RescueFile . 取消选中复选框就像魅力一样,但如果按下文件的父级,RescueFile不会受到影响(未找到) . 但是如果我直接按下文件,例如magnifier.png,它就可以工作 .

我试图追踪它是否是指针问题,但似乎所有对象都指向它们应该的对象 . 如果我通过它的父节点递归地传递ExtendedTreeWidgetItem,我不明白rescueFile会在哪里消失 .