我试图弄清楚如何使用QTableView小部件显示我拥有子集的表。我遇到了一个奇怪的错误,我不确定这是一个错误还是我的代码有问题。我可以很好地加载数据集。当程序运行并按下数据加载按钮时,QTableView将显示适当数量的行和列名,但大多数(如果不是全部)数据丢失。但是,如果您尝试通过单击任何列名对表进行排序,则会显示数据。当我第一次加载数据集时,一些错误消息会说“#根据定义,我们只有第0个轴”或“#我们产生了一个标量?”。你知道吗
我想这可能是我使用的csv文件的一个问题,但是我可以用一个随机的数字表轻松地重现它。注释掉sort函数仍然会导致没有数据显示,但是现在排序不能用来显示数据。我在示例中使用的代码是对答案here的修改。你知道吗
导入必要的包和函数。你知道吗
from PyQt5 import QtCore, QtGui, QtWidgets
import pandas as pd
import numpy as np
class PandasModel(QtCore.QAbstractTableModel):
def __init__(self, df = pd.DataFrame(), parent=None):
QtCore.QAbstractTableModel.__init__(self, parent=parent)
self._df = df
def headerData(self, section, orientation, role=QtCore.Qt.DisplayRole):
if role != QtCore.Qt.DisplayRole:
return QtCore.QVariant()
if orientation == QtCore.Qt.Horizontal:
try:
return self._df.columns.tolist()[section]
except (IndexError, ):
return QtCore.QVariant()
elif orientation == QtCore.Qt.Vertical:
try:
# return self.df.index.tolist()
return self._df.index.tolist()[section]
except (IndexError, ):
return QtCore.QVariant()
def data(self, index, role=QtCore.Qt.DisplayRole):
if role != QtCore.Qt.DisplayRole:
return QtCore.QVariant()
if not index.isValid():
return QtCore.QVariant()
return QtCore.QVariant(str(self._df.ix[index.row(), index.column()]))
def setData(self, index, value, role):
row = self._df.index[index.row()]
col = self._df.columns[index.column()]
if hasattr(value, 'toPyObject'):
# PyQt4 gets a QVariant
value = value.toPyObject()
else:
# PySide gets an unicode
dtype = self._df[col].dtype
if dtype != object:
value = None if value == '' else dtype.type(value)
self._df.set_value(row, col, value)
return True
def rowCount(self, parent=QtCore.QModelIndex()):
return len(self._df.index)
def columnCount(self, parent=QtCore.QModelIndex()):
return len(self._df.columns)
def sort(self, column, order):
colname = self._df.columns.tolist()[column]
self.layoutAboutToBeChanged.emit()
self._df.sort_values(colname, ascending= order == QtCore.Qt.AscendingOrder, inplace=True)
self._df.reset_index(inplace=True, drop=True)
self.layoutChanged.emit()
显示表并处理子集的应用程序。你知道吗
class Widget(QtWidgets.QWidget):
def __init__(self, parent=None):
QtWidgets.QWidget.__init__(self, parent=None)
vLayout = QtWidgets.QVBoxLayout(self)
hLayout = QtWidgets.QHBoxLayout()
self.loadBtn = QtWidgets.QPushButton("Load Table", self)
hLayout.addWidget(self.loadBtn)
vLayout.addLayout(hLayout)
self.pandasTv = QtWidgets.QTableView(self)
vLayout.addWidget(self.pandasTv)
self.loadBtn.clicked.connect(self.loadTable)
#self.pandasTv.setSortingEnabled(True)
def loadTable(self):
dfNum = pd.DataFrame(np.random.randint(0,100,size=(20, 4)), columns=list('ABCD'))
dfNumSub = dfNum.loc[dfNum['A'].astype(str).str.contains('1')]
model = PandasModel(dfNumSub)
self.pandasTv.setModel(model)
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
w = Widget()
w.show()
sys.exit(app.exec_())
我希望所有搜索到的数据都显示出来,而不必对列进行排序。你知道吗
编辑:此错误仅在对数据帧进行子集设置时发生。改为将代码改为显示dfNum将导致所有数据都显示得很好。你知道吗
使数据出现的不是对数据帧本身进行排序,而是在
PandasModel.sort()
末尾对self._df.reset_index()
的调用。没有这个语句,PandasModel.data()
中对self._df.ix()
的调用似乎中断了。要解决这个问题,您可以尝试在Widget.loadTable
中设置模型之前应用dfNumSub.reset_index()
,或者您可以将self._df.ix()
更改为PandasModel.data()
中的self._df.iloc()
,因为DataFrame.ix
已被弃用。你知道吗相关问题 更多 >
编程相关推荐