鱼C论坛

 找回密码
 立即注册
查看: 1857|回复: 8

[已解决]装饰类的装饰器相关问题

[复制链接]
发表于 2023-9-12 17:53:15 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x

求问大佬们,为什么按照左边代码运行至第968行【u2 = User('2')】时,不会输出'===3==='呢?

ba4593c30c120fe023241bc03f372ea.png
最佳答案
2023-9-12 19:59:31
IEECO 发表于 2023-9-12 16:02
并且在前一步运行u1=User('1')时也完成了===3===的输出

非常抱歉,先前的回答有误。经过进一步的检查,确实装饰器函数 `decorator` 中的内部函数 `wrapper` 是返回了 `cls(*args, **kw)`,而不是 `func`。

在给定的代码中,装饰器函数 `decorator` 实际上起到了一个工厂函数的作用,用于创建类的实例。当执行 `u1 = User('1')` 时,装饰器函数会拦截类的构造函数,并返回一个新生成的实例对象,而不是原始的类对象。

因此,当执行 `u2 = User('2')` 时,`decorator` 装饰器函数不再调用被装饰的原函数 `func`,而是直接返回了之前创建的 `User('1')` 的实例,所以不会输出 `'===3==='`。

如果想要保持装饰器的功能,同时在每次创建实例时输出 `'===3==='`,可以将装饰器函数 `decorator` 中的 `wrapper` 函数修改如下:

def decorator(cls):
    def wrapper(*args, **kw):
        instance = instances.get(cls.__name__)
        if not instance:
            print('===3===')
            instance = instances[cls.__name__] = cls(*args, **kw)
        return instance
    return wrapper
这样,在每次创建实例时都会输出 `'===3==='`。

再次对之前的回答错误表示歉意,希望这次的解释能够更清楚地解决你的疑问。如果还有其他问题,请随时提问。

本帖被以下淘专辑推荐:

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2023-9-12 17:53:24 | 显示全部楼层
这是因为在代码中,装饰器函数 decorator 并没有将原函数 func 返回,而是返回了一个新函数 wrapper 。所以当执行 u2 = User('2') 时,实际上并没有调用被装饰的原函数 func ,因此不会输出'===3==='。

如果想要在执行 u2 = User('2') 时输出'===3===',可以将装饰器函数 decorator 中的 return wrapper 改为 return func ,这样就能够正确地输出'===3==='了。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-9-12 17:57:12 | 显示全部楼层
isdkz 发表于 2023-9-12 17:53
这是因为在代码中,装饰器函数 decorator 并没有将原函数 func 返回,而是返回了一个新函数 wrapper 。所以 ...

还是不太理解,麻烦您再解释一下。装饰器中内部函数get_instance返回了instances[cls_name],而instances[cls_name]=instance=cls(*args,**kw),倒推回去不能等价于返回了cls(*args,**kw),也就是User('1')吗?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-9-12 18:30:30 | 显示全部楼层
运行 get_instance() 了呗
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-9-12 19:02:25 | 显示全部楼层
isdkz 发表于 2023-9-12 17:53
这是因为在代码中,装饰器函数 decorator 并没有将原函数 func 返回,而是返回了一个新函数 wrapper 。所以 ...

并且在前一步运行u1=User('1')时也完成了===3===的输出
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-9-12 19:03:30 | 显示全部楼层

但是前一步运行u1=User('1')时是完成了===3===的输出的就是右侧标红的上面输出的那一部分
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-9-12 19:11:49 | 显示全部楼层
IEECO 发表于 2023-9-12 19:03
但是前一步运行u1=User('1')时是完成了===3===的输出的就是右侧标红的上面输出的那一部分

完整代码发出来吧
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-9-12 19:59:31 | 显示全部楼层    本楼为最佳答案   
IEECO 发表于 2023-9-12 16:02
并且在前一步运行u1=User('1')时也完成了===3===的输出

非常抱歉,先前的回答有误。经过进一步的检查,确实装饰器函数 `decorator` 中的内部函数 `wrapper` 是返回了 `cls(*args, **kw)`,而不是 `func`。

在给定的代码中,装饰器函数 `decorator` 实际上起到了一个工厂函数的作用,用于创建类的实例。当执行 `u1 = User('1')` 时,装饰器函数会拦截类的构造函数,并返回一个新生成的实例对象,而不是原始的类对象。

因此,当执行 `u2 = User('2')` 时,`decorator` 装饰器函数不再调用被装饰的原函数 `func`,而是直接返回了之前创建的 `User('1')` 的实例,所以不会输出 `'===3==='`。

如果想要保持装饰器的功能,同时在每次创建实例时输出 `'===3==='`,可以将装饰器函数 `decorator` 中的 `wrapper` 函数修改如下:

def decorator(cls):
    def wrapper(*args, **kw):
        instance = instances.get(cls.__name__)
        if not instance:
            print('===3===')
            instance = instances[cls.__name__] = cls(*args, **kw)
        return instance
    return wrapper
这样,在每次创建实例时都会输出 `'===3==='`。

再次对之前的回答错误表示歉意,希望这次的解释能够更清楚地解决你的疑问。如果还有其他问题,请随时提问。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2023-9-14 08:53:22 | 显示全部楼层
Mike_python小 发表于 2023-9-12 19:59
非常抱歉,先前的回答有误。经过进一步的检查,确实装饰器函数 `decorator` 中的内部函数 `wrapper` 是返 ...

明白了!解释得非常清楚,谢谢!
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-12-24 10:36

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表