亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

基于Python實現(xiàn)多圖繪制系統(tǒng)

 更新時間:2024年02月18日 09:14:16   作者:微小冷  
這篇文章主要為大家詳細(xì)介紹了如何基于Python實現(xiàn)一個簡單的多圖繪制系統(tǒng),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下

參考:從零開始實現(xiàn)一個三維繪圖系統(tǒng)

需求和框架

本文希望實現(xiàn)下圖所示的繪圖系統(tǒng),下面詳細(xì)分析需求變化。

和之前實現(xiàn)的繪圖系統(tǒng)相比,首先是多了【新增】和【刪除】這兩個按鈕,其功能是控制繪圖數(shù)據(jù)的個數(shù),即下方的【坐標(biāo)0】和【坐標(biāo)1】。由于不同的坐標(biāo)之間有著相同的組成,故而需要封裝成類;而每個坐標(biāo)中的x,y,z坐標(biāo)軸,又十分相似,故而也需要封裝成類。

綜合來看,和此前的簡單繪圖系統(tǒng)相比,若想實現(xiàn)多圖繪制的功能,至少需要實現(xiàn)三個類,分別表示坐標(biāo)軸、坐標(biāo)系以及坐標(biāo)系統(tǒng)。

下面列出整個項目需要導(dǎo)入的模塊,以及這三個類的命名

import tkinter as tk
import tkinter.ttk as ttk

import matplotlib as mpl
mpl.use('TkAgg')
import matplotlib.pyplot as plt
from matplotlib.backends.backend_tkagg import (
    FigureCanvasTkAgg, NavigationToolbar2Tk)
from matplotlib.figure import Figure

import numpy as np

class AxisFrame(ttk.Frame):
    pass

class AxisList(ttk.Frame):
    pass

class DarwSystem():
    pass

下面逐一實現(xiàn),并講解。

AxisFrame

下圖是自定義的AxisList組件,其中每一項x,y,z均對應(yīng)一組AxisFrame對象,下面就分析AxisFrame對象的特點。

首先,AxisFrame由三部分組成:坐標(biāo)標(biāo)簽、輸入類型和輸入框。目前來說,由于只實現(xiàn)了讀取Python源碼這種方法,故而輸入類型的下拉框顯得有些多余,其目的是便于后期擴展。

據(jù)此,可得到其代碼

class AxisFrame(ttk.Frame):
    # widths 是每個控件的寬度
    def __init__(self, master, label, mode, widths, **options):
        super().__init__(master, **options)
        self.pack()
        self.label = label
        self.initVar(mode)
        self.initWidgets(widths)

    def initVar(self, mode):
        self.MODES = ("源代碼",)
        self.mode = tk.StringVar()
        self.setMode(mode)

    def initWidgets(self, widths):
        tk.Label(self, text=self.label,
            width=widths[0]).pack(side=tk.LEFT)
        self.slct = ttk.Combobox(self,
            width=widths[1], textvariable=self.mode)
        self.slct['value'] = self.MODES
        self.slct.pack(side=tk.LEFT)
        self.entry = tk.Entry(self, width=widths[2])
        self.entry.pack(padx=5, side=tk.LEFT, fill=tk.X)

    def setMode(self, mode):
        if type(mode) != str:
            mode = self.MODES[mode]
        self.mode.set(mode)

    def getData(self, **txyz):
        if self.mode.get() == "源代碼":
            self.readPython(**txyz)
        return self.data

    def readPython(self, t=None, x=None, y=None, z=None):
        self.data = eval(self.entry.get())

其中,initVar用于初始化輸入模式;setMode用于設(shè)置輸入模式;initWidgets用于布局,這三個函數(shù)只需了解tkinter的工作原理,便無需詳解。

getData用于將輸入框中的內(nèi)容轉(zhuǎn)換為數(shù)值,由于目前只有【源代碼】這一種模式,所以只用到了readPython這一種讀取函數(shù)。readPython的邏輯很簡單,根據(jù)輸入的t,x,y,z的值,以及輸入框中的表達(dá)式,來實現(xiàn)python數(shù)據(jù)的讀取。

AxisList

繼續(xù)分析AxisList組件,除了三個AxisFrame之外,還有一個顯隱開關(guān)【坐標(biāo)0】,以及坐標(biāo)選框【xyz】,用于調(diào)整是否使用x,y,z等坐標(biāo)軸,即實現(xiàn)下面的效果。

從交互邏輯而言,這里有兩個功能,一是點擊【坐標(biāo)0】以顯示或隱藏其內(nèi)容,二是選擇下拉框,以篩選x,y,z等坐標(biāo)軸。實現(xiàn)代碼如下

class AxisList(ttk.Frame):
    def __init__(self, master, title, mode, widths, **options):
        super().__init__(master, **options)
        self.pack()
        self.afs = {}
        self.data = {}
        self.initWidgets(title, widths)
        self.initAxis(mode, widths)

    def initWidgets(self, title, widths):
        self.btn = ttk.Button(self, text=title, width=sum(widths)+5,
            command=self.click)
        self.btn.pack(side=tk.TOP, fill=tk.X, expand=tk.YES)
        self._c = ttk.Frame(self)
        self.collapsed = True
        self.click()
        self.initVar()
        self.initFeature()

    def initVar(self):
        self.AXISES = ("x", "xy", "xyz", "tx", "txy", "txyz")
        self.axMode = tk.StringVar()
        self.axMode.set("xyz")

    def initAxis(self, mode, widths):
        for flag in 'txyz':
            self.afs[flag] = AxisFrame(self._c, flag, mode, widths)
        self.axChanged()

    def initFeature(self):
        frm = ttk.Frame(self._c)
        frm.pack(pady=2, side=tk.TOP, fill=tk.X)

        slct = ttk.Combobox(frm, width=5, textvariable=self.axMode)
        slct["value"] = self.AXISES
        slct.pack(side=tk.LEFT)
        slct.bind('<<ComboboxSelected>>', self.axChanged)

    def axChanged(self, evt=None):
        for flag in 'txyz':
            self.afs[flag].pack_forget()
        for flag in self.axMode.get():
            self.afs[flag].pack(side=tk.TOP, fill=tk.X)

    def click(self):
        if self.collapsed:
            self._c.pack(side=tk.TOP, fill=tk.BOTH, expand=tk.YES)
        else:
            self._c.pack_forget()
        self.collapsed = not self.collapsed

    def getDatas(self):
        dct = {}
        data = {}
        for flag in self.axMode.get():
            data[flag] = self.afs[flag].getData(**dct)
            dct[flag] = data[flag]
        return data

首先,點擊【坐標(biāo)0】之后,除了【坐標(biāo)0】之外其他的內(nèi)容全部隱藏,所以【坐標(biāo)0】與其內(nèi)容需要分開處理,從而為其他內(nèi)容分配一個Frame,由于這個Frame將會頻繁地被用到,所以給一個簡短的名稱_c。

然后,下拉框中共給定6種坐標(biāo)類型,之所以有t tt選項,目的是日后可能會實現(xiàn)的動圖繪制。

接下來,initVar, initWidgets, initAxis, initFeature是布局函數(shù),無需多言。

axChanged即更改坐標(biāo)軸系組成時的響應(yīng)函數(shù),首先隱藏所有坐標(biāo)軸,再根據(jù)選擇的坐標(biāo)軸一個一個重新放出來。

click是點擊【坐標(biāo)0】時的相應(yīng),如果_c已經(jīng)隱藏,就將其顯示出來;如果當(dāng)前可見,就讓其變得不可見。

最后,getDatas是坐標(biāo)系的關(guān)鍵函數(shù),其功能是調(diào)用AxisFrame種中的getData函數(shù),其中的dct裝載已經(jīng)獲取到的數(shù)值,通過字典的雙星號索引方法,將字符串key轉(zhuǎn)換為變量名稱,以實現(xiàn)最終的數(shù)值讀取。

DarwSystem

DrawSystem的代碼雖然較長,但內(nèi)容并不復(fù)雜,其中setFrmFig更是幾乎原封不動地沿用了此前的代碼。

class DarwSystem():
    def __init__(self):
        self.root = tk.Tk()
        self.root.title("數(shù)據(jù)展示工具")
        self.als = []

        self.setFrmCtrl()
        self.setFrmFig()
        self.root.mainloop()

    def setFrmCtrl(self):
        frmCtrl = ttk.Frame(self.root,width=320)
        frmCtrl.pack(side=tk.RIGHT, fill=tk.Y)
        frm = ttk.Frame(frmCtrl, width=320)
        frm.pack(side=tk.TOP, fill=tk.X)
        self.setCtrlButtons(frm)
        self.frmAxis = ttk.Frame(frmCtrl)
        self.frmAxis.pack(side=tk.TOP, fill=tk.X)
        self.addLast(None)

    def addLast(self, evt):
        title = f"坐標(biāo){len(self.als)}"
        al = AxisList(self.frmAxis, title, 0, [5,10,20])
        al.pack(side=tk.TOP, fill=tk.X)
        self.als.append(al)

    def deleteLast(self, evt):
        self.als[-1].pack_forget()
        del self.als[-1]

    def setCtrlButtons(self, frm):
        ttk.Button(frm, text="繪圖",width=5,
            command=self.btnDrawImg).pack(side=tk.LEFT)
        btn = ttk.Button(frm, text="新增", width=5)
        btn.pack(side=tk.LEFT)
        btn.bind("<Button-1>", self.addLast)

        btn = ttk.Button(frm, text="刪除", width=5)
        btn.pack(side=tk.LEFT)
        btn.bind("<Button-1>", self.deleteLast)

    def btnDrawImg(self):
        self.fig.clf()
        ds = [al.getDatas() for al in self.als]
        self.drawPlot(ds)
        self.fig.subplots_adjust(left=0.1, right=0.95, top=0.95, bottom=0.08)
        self.canvas.draw()

    def drawPlot(self, ds):
        keys = 'xyz' if 'z' in ds[0] else 'xy'
        p = '3d' if 'z' in ds[0] else None
        ax = self.fig.add_subplot(projection=p)
        for data in ds:
            ax.plot(*[data[key] for key in keys])

    def setFrmFig(self):
        frmFig = ttk.Frame(self.root)
        frmFig.pack(side=tk.LEFT,fill=tk.BOTH,expand=tk.YES)
        self.fig = Figure()
        self.canvas = FigureCanvasTkAgg(self.fig,frmFig)
        self.canvas.get_tk_widget().pack(
            side=tk.TOP,fill=tk.BOTH,expand=tk.YES)
        self.toolbar = NavigationToolbar2Tk(self.canvas,frmFig,
            pack_toolbar=False)
        self.toolbar.update()
        self.toolbar.pack(side=tk.RIGHT)

相比之下,其實只添加了兩個功能:

  • 【添加】和【刪除】坐標(biāo)系
  • 多個坐標(biāo)軸的繪圖函數(shù)

對于前者,其通過addLast在最后面新增一個坐標(biāo)列表,delLast則刪除最后一個坐標(biāo)列表,實現(xiàn)并不復(fù)雜。

對于后者,只需遍歷所有AxisList并提取數(shù)據(jù),然后將所有數(shù)據(jù)傳遞給繪圖函數(shù)進行圖像繪制。

至此,需求得以實現(xiàn),下面演示一下其工作流程

到此這篇關(guān)于基于Python實現(xiàn)多圖繪制系統(tǒng)的文章就介紹到這了,更多相關(guān)Python多圖繪制系統(tǒng)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Python中enumerate函數(shù)代碼解析

    Python中enumerate函數(shù)代碼解析

    這篇文章主要介紹了Python中enumerate函數(shù)代碼解析,涉及函數(shù)說明以及相關(guān)示例,具有一定參考價值,需要的朋友可以了解下。
    2017-10-10
  • 如何將Python徹底卸載的三種方法

    如何將Python徹底卸載的三種方法

    通常我們在一些軟件的使用上有碰壁,第一反應(yīng)就是卸載重裝,所以有小伙伴就問我Python怎么卸載才能徹底卸載干凈,今天這篇文章,小編就來教大家如何徹底卸載Python,需要的朋友可以參考下
    2025-04-04
  • Scrapy爬蟲多線程導(dǎo)致抓取錯亂的問題解決

    Scrapy爬蟲多線程導(dǎo)致抓取錯亂的問題解決

    本文針對Scrapy爬蟲多線程導(dǎo)致抓取錯亂的問題進行了深入分析,并提出了相應(yīng)的解決方案,具有一定的參考價值,感興趣的可以了解一下
    2023-11-11
  • 一文詳解Python的pyc文件

    一文詳解Python的pyc文件

    Python?程序在執(zhí)行過程中,會產(chǎn)生一些中間文件,其中最常見的就是?pyc?文件,pyc?文件是?Python?的二進制字節(jié)碼文件,本文將通過簡潔的語言、實際的代碼和案例,通俗易懂地解釋?pyc?文件的相關(guān)知識,感興趣的小伙伴跟著小編一起來看看吧
    2024-12-12
  • python模擬事件觸發(fā)機制詳解

    python模擬事件觸發(fā)機制詳解

    這篇文章主要為大家詳細(xì)介紹了python模擬事件觸發(fā)機制的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • python3實現(xiàn)字符串的全排列的方法(無重復(fù)字符)

    python3實現(xiàn)字符串的全排列的方法(無重復(fù)字符)

    這篇文章主要介紹了python3實現(xiàn)字符串的全排列的方法(無重復(fù)字符),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-07-07
  • Python 下載Bing壁紙的示例

    Python 下載Bing壁紙的示例

    這篇文章主要介紹了Python 下載Bing壁紙的示例,幫助大家更好的理解和使用python,感興趣的朋友可以了解下
    2020-09-09
  • Python實現(xiàn)微信小程序自動操作工具

    Python實現(xiàn)微信小程序自動操作工具

    這篇文章主要為大家詳細(xì)介紹了如何利用Python實現(xiàn)微信小程序自動化操作的小工具,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下
    2023-01-01
  • 詳解Python2.x中對Unicode編碼的使用

    詳解Python2.x中對Unicode編碼的使用

    這篇文章主要介紹了詳解Python2.x中對Unicode編碼的使用,Python3中Unicode被作為默認(rèn)的編碼來使用,而在目前仍被廣泛應(yīng)用的Python2的版本中Unicode卻是一個在使用中需要注意的地方,需要的朋友可以參考下
    2015-04-04
  • Python利用splinter實現(xiàn)瀏覽器自動化操作方法

    Python利用splinter實現(xiàn)瀏覽器自動化操作方法

    今天小編就為大家分享一篇Python利用splinter實現(xiàn)瀏覽器自動化操作方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-05-05

最新評論