咸蛋之咸鱼翻身 发表于 2020-4-17 17:37:44

求教下面代码的理解

(转换公式:华氏度 = 摄氏度*1.8+32)
class C2F(float):
      "摄氏度转换为华氏度"
      def __new__(cls, arg=0.0):
                return float.__new__(cls, arg * 1.8 + 32)

heidern0612 发表于 2020-4-17 17:55:19

定义一个C2F的类。

在你传入参数后,调用__new__魔法方法。

魔法方法返回当前实例传入值得 * 1.8 + 32给__new__方法,完成初始化。

也就是说,在你传入值后,直接把值修改 * 1.8 + 32给__new__方法,然后完成初始化。

sunrise085 发表于 2020-4-17 17:58:51

估计你是对__new__()和__init__()的区别有疑问吧
看看这个例子
class C2F(float):
    "摄氏度转换为华氏度"
    def __new__(cls, arg=0.0):
      return float.__new__(cls, arg * 1.8 + 32)
    def __init__(self,num=0.0):
      self.C=num
a=C2F(37)
print(a)
print(a.C)
print(type(a))
1.首先用法不同
  __new__()用于创建实例,所以该方法是在实例创建之前被调用,它是类级别的方法,是个静态方法;
   __init__() 用于初始化实例,所以该方法是在实例对象创建后被调用,它是实例级别的方法,用于设置对象属性的一些初始值。
  由此可知,__new__()在__init__() 之前被调用。如果__new__() 创建的是当前类的实例,会自动调用__init__()函数,通过return调用的__new__()的参数cls来保证是当前类实例,如果是其他类的类名,那么创建返回的是其他类实例,就不会调用当前类的__init__()函数。
2.其次传入参数不同:
  __new__()至少有一个参数cls,代表当前类,此参数在实例化时由Python解释器自动识别;
  __init__()至少有一个参数self,就是这个__new__()返回的实例,__init__()在__new__()的基础上完成一些初始化的操作。
3.返回值不同:
  __new__()必须有返回值,返回实例对象;
  __init__()不需要返回值。

咸蛋之咸鱼翻身 发表于 2020-4-18 11:54:27

heidern0612 发表于 2020-4-17 17:55
定义一个C2F的类。

在你传入参数后,调用__new__魔法方法。


你好,我不明白的是为什么不是return arg * 1.8 + 32 而是return float.__new__(cls, arg * 1.8 + 32),应该就是你说的传值给__new__方法我不太懂

咸蛋之咸鱼翻身 发表于 2020-4-18 12:07:48

sunrise085 发表于 2020-4-17 17:58
估计你是对__new__()和__init__()的区别有疑问吧
看看这个例子



经过你这样一说我有点理解__new__的作用,但我还是不太理解 return float.__new__(cls, arg * 1.8 + 32),

heidern0612 发表于 2020-4-18 12:16:50

本帖最后由 heidern0612 于 2020-4-18 12:21 编辑

咸蛋之咸鱼翻身 发表于 2020-4-18 11:54
你好,我不明白的是为什么不是return arg * 1.8 + 32 而是return float.__new__(cls, arg * 1.8 + 32),应 ...

__new__返回的一般都是一个实例对象,而不是一个具体的数值。

你直接返回简单的数值倒不是不可以,只不过简单的数字发现不了什么问题。

假如你定义比较复杂,返回不了简单的数值什么的呢,比如说你需要具体对__new__进行一切功能复杂的操作,如定义函数,定义列表或者属性之类的。

这时候不返回父类的__new__方法大概就操作不了了。

永恒的蓝色梦想 发表于 2020-4-18 12:32:51

咸蛋之咸鱼翻身 发表于 2020-4-18 11:54
你好,我不明白的是为什么不是return arg * 1.8 + 32 而是return float.__new__(cls, arg * 1.8 + 32),应 ...

return arg * 1.8 + 32 会返回 float ,return float.__new__(cls, arg * 1.8 + 32) 会返回一个 C2F 对象

咸蛋之咸鱼翻身 发表于 2020-4-18 14:34:44

heidern0612 发表于 2020-4-18 12:16
__new__返回的一般都是一个实例对象,而不是一个具体的数值。

你直接返回简单的数值倒不是不可以, ...

那请问是不是float.__new__(cls, arg * 1.8 + 32)等于一个实例对象,return float.__new__(cls, arg * 1.8 + 32)等于返回实例对象

heidern0612 发表于 2020-4-18 14:36:21

咸蛋之咸鱼翻身 发表于 2020-4-18 14:34
那请问是不是float.__new__(cls, arg * 1.8 + 32)等于一个实例对象,return float.__new__(cls, arg * 1. ...

下面赋值调用就算实例了,没调用始终都是蓝图。

咸蛋之咸鱼翻身 发表于 2020-4-18 14:55:09

heidern0612 发表于 2020-4-18 14:36
下面赋值调用就算实例了,没调用始终都是蓝图。

有点理解了{:10_301:}
页: [1]
查看完整版本: 求教下面代码的理解