鱼C论坛

 找回密码
 立即注册
查看: 1623|回复: 6

[已解决]43魔法方法 问题求助

[复制链接]
发表于 2022-5-16 19:50:00 | 显示全部楼层 |阅读模式

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

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

x
动动手最后一题
1. 定义一个单词(Word)类继承自字符串,重写比较操作符(参考自学:Python 魔法方法详解),当两个 Word 类对象进行比较时,根据单词的长度来进行比较大小。
加分要求:实例化时如果传入的是带空格的字符串,则取第一个空格前的单词作为参数。

我的答案
class Word(str):
    def __init__(self,str1):
        self.str1 = str1.strip()
        if ' ' in self.str1:
            self.str1 = self.str1.split(' ')[0]

    def __lt__(self, other):
        return len(self) < len(other)

    def __le__(self, other):
        return len(self) <=len(other)

    def __eq__(self, other):
        return len(self) == len(other)

    def __ne__(self, other):
        return len(self) != len(other)

    def __gt__(self, other):
        return len(self) > len(other)

    def __ge__(self, other):
        return len(self) >= len(other)

因为没有怎么思考, 没有用到__new__, 但是我发现我用 __init__ 好像也没什么太大问题呀
非要用 __new__吗。。。
最佳答案
2022-5-17 15:28:59
爱笑的无赖 发表于 2022-5-16 19:52
这是小甲鱼的答案  



有差别的,那些都不是问题,这里主要原因是:

因为 Str 是不可变类型,虽然你在 __init__ 中重新创建了一个 str1

但是对于 Word 类来说,self 实例本身永远是你第一次实例化时候传入的字母

即你可以用你代码试着运行:
w1 = Word("FishC Python")
w2 = Word("Lalala")
print(w1 > w2)

你的代码肯定返回 True,而甲鱼哥代码返回的是 False

就是因为甲鱼哥重写了 __new__ 将 self 变为 FishC 而不是 FishC Python

但是你是重写 __init__ ,有因为 str 是不可变的,所以导致你的 self 是传入时的 self,即 FishC Python

你这里创建的 str1 是无用的,在比较运算符中比较的是 self 实例对象的内容,而不是你新定义的  str1 字符串内容

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

使用道具 举报

 楼主| 发表于 2022-5-16 19:52:37 | 显示全部楼层
这是小甲鱼的答案  
class Word(str):
'''存储单词的类,定义比较单词的几种方法'''

    def __new__(cls, word):
        # 注意我们必须要用到 __new__ 方法,因为 str 是不可变类型
        # 所以我们必须在创建的时候将它初始化
        if ' ' in word:
            print "Value contains spaces. Truncating to first space."
            word = word[:word.index(' ')] #单词是第一个空格之前的所有字符
        return str.__new__(cls, word)

    def __gt__(self, other):
        return len(self) > len(other)
    def __lt__(self, other):
        return len(self) < len(other)
    def __ge__(self, other):
        return len(self) >= len(other)
    def __le__(self, other):
        return len(self) <= len(other)


我的理解是, 如果像小甲鱼那样  用到 word = word[:word.index(' ')]的话, 应该是要用__new__的。 因为本身是修改到了string

我因为用str.split()先把字符串变成列表, 在取列表第一位, 这样子是不需要用__new__
这样理解对吗?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2022-5-16 20:45:22 | 显示全部楼层
个人觉得,__init__()和__new__()方法的本质区别在于__init__()没有return返回值,只能做一些变量的定义;而__new_是有返回值的。
有些需要返回值的时候,你如果不在__new__里面定义就要再单独定义一个方法来返回。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2022-5-17 15:28:59 | 显示全部楼层    本楼为最佳答案   
爱笑的无赖 发表于 2022-5-16 19:52
这是小甲鱼的答案  



有差别的,那些都不是问题,这里主要原因是:

因为 Str 是不可变类型,虽然你在 __init__ 中重新创建了一个 str1

但是对于 Word 类来说,self 实例本身永远是你第一次实例化时候传入的字母

即你可以用你代码试着运行:
w1 = Word("FishC Python")
w2 = Word("Lalala")
print(w1 > w2)

你的代码肯定返回 True,而甲鱼哥代码返回的是 False

就是因为甲鱼哥重写了 __new__ 将 self 变为 FishC 而不是 FishC Python

但是你是重写 __init__ ,有因为 str 是不可变的,所以导致你的 self 是传入时的 self,即 FishC Python

你这里创建的 str1 是无用的,在比较运算符中比较的是 self 实例对象的内容,而不是你新定义的  str1 字符串内容

评分

参与人数 1荣誉 +5 鱼币 +5 贡献 +3 收起 理由
爱笑的无赖 + 5 + 5 + 3

查看全部评分

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

使用道具 举报

 楼主| 发表于 2022-5-18 10:37:36 | 显示全部楼层
Twilight6 发表于 2022-5-17 15:28
有差别的,那些都不是问题,这里主要原因是:

因为 Str 是不可变类型,虽然你在 __init__ 中重新创 ...

谢版主大大。 受教了
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2022-5-18 10:39:52 | 显示全部楼层
67193114 发表于 2022-5-16 20:45
个人觉得,__init__()和__new__()方法的本质区别在于__init__()没有return返回值,只能做一些变量的定义; ...

还是有点区别的。 new方法是走在init前面的第一个魔法方法。 new方法走了之后,参数才会被传入init。  改变字符串的操作貌似除了new方法外, 自己定义方法貌似不能改变的。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 1 反对 0

使用道具 举报

发表于 2022-5-20 18:47:48 | 显示全部楼层
爱笑的无赖 发表于 2022-5-18 10:39
还是有点区别的。 new方法是走在init前面的第一个魔法方法。 new方法走了之后,参数才会被传入init。  改 ...

嗯呐!
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

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

本版积分规则

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

GMT+8, 2024-11-18 05:52

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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