我怎样才能让这些计数器装饰器为我的 in class 方法工作
how can I make these counter decorator work for my in class method
我是装饰者的新手,相信我在很大程度上理解他们的 purpose/the 大体要点。不幸的是,这个装饰器及其在 class 中的使用存在一些问题。我正在尝试使用 profile + str(<counter value>)
创建的新密钥填充字典 _profile_dict = {}
。我不想实例化 class 并将其设置为我的 if __name__ == '__main'
语句中的变量,因此我尝试使用 create_profiles
函数执行 class。这是我的代码,
_profile_dict = {}
class Profile:
def __init__(self):
self.first_name = input('first name: ')
self.last_name = input('last name: ')
self.email = input('email: ')
self.address1 = input('primary address: ')
self.address2 = input('secondary address: ')
self.city = input('city: ')
self.country = input('country: ')
self.province = input('province: ')
self.zip = input('zip: ')
self.phone = input('phone number:')
self.cc_name = input('Name on credit card: ')
self.cc_num = input('Credit card number: ')
self.exp_month = input('card expiration month: ')
self.exp_year = input('card expiration year: ')
self.cvv = input('cvv: ')
self.json_obj = {"utf8": "✓",
"_method": "",
"authenticity_token": "",
"previous_step": "",
"checkout[email]": self.email,
"checkout[buyer_accepts_marketing]": 1,
"checkout[shipping_address][first_name]": self.first_name,
"checkout[shipping_address][last_name]": self.last_name,
"checkout[shipping_address][address1]": self.address1,
"checkout[shipping_address][address2]": self.address2,
"checkout[shipping_address][city]": self.city,
"checkout[shipping_address][country]": self.country,
"checkout[shipping_address][province]": self.province,
"checkout[shipping_address][zip]": self.zip,
"checkout[shipping_address][phone]": self.phone,
"step": ""}
def counter(self, func):
def wrapper():
wrapper.counter += 1
return func(wrapper.counter)
wrapper.counter = 0
return wrapper
@counter
def _populate_profile_dict(self, count):
profile = 'profile'
_profile_dict[profile + str(count)] = self.json_obj
print(_profile_dict)
def create_profiles():
Profile()._populate_profile_dict()
if __name__ == "__main__":
create_profiles()
目前,我收到以下错误,
Traceback (most recent call last):
File "C:/Users/Carsten/Profile_Gen/src/config.py", line 6, in <module>
class Profile:
File "C:/Users/Carsten/Profile_gen/src/config.py", line 49, in Profile
@counter
TypeError: counter() missing 1 required positional argument: 'func'
尽管从我刚刚在这里读到的 Python decorators in classes 来看,我认为这是不可能的。是否有可能完成我想用@class方法实现的目标?
只是将最后3个functions/methods改成这样,仍然没有输出:
@classmethod
def counter(cls, func):
def wrapper():
wrapper.counter += 1
return func(wrapper.counter)
wrapper.counter = 0
return wrapper
def _populate_profile_dict(self, count):
profile = 'profile'
_profile_dict[profile + str(count)] = self.json_obj
print(_profile_dict)
#def create_profiles():
# Profile()._populate_profile_dict()
if __name__ == "__main__":
Profile()._populate_profile_dict =
Profile.counter(Profile._populate_profile_dict)
如有任何帮助,我们将不胜感激。感谢您的时间。我觉得装饰器以及我议程上的其他一些事情肯定会帮助我进入 python 的更多中间阶段。
如果你不需要使用装饰器,那么你可以测量长度
你的字典:
def _populate_profile_dict(self):
_profile_dict['profile' + str(len(_profile_dict))] = self.json_obj
print(_profile_dict)
让我暂时忽略你的第二次尝试,因为第一个更接近工作:
def counter(self, func):
def wrapper():
wrapper.counter += 1
return func(wrapper.counter)
wrapper.counter = 0
return wrapper
@counter
def _populate_profile_dict(self, count):
# ...
这不起作用的原因是装饰器语法试图这样做:
_populate_profile_dict = counter(_populate_profile_dict)
… 但是 counter
被定义为接受两个参数,而不是一个,self
和一个函数。这里显然没有self
。事实上,甚至 class 本身都不存在,更不用说任何方法了。你只是用函数调用它,所以这就是它需要的。
如果您只删除 self
参数,装饰器部分将正常工作:
def counter(func):
def wrapper():
wrapper.counter += 1
return func(wrapper.counter)
wrapper.counter = 0
return wrapper
现在,counter
不再用作 class 实例上的方法——但这没关系,因为您永远不会打算那样使用它。
与此同时,在 class 仍在定义期间,它作为一个普通函数非常有用。这正是您想要的使用方式。
counter
最终看起来像 Profile
实例的 public 方法,这可能有点令人困惑,尽管它实际上不能那样调用,这非常令人困惑。但是您可以通过将其重命名为 _counter
或在 class 定义的最后执行 del counter
来解决该问题。
无论如何,装饰器部分可以,但还有第二个问题:
def wrapper():
这定义了一个没有参数的函数。但是你正在尝试创建可以作为普通实例方法调用的东西,所以它会被传递一个 self
参数。您需要接受 self
参数,并且还需要将其传递给包装函数。所以:
def counter(func):
def wrapper(self):
wrapper.counter += 1
return func(self, wrapper.counter)
# etc.
现在,一切都按照(我认为)您的预期进行。
说到这里,这段代码有点奇怪:
def create_profiles():
Profile()._populate_profile_dict()
这将创建一个 Profile
实例,对其调用私有方法,然后丢弃该实例。我不确定你为什么要这样做。当然,它向私有全局 _profile_dict
添加了一个 profile1
条目,但这对您有何帮助?
现在进行第二次尝试:
@classmethod
def counter(cls, func):
def wrapper():
wrapper.counter += 1
return func(wrapper.counter)
wrapper.counter = 0
return wrapper
现在您已成功创建 public classmethod
,您尝试将其显式调用为 classmethod
,这是完美的:
Profile()._populate_profile_dict =
Profile.counter(Profile._populate_profile_dict)
但它还有其他多个问题:
- 尝试像这样将作业分成两行是
SyntaxError
。
Profile.counter(Profile._populate_profile_dict)
return是一个新函数,你不调用它,所以它只是将函数本身赋值。
- 当您 do 调用它时,它期望有两个参数,一个
self
和一个 count
,但您只传递了 count
.在这里,不清楚您想从哪里获得 self
。也许你想在 Profile
的实例上调用它,所以你得到一个 self
并且可以通过它?
- 您正在将值分配给您刚刚创建并准备丢弃的
Profile
实例的实例属性,这没有明显的效果。
- 该实例属性与您要调用的方法同名——这是合法的,并且不会在此处引起任何特殊问题,但它非常令人困惑。
我能看到的最接近您尝试做的合理事情是使 counter
return 某些可调用的方法,如上所示:
@classmethod
def counter(cls, func):
def wrapper(self):
wrapper.counter += 1
return func(self, wrapper.counter)
wrapper.counter = 0
return wrapper
…然后使用结果进行 monkeypatch Profile
,然后在 Profile
实例上调用该方法,如下所示:
if __name__ == "__main__":
Profile._populate_profile_dict = Profile.counter(Profile._populate_profile_dict)
Profile()._populate_profile_dict()
现在可以使用了,但我不确定它是否是您想要的。
我是装饰者的新手,相信我在很大程度上理解他们的 purpose/the 大体要点。不幸的是,这个装饰器及其在 class 中的使用存在一些问题。我正在尝试使用 profile + str(<counter value>)
创建的新密钥填充字典 _profile_dict = {}
。我不想实例化 class 并将其设置为我的 if __name__ == '__main'
语句中的变量,因此我尝试使用 create_profiles
函数执行 class。这是我的代码,
_profile_dict = {}
class Profile:
def __init__(self):
self.first_name = input('first name: ')
self.last_name = input('last name: ')
self.email = input('email: ')
self.address1 = input('primary address: ')
self.address2 = input('secondary address: ')
self.city = input('city: ')
self.country = input('country: ')
self.province = input('province: ')
self.zip = input('zip: ')
self.phone = input('phone number:')
self.cc_name = input('Name on credit card: ')
self.cc_num = input('Credit card number: ')
self.exp_month = input('card expiration month: ')
self.exp_year = input('card expiration year: ')
self.cvv = input('cvv: ')
self.json_obj = {"utf8": "✓",
"_method": "",
"authenticity_token": "",
"previous_step": "",
"checkout[email]": self.email,
"checkout[buyer_accepts_marketing]": 1,
"checkout[shipping_address][first_name]": self.first_name,
"checkout[shipping_address][last_name]": self.last_name,
"checkout[shipping_address][address1]": self.address1,
"checkout[shipping_address][address2]": self.address2,
"checkout[shipping_address][city]": self.city,
"checkout[shipping_address][country]": self.country,
"checkout[shipping_address][province]": self.province,
"checkout[shipping_address][zip]": self.zip,
"checkout[shipping_address][phone]": self.phone,
"step": ""}
def counter(self, func):
def wrapper():
wrapper.counter += 1
return func(wrapper.counter)
wrapper.counter = 0
return wrapper
@counter
def _populate_profile_dict(self, count):
profile = 'profile'
_profile_dict[profile + str(count)] = self.json_obj
print(_profile_dict)
def create_profiles():
Profile()._populate_profile_dict()
if __name__ == "__main__":
create_profiles()
目前,我收到以下错误,
Traceback (most recent call last):
File "C:/Users/Carsten/Profile_Gen/src/config.py", line 6, in <module>
class Profile:
File "C:/Users/Carsten/Profile_gen/src/config.py", line 49, in Profile
@counter
TypeError: counter() missing 1 required positional argument: 'func'
尽管从我刚刚在这里读到的 Python decorators in classes 来看,我认为这是不可能的。是否有可能完成我想用@class方法实现的目标?
只是将最后3个functions/methods改成这样,仍然没有输出:
@classmethod
def counter(cls, func):
def wrapper():
wrapper.counter += 1
return func(wrapper.counter)
wrapper.counter = 0
return wrapper
def _populate_profile_dict(self, count):
profile = 'profile'
_profile_dict[profile + str(count)] = self.json_obj
print(_profile_dict)
#def create_profiles():
# Profile()._populate_profile_dict()
if __name__ == "__main__":
Profile()._populate_profile_dict =
Profile.counter(Profile._populate_profile_dict)
如有任何帮助,我们将不胜感激。感谢您的时间。我觉得装饰器以及我议程上的其他一些事情肯定会帮助我进入 python 的更多中间阶段。
如果你不需要使用装饰器,那么你可以测量长度 你的字典:
def _populate_profile_dict(self):
_profile_dict['profile' + str(len(_profile_dict))] = self.json_obj
print(_profile_dict)
让我暂时忽略你的第二次尝试,因为第一个更接近工作:
def counter(self, func):
def wrapper():
wrapper.counter += 1
return func(wrapper.counter)
wrapper.counter = 0
return wrapper
@counter
def _populate_profile_dict(self, count):
# ...
这不起作用的原因是装饰器语法试图这样做:
_populate_profile_dict = counter(_populate_profile_dict)
… 但是 counter
被定义为接受两个参数,而不是一个,self
和一个函数。这里显然没有self
。事实上,甚至 class 本身都不存在,更不用说任何方法了。你只是用函数调用它,所以这就是它需要的。
如果您只删除 self
参数,装饰器部分将正常工作:
def counter(func):
def wrapper():
wrapper.counter += 1
return func(wrapper.counter)
wrapper.counter = 0
return wrapper
现在,counter
不再用作 class 实例上的方法——但这没关系,因为您永远不会打算那样使用它。
与此同时,在 class 仍在定义期间,它作为一个普通函数非常有用。这正是您想要的使用方式。
counter
最终看起来像 Profile
实例的 public 方法,这可能有点令人困惑,尽管它实际上不能那样调用,这非常令人困惑。但是您可以通过将其重命名为 _counter
或在 class 定义的最后执行 del counter
来解决该问题。
无论如何,装饰器部分可以,但还有第二个问题:
def wrapper():
这定义了一个没有参数的函数。但是你正在尝试创建可以作为普通实例方法调用的东西,所以它会被传递一个 self
参数。您需要接受 self
参数,并且还需要将其传递给包装函数。所以:
def counter(func):
def wrapper(self):
wrapper.counter += 1
return func(self, wrapper.counter)
# etc.
现在,一切都按照(我认为)您的预期进行。
说到这里,这段代码有点奇怪:
def create_profiles():
Profile()._populate_profile_dict()
这将创建一个 Profile
实例,对其调用私有方法,然后丢弃该实例。我不确定你为什么要这样做。当然,它向私有全局 _profile_dict
添加了一个 profile1
条目,但这对您有何帮助?
现在进行第二次尝试:
@classmethod
def counter(cls, func):
def wrapper():
wrapper.counter += 1
return func(wrapper.counter)
wrapper.counter = 0
return wrapper
现在您已成功创建 public classmethod
,您尝试将其显式调用为 classmethod
,这是完美的:
Profile()._populate_profile_dict =
Profile.counter(Profile._populate_profile_dict)
但它还有其他多个问题:
- 尝试像这样将作业分成两行是
SyntaxError
。 Profile.counter(Profile._populate_profile_dict)
return是一个新函数,你不调用它,所以它只是将函数本身赋值。- 当您 do 调用它时,它期望有两个参数,一个
self
和一个count
,但您只传递了count
.在这里,不清楚您想从哪里获得self
。也许你想在Profile
的实例上调用它,所以你得到一个self
并且可以通过它? - 您正在将值分配给您刚刚创建并准备丢弃的
Profile
实例的实例属性,这没有明显的效果。 - 该实例属性与您要调用的方法同名——这是合法的,并且不会在此处引起任何特殊问题,但它非常令人困惑。
我能看到的最接近您尝试做的合理事情是使 counter
return 某些可调用的方法,如上所示:
@classmethod
def counter(cls, func):
def wrapper(self):
wrapper.counter += 1
return func(self, wrapper.counter)
wrapper.counter = 0
return wrapper
…然后使用结果进行 monkeypatch Profile
,然后在 Profile
实例上调用该方法,如下所示:
if __name__ == "__main__":
Profile._populate_profile_dict = Profile.counter(Profile._populate_profile_dict)
Profile()._populate_profile_dict()
现在可以使用了,但我不确定它是否是您想要的。