【问题标题】:Python object has no attributePython 对象没有属性
【发布时间】:2019-09-07 18:16:53
【问题描述】:

我遇到了一个奇怪的问题。 我有一个可以运行的测试脚本。现在我在我的主脚本中使用了相同的代码,现在它不起作用,我不知道为什么不这样做。 所以我希望有人能看看出了什么问题。

当我运行脚本时,我会抛出一个错误:

File "E:/Python/Amp_Glade_1.py", line 226, in
  on_volume_scale_value_changed
    self.update_control("volume", x)
File "E:/Python/Amp_Glade_1.py", line 264, in update_control
    self.control[self.name] = self.value
AttributeError: 'GTK_Main' object has no attribute 'control'

所以在:

def on_volume_scale_value_changed(self, object, data=None):
    x = int(self.volume.get_value())
    y = self.volume_step[x]
    self.label_volume.set_text(str(y))
    self.update_control("volume", x)

我打电话:

def update_control(self, name, value):
    self.name = name
    self.value = value
    self.control[self.name] = self.value

而且我看不出什么会导致该错误,因为它在我的测试脚本中工作。 这是 self.control 的字典:

self.control = {"volume": 3, "bass": 0,
                    "middle": 0, "treble": 0,
                    "center": 16, "pre_gain": 0}

这是脚本:

#!/usr/bin/env python
'''
GUI / Amlifier control
Connects to ESP8266 on wifi
Controls Pre-Amp and Amplifier

'''

try:
    import os
 except:
    print ("No os")

try:
    import thread
 except:
print ("No thread")

try:
    import gi
except:
    print ("No gi")

try:
    import sys
except:
    print ("No sys")

try:
    gi.require_version('Gtk', '3.0')
    from gi.repository import GObject, Gtk as Gtk
except:
    print ("No Gtk")

try:
    import socket
except:
    print ("No socket")

try:
    from datetime import datetime, timedelta
except:
    print ("No datetime, timedelta")

try:
    # neede for window.get_xid(), xvimagesink.set_window_handle(),     respectively:
    gi.require_version('GdkX11', '3.0')
    from gi.repository import GdkX11
except:
    print ("No GdkX11")
    sys.exit()


class GTK_Main(object):

    def __init__(self, sock=None):

        ## Create window
        self.gladefile  = ("Amp_2.glade")
        self.builder = Gtk.Builder()
        self.builder.add_from_file(self.gladefile)
        self.builder.connect_signals(self)

        ## Create objects by name frpm gladefile
        self.window = self.builder.get_object("window")
        self.Store = self.builder.get_object("Store")
        self.Mute = self.builder.get_object("Mute")
        self.Load = self.builder.get_object("Load")
        self.Reset = self.builder.get_object("Reset")
        self.bass = self.builder.get_object("bass_scale")
        self.middle = self.builder.get_object("middle_scale")
        self.treble = self.builder.get_object("treble_scale")
        self.volume = self.builder.get_object("volume_scale")
        self.pre_gain = self.builder.get_object("gain")
        self.center = self.builder.get_object("center_scale")
        self.label_bass = self.builder.get_object("label_bass")
        self.label_middle = self.builder.get_object("label_middle")
        self.label_treble = self.builder.get_object("label_treble")
        self.label_volume = self.builder.get_object("label_volume")
        self.label_pre_gain = self.builder.get_object("label_gain")
        self.label_center = self.builder.get_object("label_center")
        self.label_IP = self.builder.get_object("label_IP")
        self.label_Port = self.builder.get_object("label_Port")
        self.label_connected = self.builder.get_object("label_connected")
        self.image_mute = self.builder.get_object("image1")
        self.image_radio_1 = self.builder.get_object("image2")
        self.image_radio_2 = self.builder.get_object("image3")
        self.image_radio_3 = self.builder.get_object("image4")
        self.image_radio_4 = self.builder.get_object("image5")
        self.radiobutten_1 = self.builder.get_object("radiobutton1")
        self.radiobutten_2 = self.builder.get_object("radiobutton2")
        self.radiobutten_3 = self.builder.get_object("radiobutton3")
        self.radiobutten_4 = self.builder.get_object("radiobutton4")

        self.window.show_all()

        ## Set init values
        self.mute_state = False
        self.center.set_value(16)

        self.host = '192.168.0.23'
        self.port = 8888

        ## Create TCP socket
        if sock is None:
            self.sock = socket.socket(socket.AF_INET,
                                  socket.SOCK_STREAM)
        else:
            self.sock = sock

        self.connect(self.host, self.port)

        ## Scale steps
        self.volume_step = (-40, -32. -24, -16, -8, -7,
                        -6, -5, -4, -3, -2, -1, 0)

        self.pre_gain_step = (0, 2, 4, 6, 8, 10, 12, 14,
                          16, 18, 20, 22, 24, 26, 28, 30)

        self.bass_step = (-14, -12, -10, -8, -6, -4, -2, 0,
                      2, 4, 6, 8, 10, 12, 14)

        self.middle_step = (-14, -12, -10, -8, -6, -4, -2, 0,
                      2, 4, 6, 8, 10, 12, 14)

        self.treble = (-14, -12, -10, -8, -6, -4, -2, 0,
                      2, 4, 6, 8, 10, 12, 14)

        self.right_step = (0, -1, -2, -3, -4, -5, -6,
                       -7, -8, -16, -24, -32,
                       -40, -48, -56, -64, -72)

        self.left_step = (0, -1, -2, -3, -4, -5, -6,
                       -7, -8, -16, -24, -32,
                       -40, -48, -56, -64, -72)

        ## Set init scale values
        self.volume.set_value(3)

        ## JSON
        self.control = {"volume": 3, "bass": 0,
                    "middle": 0, "treble": 0,
                    "center": 16, "pre_gain": 0}

        ## Graphic elements to control
        self.GUI_labels = {"volume": self.label_volume,
                       "bass": self.label_bass,
                       "middle": self.label_middle,
                       "treble": self.label_treble,
                       "center": self.label_center,
                       "pre_gain": self.label_pre_gain}

        self.GUI_items = {"volume": self.volume, "bass": self.bass,
                      "middle": self.middle, "treble": self.treble,
                      "center": self.center, "pre_gain": self.pre_gain}

        ## start ping to server
        thread.start_new_thread(self.ping_send(), )

        ## Create handles
    def on_window_destroy(self, obkect, data=None):
        self.sock.close()
        thread.exit()
        Gtk.main_quit()

    def connect(self, host, port):
        try:
            self.sock.connect((self.host, self.port))
            self.label_connected.set_text("Connection:")
            self.label_IP.set_text(self.host)
            self.label_Port.set_text(str(self.port))
        except Exception as e:
            print("No connection. Exceptions is %s:" % (e))
            self.label_connected.set_text("No connection")

        return None

    def on_Mute_released(self, object, data=None):
        if not self.mute_state:
            self.image_mute.set_from_file("circle_red_small.png")
            self.mute_state = True
        else:
            self.image_mute.set_from_file("circle_blue_small.png")
            self.mute_state = False

    def on_radiobutton1_pressed(self, object, data=None):
        self.image_radio_1.set_from_file("circle_green_small.png")
        self.image_radio_2.set_from_file("circle_blue_small.png")
        self.image_radio_3.set_from_file("circle_blue_small.png")
        self.image_radio_4.set_from_file("circle_blue_small.png")

    def on_radiobutton2_pressed(self, object, data=None):
        self.image_radio_2.set_from_file("circle_green_small.png")
        self.image_radio_1.set_from_file("circle_blue_small.png")
        self.image_radio_3.set_from_file("circle_blue_small.png")
        self.image_radio_4.set_from_file("circle_blue_small.png")

    def on_radiobutton3_pressed(self, object, data=None):
        self.image_radio_3.set_from_file("circle_green_small.png")
        self.image_radio_1.set_from_file("circle_blue_small.png")
        self.image_radio_2.set_from_file("circle_blue_small.png")
        self.image_radio_4.set_from_file("circle_blue_small.png")

    def on_radiobutton4_pressed(self, object, data=None):
        self.image_radio_4.set_from_file("circle_green_small.png")
        self.image_radio_1.set_from_file("circle_blue_small.png")
        self.image_radio_2.set_from_file("circle_blue_small.png")
        self.image_radio_3.set_from_file("circle_blue_small.png")

    def on_Reset_released(self, object, data=None):
        pass

    def on_Store_released(self, object, data=None):
        pass

    def on_Load_released(self, object, data=None):
        pass

    def on_center_scale_value_changed(self, object, data=None):
        pass

    def on_volume_scale_value_changed(self, object, data=None):
        x = int(self.volume.get_value())
        y = self.volume_step[x]
        self.label_volume.set_text(str(y))
        self.update_control("volume", x)

    def on_bass_scale_value_changed(self, object, data=None):
        pass

    def on_middle_scale_value_changed(self, object, data=None):
        pass

    def on_treble_scale_value_changed(self, object, data=None):
        pass

    def on_gain_scale_value_changed(self, object, data=None):
        pass

    def ping_send(self):
        now = datetime.now()
        next_ping= now + timedelta(hours = 0, minutes = 0, seconds = 5)
        while True:
            if next_ping <= datetime.now():
                self.mysend("OK")
                now = next_ping
                next_ping = datetime.now() + timedelta(hours = 0, minutes     = 0,
                                               seconds = 5)
            while Gtk.events_pending():
                Gtk.main_iteration()

    def mysend(self, msg):
        totalsent = 0
        MSGLEN = len(msg)
        while totalsent < MSGLEN:
            sent = self.sock.send(msg[totalsent:])
            if sent == 0:
                raise RuntimeError("Socket connection broken")
            totalsent = totalsent + sent

    def update_control(self, name, value):
        self.name = name
        self.value = value
        self.control[self.name] = self.value

    def write_json(self):
        with open("amp.dat", "w+") as jsonFile:
            jsonFile.write(json.dumps(self.control))
            jsonFile.close()

    def load_json(self):
        with open("amp.dat", "r") as jsonFile:
            data = json.load(jsonFile)
            jsonFile.close()
            self.control = json.dumps(data)

    def update_GUI(self):
        for key in self.control.items():
            if key == "gain":
                pass
            else:
                x = self.GUI_labels[key]
                x.set_text(str(value))
                y = self.GUI_items[key]
                y.set_value(value)


GObject.threads_init()
GTK_Main() 
Gtk.main()

所以也许有人可以告诉你怎么了? 我真的很感激。

【问题讨论】:

  • minimal reproducible example 请。您发布的代码格式不正确,并且远不及 minimal.
  • 而且由于我们没有您的 glade 文件,您发布的代码也不完整
  • @Aran-Frey 您是否需要 Gladefile?是的,可以发布更少,但现在就是全部了。我已经指出了基本功能。所以如果你现在没有答案,请继续。
  • 要求一个最小而完整的例子有两个方面。 1)如果它是完整的,那么其他人实际上可以复制(否则它可能是一个猜测)。 2)创建一个最小示例的过程(慢慢删除脚本的部分直到你有足够的时间来证明错误)通常可以让你清楚错误是什么,这样你就可以回答自己出了什么问题。
  • 你说得对,这与林间空地文件无关。您应该删除这部分代码,从而创建一个 minimalcompleteverifiable 示例。

标签: python python-2.7 glade


【解决方案1】:

此代码存在许多问题。我将首先说明我认为是什么问题以及如何解决这个问题,然后再谈其他问题。


这里发生了很多不明显的事情,我想这是因为 GTK 框架隐含地做了一些事情。例如,我注意到 GTK_Main 已实例化,但生成的对象从未保存在任何地方。我的猜测是当你这样做时会发生某种奇怪的嵌套依赖关系

        self.builder = Gtk.Builder()
        ...
        self.builder.connect_signals(self)

并且 this 允许实例化对象与 GTK 事件循环连接。

我还怀疑 builder 以某种方式期望此对象实现像 on_volume_scale_value_changed 这样的方法,这些方法将在音量缩放值更改时执行。

如果是这样,那么下面的代码会导致错误

        ## Set init scale values
        self.volume.set_value(3)

        ## JSON
        self.control = {"volume": 3, "bass": 0,
                    "middle": 0, "treble": 0,
                    "center": 16, "pre_gain": 0}

这是因为设置音量会触发on_volume_scale_value_changed 被调用,它会尝试访问self.control,但您直到音量设置之后才定义self.control。颠倒顺序应该解决问题。

一般来说,你应该

  1. 总是在执行任何逻辑(例如设置音量)之前设置数据成员(例如 self.control,因为它只是保存数据),并且
  2. 注意事物的调用顺序,以免陷入这样的循环。

话虽如此,你将遇到的下一个问题是:

    def load_json(self):
        with open("amp.dat", "r") as jsonFile:
            data = json.load(jsonFile)
            jsonFile.close()
            self.control = json.dumps(data)

self.control 应该是一个字典,但您将它设置为json.dumps 的输出,这将是一个字符串。因此,如果您调用该函数,您的代码将停止工作。你应该做self.control = json.load(jsonFile)

此外,在上下文管理器(with 块)中,文件在退出时会自动关闭,因此显式调用 jsonFile.close() 充其量是多余的 - 最坏的情况是它可能会在退出 with 块时引发错误,因为该文件已关闭。


所有的

try:
    import sys
 except:
    print ("No sys")

由于以下原因非常糟糕

  1. 你正在做一个毯子except,而不是捕获一个特定的错误。我假设你正试图抓住ImportError,所以你应该这样做except ImportError。您知道可能会引发哪些其他类型的异常,因此您不知道什么会被吞下并使您的系统处于无效状态。这让我想到了
  2. 您正在尝试导入一个模块,但如果导入失败,您只是打印它不存在并继续前进。在ossys 的情况下,这很好,因为您从不使用os(尽管我会说您应该避免导入您不使用的东西)并且sys 仅用于sys.exit()(这会触发因为sys 丢失了所以退出),但是如果datetime 无法导入怎么办?您在代码中使用了datetime,因此当您尝试使用datetime 时,您的代码可能会运行不知道多久然后突然失败。一般而言,尽早进行描述性失败要比让您的系统处于随时可能失败的损坏状态要好得多。
  3. 您执行此操作的大部分导入来自标准库。如果缺少标准库中的某些内容,则存在问题。尤其是sys...我认为没有sys,python 无法启动。
  4. 如果模块不存在,它将引发ImportError 并告诉用户它不存在,然后停止代码。这就是您想要的,因为如果缺少 GTK,您将无法运行您的代码。

道德:永远不要吞下错误并继续前进,因为这会使您的系统处于崩溃状态,任何时候事情都可能无缘无故地失败。

【讨论】:

  • 1: self.builder = Gtk.Builder 就是实例化为self.builder的Gtk.builder,所以被保存了。 self.builder.connect_signals(self) 是连接到对象信号的构建器,它在 Gladefile 中并且现在是对象的一部分,因为 self.builder 已经加载了 Gladefile。它是如何与 Glade 文件一起工作的。 2:是的,尝试:导入可能更优雅。从我的 python2.7 不一定安装所有模块的时候开始,它只是一个旧习惯。这只是我自己的通知者。 os 仅在我需要时才是 importet。
  • 3:关闭 jsonfile 似乎没有问题。文件中的数据已经存储在数据中。只要该函数在使用中,来自 jsonfile 的数据就会保存在变量“data”中。
  • 4: 刚刚做了一个测试 where def update_control: is move up in the script before, the function get call 1. time 并没有区别,因为函数已经编码,它们也产生了错误在启动时。但是所有这些功能都在正常工作。当它们在使用时,它们不会产生任何错误。所以在一天结束时,我会吞下这些错误,因为似乎没有人能够解释为什么它们突然出现,当他们没有在我测试函数的测试脚本中这样做时。
【解决方案2】:

我希望这只是一个错字onkect,而不是在实际代码中?

 def on_window_destroy(self, obkect, data=None):
         self.sock.close()
         thread.exit()
         Gtk.main_quit()

【讨论】:

  • 是的错字,但不是我的问题的原因。如果它只有... :-)
【解决方案3】:

这一定是编译器中的某种错误。 经过大量测试,注释掉了几个区域,制作了新的变量 它仍然会在启动时产生错误。 但是,该功能按预期工作。

【讨论】: