Python中的“子类”和自我

时间:2013-11-08 04:08:40

标签: python descriptor

注意:我看到我需要更清楚地知道在我要求如何做之前我想要每个属性/描述符/类/方法做什么!我认为我的问题目前无法回答。谢谢大家的帮助。

感谢icktoofay和BrenBarn,我开始理解描述和属性,但现在我有一个稍微难以问的问题:

我现在看到这些是如何工作的:

class Blub(object):
    def __get__(self, instance, owner):
        print('Blub gets ' + instance._blub)
        return instance._blub

    def __set__(self, instance, value):
        print('Blub becomes ' + value)
        instance._blub = value

class Quish(object):
    blub = Blub()

    def __init__(self, value):
        self.blub = value

a = Quish('one')如何运作(产生“Blub成为一个”)但是对这段代码有所了解:

import os
import glob

class Index(object):
    def __init__(self, dir=os.getcwd()):
        self.name = dir    #index name is directory of indexes
        # index is the list of indexes
        self.index = glob.glob(os.path.join(self.name, 'BatchStarted*'))
        # which is the pointer to the index (index[which] == BatchStarted_12312013_115959.txt)
        self.which = 0
        # self.file = self.File(self.index[self.which])
    def get(self):
        return self.index[self.which]
    def next(self):
        self.which += 1
        if self.which < len(self.index):
            return self.get()
        else:
            # loop back to the first
            self.which = 0
            return None
    def back(self):
        if self.which > 0:
            self.which -= 1
        return self.get()

class File(object):
    def __init__(self, file):
        # if the file exists, we'll use it.
        if os.path.isfile(file):
            self.name = file
        # otherwise, our name is none and we return.
        else:
            self.name = None
            return None
        # 'file' attribute is the actual file object
        self.file = open(self.name, 'r')
        self.line = Lines(self.file)

class Lines(object):
    # pass through the actual file object (not filename)
    def __init__(self, file):
        self.file = file
        # line is the list if this file's lines
        self.line = self.file.readlines()
        self.which = 0
        self.extension = Extension(self.line[self.which])
    def __get__(self):
        return self.line[self.which]
    def __set__(self, value):
        self.which = value
    def next(self):
        self.which += 1
        return self.__get__()
    def back(self):
        self.which -= 1
        return self.__get__()

class Extension(object):
    def __init__(self, lineStr):
        # check to make sure a string is passed
        if lineStr:
            self.lineStr = lineStr
            self.line = self.lineStr.split('|')
            self.pathStr = self.line[0]
            self.path = self.pathStr.split('\\')
            self.fileStr = self.path[-1]
            self.file = self.fileStr.split('.')
        else:
            self.lineStr = None                    
    def __get__(self):
        self.line = self.lineStr.split('|')
        self.pathStr = self.line[0]
        self.path = self.pathStr.split('\\')
        self.fileStr = self.path[-1]
        self.file = self.fileStr.split('.')
        return self.file[-1]
    def __set__(self, ext):
        self.file[-1] = ext
        self.fileStr = '.'.join(self.file)
        self.path[-1] = fileStr
        self.pathStr = '\\'.join(self.path)
        self.line[0] = self.pathStr
        self.lineStr = '|'.join(self.line)

首先,这里可能会有一些拼写错误,因为我一直在研究它并让它半开。那不是我的观点。我的观点是,在icktoofay的例子中,没有任何东西传递给Blub()。有什么方法可以做我在这里做的事情,那就是设置一些“自我”属性,并在做了一些处理后,把它带到下一堂课?这会更适合房产吗?

我希望如此:

>>> i = Index()         # i contains list of index files
>>> f = File(i.get())   # f is now one of those files
>>> f.line
'\\\\server\\share\\folder\\file0.txt|Name|Sean|Date|10-20-2000|Type|1'
>>> f.line.extension
'txt'
>>> f.line.extension = 'rtf'
>>> f.line
'\\\\server\\share\\folder\\file0.rtf|Name|Sean|Date|10-20-2000|Type|1'

1 个答案:

答案 0 :(得分:1)

你可以这样做,但问题不在于属性/描述符,而是关于创建提供所需行为的类的更多信息。

所以,当你做f.line时,那是一个对象。当您执行f.line.extension时,即(f.line).extension ---即首先是f.line,然后获取extension所属的f.line属性。

重要的是 f.line无法知道您以后是否会尝试访问其extension 。因此,您不能f.line为“普通”f.line做一件事,为f.line.extension做另一件事。 f.line部分必须相同,extension部分不能改变它。

您似乎想要做的解决方案是让f.line返回某种对象,这些对象在某种程度上看起来或像字符串一样工作,但也允许设置属性并相应地更新自身。具体如何做到这一点取决于你需要多少f.lines表现得像一个字符串,以及你需要多少其他东西。基本上你需要f.line成为一个“看门人”对象,通过像字符串一样处理某些操作(例如,你显然希望它显示作为一个字符串),并处理其他对象自定义方式(例如,您显然希望能够在其上设置extension属性并更新其内容。)

这是一个简单的例子:

class Line(object):
    def __init__(self, txt):
        self.base, self.extension = txt.split('.')

    def __str__(self):
        return self.base + "." + self.extension

现在你可以做到:

>>> line = Line('file.txt')
>>> print line
file.txt
>>> line.extension
'txt'
>>> line.extension = 'foo'
>>> print line
file.foo

但是,请注意我print line,而不只是line。通过编写__str__方法,我定义了print line时发生的行为。但是如果你在没有打印的情况下评估它是“原始的”,你会发现它不是真正的字符串:

>>> line
<__main__.Line object at 0x000000000233D278>

您也可以覆盖此行为(通过定义__repr__),但您想要吗?这取决于您希望如何使用line。关键是你需要决定你希望line在什么情况下做什么,然后制作一个能够做到这一点的课程。