setData 很慢
setData is very slow
我正在尝试从我开发的程序中优化方法。
基本上,它是一个显示信息的 GUI(我使用 PyQt 库)。信息存储在 sqlite 数据库中。我使用 QSqlTableModel 和 QTableview 来显示这些信息。这是一个很常见的组合。
数据库中的一个字段是布尔值,称为"new"。我要优化的方法的目的是将这个布尔值设置为0。
方法如下:
def markOneRead(self, element):
"""Slot to mark an article read"""
print("\n")
print("start markoneread")
start_time = datetime.datetime.now()
# Get the QTableView object (I have several)
table = self.liste_tables_in_tabs[self.onglets.currentIndex()]
# Save the current selected line
line = table.selectionModel().currentIndex().row()
print("before bdd change")
elapsed_time = datetime.datetime.now() - start_time
print(elapsed_time)
# Change the data in the model
# The 12th column is the field "new". I write 0
# !!!!! Very long action
table.model().setData(table.model().index(line, 12), 0)
print("before searchbutton")
elapsed_time = datetime.datetime.now() - start_time
输出是这样的:
before bdd change
0:00:00.000141
before searchbutton
0:00:03.064438
基本上,这一行:
table.model().setData(table.model().index(line, 12), 0)
执行需要 3 秒。那很长,我只是更新数据库中的一个项目,它不应该那么长。我的数据库有 25000 个项目,但我认为它不会改变什么。
编辑:
也许是因为模型立即执行更改,并尝试重新加载所有数据?
你知道如何解决这个问题吗?
编辑 2:
其实问题出在数据重载上。如果我更改模型的 editStrategy:
model.setEditStrategy(QtSql.QSqlTableModel.OnManualSubmit)
现在不再需要3秒了,但是视图没有更新,调用方法后new仍然设置为1。
所以我想知道是否有办法在模型更改后 "reload" 只有一个项目,一个索引?
也许这有帮助:
与:
index = table.model().index(line, 12)
table.model().dataChanged.emit(index,index)
或:
table.model().dataChanged.emit(table.model().index(line, 12),table.model().index(line, 12))
您可以定义受更改影响的项目
参见 documentation。
通过table.model().dataChanged
-发出信号table.model.setData()
之后的自动更新或重绘应该被限制在代表top-left和bottom-right[=27]的两个索引定义的区域=].如果两个索引相同,则只有一个项目受到影响。
QSqlTableModel
非常方便,但并不神奇。我的猜测是执行更新语句花费了大部分时间。尝试手动更新几行,看看需要多长时间。
如果这也很慢,那么您可能需要 table 上的索引,以便 sqlite 可以更快地定位行。
我正在尝试从我开发的程序中优化方法。
基本上,它是一个显示信息的 GUI(我使用 PyQt 库)。信息存储在 sqlite 数据库中。我使用 QSqlTableModel 和 QTableview 来显示这些信息。这是一个很常见的组合。
数据库中的一个字段是布尔值,称为"new"。我要优化的方法的目的是将这个布尔值设置为0。
方法如下:
def markOneRead(self, element):
"""Slot to mark an article read"""
print("\n")
print("start markoneread")
start_time = datetime.datetime.now()
# Get the QTableView object (I have several)
table = self.liste_tables_in_tabs[self.onglets.currentIndex()]
# Save the current selected line
line = table.selectionModel().currentIndex().row()
print("before bdd change")
elapsed_time = datetime.datetime.now() - start_time
print(elapsed_time)
# Change the data in the model
# The 12th column is the field "new". I write 0
# !!!!! Very long action
table.model().setData(table.model().index(line, 12), 0)
print("before searchbutton")
elapsed_time = datetime.datetime.now() - start_time
输出是这样的:
before bdd change
0:00:00.000141
before searchbutton
0:00:03.064438
基本上,这一行:
table.model().setData(table.model().index(line, 12), 0)
执行需要 3 秒。那很长,我只是更新数据库中的一个项目,它不应该那么长。我的数据库有 25000 个项目,但我认为它不会改变什么。
编辑: 也许是因为模型立即执行更改,并尝试重新加载所有数据?
你知道如何解决这个问题吗?
编辑 2:
其实问题出在数据重载上。如果我更改模型的 editStrategy:
model.setEditStrategy(QtSql.QSqlTableModel.OnManualSubmit)
现在不再需要3秒了,但是视图没有更新,调用方法后new仍然设置为1。
所以我想知道是否有办法在模型更改后 "reload" 只有一个项目,一个索引?
也许这有帮助:
与:
index = table.model().index(line, 12)
table.model().dataChanged.emit(index,index)
或:
table.model().dataChanged.emit(table.model().index(line, 12),table.model().index(line, 12))
您可以定义受更改影响的项目 参见 documentation。
通过table.model().dataChanged
-发出信号table.model.setData()
之后的自动更新或重绘应该被限制在代表top-left和bottom-right[=27]的两个索引定义的区域=].如果两个索引相同,则只有一个项目受到影响。
QSqlTableModel
非常方便,但并不神奇。我的猜测是执行更新语句花费了大部分时间。尝试手动更新几行,看看需要多长时间。
如果这也很慢,那么您可能需要 table 上的索引,以便 sqlite 可以更快地定位行。