Tkinter - 如何将实例变量传递给另一个类?

2024-02-19

我用谷歌搜索了很多,但没有成功。上周发布了一个问题,但没有答案,因为似乎太长了......

希望新问题更加清晰。

这只是一小段代码,如果您运行它,您将能够重现该问题。我基本上需要做的是获取用户输入(来自 mainGUI 类的 E1)并将其传递给数据库类中的 insert() 函数。当我尝试添加条目时出现的错误是:

"self.curs.execute("插入日记值 (?)", (日期)) sqlite3.ProgrammingError:提供的绑定数量不正确。当前语句使用 1,并且提供了 0。”

我可以毫无问题地读取数据库。

任何帮助将非常感激。

from tkinter import *
import sqlite3

    class mainGUI(object):

    def __init__(self,window):

        self.window=window 

        self.E1 = Entry(window)                           
        self.E1.grid(row=1, column=0)
        self.EG1 = self.E1.get()

        global E4
        E4 = Listbox(window)                           
        E4.grid(row=2)

        B1 = Button(window, text="Add entry", command=lambda: database.insert(self.EG1)) 

        B1.grid(row=1, column=4)

        B2 = Button(window, text="View all", command=database.view_all)  
        B2.grid(row=2, column=4, sticky="WN")

        window.mainloop()

class Database(mainGUI):

    def __init__(self, db):
        self.conn = sqlite3.connect(db)
        self.curs = self.conn.cursor() 
        self.curs.execute("CREATE TABLE IF NOT EXISTS diary (date TEXT)")   
        self.conn.commit()

    def insert(self,date): 
        database.add_entry(date) 
        E4.delete(0, END)
        E4.insert(END,(date))

    def add_entry(self,date):     
        self.curs.execute("INSERT INTO diary VALUES (?)", (date))
        self.conn.commit()

    def view_all(self): 
        E4.delete(0, END)    
        self.curs.execute("SELECT * FROM diary")
        data = self.curs.fetchall()
        for row in data:
           E4.insert(END,row)                           

if __name__ == "__main__":   
    database = Database("dbase.db")
    window=Tk()
    gui = mainGUI(window)

如果您使用类,则不应使用全局变量。相反,您需要访问对象实例的属性和方法。这意味着创建E4作为实例变量,并通过 GUI 类的实例访问它。

另外,你的数据库应该not继承自mainGUI。这不是应该如何使用继承。

您对数据库进行了编码,因此需要更改 GUI,这是不寻常的。相反,您的 GUI 应该是唯一可以修改 GUI 的东西,而您的数据库应该是唯一可以修改数据库的东西。当GUI需要向数据库发送信息或从数据库获取信息时,它可以调用数据库上的方法。

因此,第一步是将数据库对象传递给您的mainGUI类(应该命名为MainGUI根据 PEP8):

class MainGUI(object):
    def __init__(self,window, db):
        self.window = window
        self.db = db
        ...

接下来,在创建 GUI 时传入数据库对象:

if __name__ == "__main__":   
    database = Database("dbase.db")
    window=Tk()
    gui = MainGUI(window, database)

这样,您现在可以使用数据库来获取数据,并使用 GUI 来显示数据。

例如,您的 GUI 应该具有“view_all”方法,因为它涉及更改视图。它应该调用数据库的“get_all”方法。经过not using lambda,您的代码变得更容易阅读、编写和调试。

class mainGUI(object):
    def __init__(self,window, db):
        ...
        self.E4 = Listbox(...)
        ...
        B2 = Button(..., command=self.view_all) 

    def view_all(self):
        data = self.db.get_all()
        self.E4.delete(0, END)    
        for row in data:
           self.E4.insert(END,row)

最后,创建一个get_all方法在你的Database除了获取并返回数据之外什么也不做:

class MainGUI(object):
    def get_all(self):   
        self.curs.execute("SELECT * FROM diary")
        data = self.curs.fetchall()
        return data

有了上述所有内容,并将相同的概念应用于其他方法之后(即:Database类应该只设置或获取数据),你有一个可重用的Database完全独立于使用它的 GUI 的类。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Tkinter - 如何将实例变量传递给另一个类? 的相关文章

随机推荐