如何在 tkinter 中访问不同类的变量?
- 2024-12-18 08:39:00
- admin 原创
- 127
问题描述:
我搜索了很多,但仍然不知道如何在 python 中访问不同类的变量。在这种情况下,我想self.v
从PageOne
一个类到PageTwo
另一个类访问变量。
这是我的代码。
import tkinter as tk
import smtplib
TITLE_FONT = ("Helvetica", 18, "bold")
class SampleApp(tk.Tk):
def __init__(self):
tk.Tk.__init__(self)
container = tk.Frame(self)
container.pack(side="top", fill="both", expand=True)
container.grid_rowconfigure(0, weight=1)
container.grid_columnconfigure(0, weight=1)
self.frames = {}
for F in (StartPage, PageOne, PageTwo):
frame = F(container, self)
self.frames[F] = frame
frame.grid(row=0, column=0, sticky="nsew")
self.show_frame(StartPage)
def show_frame(self, c):
frame = self.frames[c]
frame.tkraise()
class StartPage(tk.Frame):
def __init__(self, parent, controller):
tk.Frame.__init__(self, parent)
label = tk.Label(self, text="PyMail",foreground = "Red", font=("Courier", 30, "bold"))
label.pack(side="top")
sublabel = tk.Label(self, text="Bringing you the
the easiest way of communication",
font=("Courier", 15))
sublabel.pack()
wallpaper = tk.PhotoImage(file='Python-logo-notext.gif')
img = tk.Label(self, image=wallpaper)
img.image = wallpaper
img.pack(side="top", expand = True)
button1 = tk.Button(self, text="Click Here to Login to your account",fg="red",
command=lambda: controller.show_frame(PageOne))
button2 = tk.Button(self, text="Go to Page Two",
command=lambda: controller.show_frame(PageTwo))
button2.pack(side="bottom")
button1.pack(side="bottom")
class PageOne(tk.Frame):
def __init__(self, parent, controller):
tk.Frame.__init__(self, parent)
self.controller=controller
label = tk.Label(self, text="Personal Information", font=TITLE_FONT, foreground="blue")
label.pack(side="top", fill="x", pady=10)
global optionv
self.optionv = tk.StringVar()
self.optionv.set("---Select One---")
optionm = tk.OptionMenu(self, self.optionv, "---Select One---", "@gmail.com", "@yahoo.com", "@hotmail.com")
t1 = tk.Label(self, text="Email Account: ")
self.v = tk.StringVar()
self.v.set("")
entry1 = tk.Entry(self, textvariable=self.v)
t2 = tk.Label(self,text="
Password: ")
self.pwd = tk.StringVar()
self.pwd.set("")
entry2 = tk.Entry(self, textvariable=self.pwd)
entry2.config(show="*")
lgbutton=tk.Button(self, text="Log In", command=self.login)
button = tk.Button(self, text="Go to the start page",
command=lambda: controller.show_frame(StartPage))
#final = tk.Label(self, textvariable=self.v)
#finalpwd = tk.Label(self, textvariable=self.pwd)
t1.pack()
entry1.pack()
optionm.pack()
t2.pack()
entry2.pack()
#final.pack()
#finalpwd.pack()
lgbutton.pack()
button.pack(side="bottom")
def login(self):
value = tk.Label(self, text="Invalid username / password", font=("Courier", 15, "bold"), foreground="red")
success = tk.Label(self, text="Login was Successful
(Click ""Continue"" to compose email)", font=("Courier", 15, "bold"), foreground="blue")
cbutton = tk.Button(self, text="Continue", command=lambda: self.controller.show_frame(PageTwo))
status = tk.Label(self, text="Please select your email domain", foreground="red")
if self.optionv.get() == "@gmail.com":
try:
global server
server = smtplib.SMTP("smtp.gmail.com", 587)
server.ehlo()
server.starttls()
server.login(self.v.get()+self.optionv.get(), self.pwd.get())
success.pack()
cbutton.pack(side="bottom")
except:
value.pack()
elif self.optionv.get() == "@yahoo.com":
try:
server = smtplib.SMTP("smtp.yahoo.com", 587)
server.ehlo()
server.starttls()
server.login(self.v.get()+self.optionv.get(), self.pwd.get())
success.pack()
cbutton.pack(side="bottom")
except:
value.pack()
elif self.optionv.get() == "@hotmail.com":
try:
server = smtplib.SMTP("smtp.live.com", 587)
server.ehlo()
server.starttls()
server.login(self.v.get()+self.optionv.get(), self.pwd.get())
success.pack()
cbutton.pack(side="bottom")
except:
value.pack()
else:
status.pack()
class PageTwo(tk.Frame):
def __init__(self, parent, controller):
tk.Frame.__init__(self, parent)
label = tk.Label(self, text="Compose Mail", font=TITLE_FONT, foreground="green")
label.pack(side="top", fill="x", pady=10)
self.reciever = tk.StringVar()
self.reciever.set("")
senderl = tk.Label(self, text="Send to: ")
rmail = tk.Entry(self, textvariable=self.reciever)
self.senderoption = tk.StringVar()
self.senderoption.set("---Select One---")
senderdomain = tk.OptionMenu(self, self.senderoption, "---Select One---", "@gmail.com", "@hotmail.com", "@yahoo.com")
self.mail = tk.StringVar()
self.mail.set("")
self.textw = tk.Entry(self, textvariable=self.mail)
button = tk.Button(self, text="Go to the start page",
command=lambda: controller.show_frame(StartPage))
sendbutton = tk.Button(self, text = "Send Mail", command=self.sendmail)
senderl.pack(side="top", anchor="w")
rmail.pack(side="top", anchor="nw")
senderdomain.pack(side="top", anchor="nw")
self.textw.pack(fill="both")
button.pack(side="bottom")
sendbutton.pack(side="bottom")
def sendmail(self):
sent = tk.Label(self, text="Email has been sent")
if self.senderoption.get() == "@gmail.com":
try:
server.sendmail(self.v.get()+self.optionv.get(), self.reciever.get()+self.senderoption.get(), "YES")
print("Success")
sent.pack()
except:
print("Unsuccesful")
print(PageOne.self.v.get())
if __name__ == "__main__":
app = SampleApp()
app.title("PyMail")
app.geometry("400x400")
app.mainloop()
解决方案 1:
从本质上讲,您的问题有一个简单的答案。“如何从对象 X 获取值?”对于任何对象,答案都是一样的:通过询问对象 X 来获取值。为此,您需要做的就是获取对该对象的引用,然后直接访问该属性。
从其他页面访问数据
在您的情况下,代码PageTwo
需要引用,PageOne
以便您可以获取v
变量。
那么,如何获取引用?代码(您从教程或从教程所复制的stackoverflow 答案中复制)旨在简化此操作。每个页面都被赋予对控制器的引用,并且该控制器对每个页面都有引用。因此,您可以要求控制器为您提供对页面的引用。
第一步是在每个类中保存对控制器的引用。有趣的是,您已经在 中执行此操作PageOne
,但您应该在所有页面中执行此操作。确保添加self.controller = controller
每个__init__
方法,如下所示:
class PageTwo(tk.Frame):
def __init__(self, parent, controller):
...
self.controller=controller
...
接下来,我们需要在控制器类中添加一个方法,该方法将返回对页面的引用。将以下函数添加到SampleApp
:
class SampleApp(tk.Tk):
...
def get_page(self, page_class):
return self.frames[page_class]
...
现在,您可以从任何“页面”中访问任何其他“页面”的对象。例如,您可以像这样PageTwo
访问v
变量:PageOne
page1 = self.controller.get_page(PageOne)
page1.v.set("Hello, world")
使用共享数据
更好的解决方案是让您的SampleApp
类创建一组所有页面共享的变量。您可以在该类中创建一个字典,然后使用控制器授予每个页面访问权限。例如:
class SampleApp(tk.Tk):
def __init__(self, *args, **kwargs):
tk.Tk.__init__(self, *args, **kwargs)
self.shared_data = {
"username": tk.StringVar(),
"password": tk.StringVar(),
...
}
然后,您可以从任何类中访问如下数据:
entry1 = tk.Entry(self, textvariable=self.controller.shared_data["username"])
...
username = self.controller.shared_data["username"].get()
这是更好的解决方案,因为您的页面不必知道其他页面是如何实现的。当一个页面依赖于另一个页面的确切实现时,这称为紧耦合。如果页面不需要知道其他页面是如何实现的,这称为松耦合。
松耦合为您提供了更大的灵活性。不是每个页面都与其他页面紧密耦合,而是所有页面都与单个对象紧密耦合:控制器。只要每个页面只知道控制器,每个页面都可以随时更改,而不会影响程序的其余部分。
当然,如果您更改控制器,则必须更改所有页面,但是如果您在控制器设计方面做得很好,那么发生这种情况的可能性就较小,而且即使发生也更容易管理。
解决方案 2:
这与全球框架有关。
如果您在类内创建变量,则该变量将仅存在于该函数内。如果您想将类(或函数)内的变量“转移”到全局框架,则可以使用global。
class firstClass():
global my_var_first
my_var_first = "first variable"
print(my_var_first) # This will work, because the var is in the global frame
class secondClass():
my_var_second = "second variable"
print(my_var_first) # This will work, as the var is in the global frame and not defined in the class
print(my_var_second) # This won't work, because there is no my_var_second in the global frame
为了将记忆可视化,您可以使用pythontutor,因为它将逐步向您展示记忆是如何创建的。
我希望我能帮助你!
编辑
我想我应该补充一点,如果你在类/函数中定义一个与全局框架中的变量同名的变量,它不会删除全局变量。相反,它会在自己的框架中创建一个新的变量(同名)。如果可用,类或函数将始终在其自己的框架中使用该变量。
x = 5
def print_variable():
x = 3
print(x)
print(x)
print_variable()
# OUTPUT:
# 5
# 3