Python 3.6.3:PyQt5 Matplotlib 条 x 最大面积。 10 项

Posted

技术标签:

【中文标题】Python 3.6.3:PyQt5 Matplotlib 条 x 最大面积。 10 项【英文标题】:Python 3.6.3: PyQt5 Matplotlib bar x Area max. 10 Items 【发布时间】:2017-10-26 08:34:24 【问题描述】:

我正在尝试使用 matplotlib 和 PyQt5 创建动态条形图。问题是 x 轴的最大元素是有限的。它会针对 9 以下的所有小节进行自我调整。超过 9 个小节会覆盖第一个小节。我使用了 matplotlib 的 GUI,将 plot 命令更改为 bar 并删除了不必要的代码。 http://www.boxcontrol.net/embedding-matplotlib-plot-on-pyqt5-gui.html

import sys
import random
import matplotlib
matplotlib.use("Qt5Agg")
from PyQt5 import QtCore
from PyQt5.QtWidgets import QApplication, QMainWindow, QMenu, QVBoxLayout, QSizePolicy, QMessageBox, QWidget
from numpy import arange, sin, pi
from matplotlib.backends.backend_qt5agg import FigureCanvasQTAgg as FigureCanvas
from matplotlib.figure import Figure

bereiche = ["a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m"]

class MyMplCanvas(FigureCanvas):
    """Ultimately, this is a QWidget (as well as a FigureCanvasAgg, etc.)."""
    def __init__(self, parent=None, width=5, height=4, dpi=100):
        fig = Figure(figsize=(width, height), dpi=dpi)
        self.axes = fig.add_subplot(111)
        # We want the axes cleared every time plot() is called
        #self.axes.hold(False) useing self.axes.cla() instead

        self.compute_initial_figure()

        FigureCanvas.__init__(self, fig)
        self.setParent(parent)

        FigureCanvas.setSizePolicy(self,
                QSizePolicy.Expanding,
                QSizePolicy.Expanding)
        FigureCanvas.updateGeometry(self)

    def compute_initial_figure(self):
        pass

class MyDynamicMplCanvas(MyMplCanvas):
    """A canvas that updates itself every second with a new plot."""
    def __init__(self, *args, **kwargs):
        MyMplCanvas.__init__(self, *args, **kwargs)
        timer = QtCore.QTimer(self)
        timer.timeout.connect(self.update_figure)
        timer.start(1000)

    def compute_initial_figure(self):
        #self.axes.set_xlim(3)
        self.axes.bar(["a", "b", "c"], [1, 2, 3], color='r')

    def update_figure(self):
        # Build a list of 11 random integers between 0 and 10 (both inclusive)
        l = [random.randint(0, 10) for i in range(len(bereiche))]

        self.axes.cla()
        #self.axes.set_xlim(11)
        self.axes.bar(bereiche, l, color='r')
        self.draw()

class ApplicationWindow(QMainWindow):
    def __init__(self):
        QMainWindow.__init__(self)
        self.setAttribute(QtCore.Qt.WA_DeleteOnClose)

        self.main_widget = QWidget(self)

        l = QVBoxLayout(self.main_widget)
        dc = MyDynamicMplCanvas(self.main_widget, width=5, height=4, dpi=100)
        l.addWidget(dc)

        self.main_widget.setFocus()
        self.setCentralWidget(self.main_widget)

if __name__ == '__main__':
    app = QApplication(sys.argv)

    aw = ApplicationWindow()
    aw.setWindowTitle("PyQt5 Matplot Example")
    aw.show()
    #sys.exit(qApp.exec_())
    app.exec_()

每次图表更新并尝试设置 xlim 但它不起作用时,我都用调用 cla 替换了保持。 另一个问题:compute_initial_figure 函数是否必要?我已经停用了它,没有任何区别。

【问题讨论】:

【参考方案1】:

我找到了解决方案:循环生成条形

不需要函数compute_initial_figure(图形会自动调整)

def update_figure(self):
    # Build a list of 11 random integers between 0 and 10 (both inclusive)
    l = [random.randint(0, 10) for i in range(len(bereiche))]
    self.axes.cla()
    i = 0
    for bereich in range(len(bereiche)):
        self.axes.bar(bereich, l[i], color='r')
        i = i + 1
    self.axes.set_xlim(-0.5, len(bereiche)-0.5)
    self.axes.set_xticks(range(len(bereiche)))
    self.axes.set_xticklabels(bereiche)
    self.draw()

首先它将为 bereiche 中的每个元素设置 xticks (int),然后它们被 set_xticklabels(bereiche) 覆盖。不一定需要 xlim。

我不知道为什么它使用循环而不是单行,例如:

self.axes.bar(bereiche, l, color='r')

【讨论】:

以上是关于Python 3.6.3:PyQt5 Matplotlib 条 x 最大面积。 10 项的主要内容,如果未能解决你的问题,请参考以下文章

python pyqt5 自定义信号和槽

Python,pyqt5

python pyqt5 QThread

python pyqt5 QTableWidget 设置表头

搭建pyqt5开发环境(python3+pycharm2019+pyqt5)

Python 3.x 安装PyQt5