Peteryo01223 发表于 2021-3-25 10:05:07

Py 二问46课作业:魔法方法,描述符

原题:
按要求编写描述符 MyDes:记录指定变量的读取和写入操作,并将记录以及触发时间保存到文件:record.txt
要求程序实现如下:
>>> class Test:
      x = Record(10, 'x')
      y = Record(8.8, 'y')

>>> test = Test()
>>> test.x
10
>>> test.y
8.8
>>> test.x = 123
>>> test.x = 1.23
>>> test.y = "I love FishC.com!"
>>>
以下为FishC的答案,我自己注释的:
import time # 引用Python中的现成时间模块

class Record: # 定义一个类
    def __init__(self, initval=None, name=None): # 对实例的属性进行初使化
      self.val = initval # 创建类的实例,这个被创建的实例(value)本身被赋值
      self.name = name # 创建类的实例,这个被创建的实例(name)本身被赋值
      self.filename = "record.txt" # 创建类的实例,这个被创建的实例(文件名)本身被赋值

    def __get__(self, instance, owner): # 重新定义描述符之一。调用一个属性时,__get__时被触发。
      with open(self.filename, 'a', encoding='utf-8') as f:
      # 'a' 打开写入,如文件存在,则追加到文件的末尾
      # 告诉 python 解释器要按照utf-8编码的方式来读取程序
            f.write("%s 变量于北京时间 %s 被读取,%s = %s\n" % \
                  (self.name, time.ctime(), self.name, str(self.val)))
      # ctime()方法,将从纪元(epoch)以"秒为单位"的时间转换为表示本地时间的字符串。
      return self.val

    def __set__(self, instance, value): # 重新定义描述符之一。为一个属性赋值时,__set__时被触发。
      filename = "%s_record.txt" % self.name
      with open(self.filename, 'a', encoding='utf-8') as f:
            f.write("%s 变量于北京时间 %s 被修改, %s = %s\n" % \
                  (self.name, time.ctime(), self.name, str(value)))
      self.val = value
问题:
最后一句,为什么还要再给 self.val 赋值一次?烦请尽量详解。谢谢。
self.val = value

Hoiste 发表于 2021-3-25 10:20:39

因为__get__方法是调用,所以return一个返回值就好了,而__set__方法相当于一个赋值操作,没有返回值,但你需要改变属性val的值以便后续的调用,当然需要进行赋值。

Hoiste 发表于 2021-3-25 10:23:36

简单来说,前面的语句只是打印了什么变量被赋值了,但实际上这个变量本身,即val并没有被改动,只有self.val = value这一步才改变了val的值。

Peteryo01223 发表于 2021-3-25 14:03:57

Hoiste 发表于 2021-3-25 10:23
简单来说,前面的语句只是打印了什么变量被赋值了,但实际上这个变量本身,即val并没有被改动,只有self.va ...

谢谢
页: [1]
查看完整版本: Py 二问46课作业:魔法方法,描述符