• 2015/9/29 Python基础(20):类的授权


    类的授权
    1.包装
    包装在Python编程世界中时经常会被提到的一个术语。它是一个通用的名字,意思是对一个已存在的对象进行包装,不管它是数据类型,还是一段代码,可以是对一个已存在的对象,增加新的,删除不要的,或者修改其他已存在的功能。
    在Python2.2以前,从Python的标准类型子类化或派生类都是不允许的,即使你现在可以这么做,这种做法也并不多。你可以包装任何类型作为一个类的核心成员,以使新对象的行为模仿你想要的数据类型中已存在的行为,并且去掉你不希望存在的行为;它可能会要做一些额外的事情。这就是“包装类型”。在附录中,我们还将讨论如何扩充Python,包装的另一种形式。
    包装包括定义一个类,它的实例拥有标准类型的核心行为,但它也通过新的或最新的功能,甚至可能通过访问实际数据的不同方法得到提高。
    还可以包装类,但这不会有太大的用途,因为已经有拥有操作对象的机制,并且在之前已经描述过,我们采用派生对标准类型有对其进行包装的方式。

    2.实现授权
    授权是包装的一个特性,可用于简化处理有关dictating功能,采用已存在的功能达到最大限度的代码重用。
    包装一个类型通常是对已存在的类型的一些定制。我们在前面提到过,这种方法可以新建,修改或者删除原有产品的功能。其它的则保持原样,或者保留已存功能和行为。授权的过程,即是所有更新的功能都是有新类的某部分来处理,但已存在的功能就授权给对象的默认属性。
    实现授权的关键点就是覆盖__getattr__()方法,在代码中包含一个对getattr()内建函数的调用。特别地,调用getattr()以得到默认对象属性(数据属性或者方法)并返回它以便访问或调用。特殊方法__getattr__()的工作方式是,当搜索一个属性时,任何局部对象(定制的对象)首先被找到。如果搜索失败了,则__getattr__()会被调用。然后调用getattr()得到一个对象的默认行为。
    也就是说,引用一个属性时,Python解释器将试着在局部名称空间中查找那个名字,比如一个自定义的方法或者局部实例属性。如果没有在局部字典中找到,则搜索类名称空间,以防一个类属性被访问。最后,如果两类搜索都失败了,搜索对原对象开始授权请求,此时,__getattr__()会被调用。

    包装对象的简例
    这个类几乎可以包装任何对象,提供基本功能。比如使用repr()和str()来处理字符串表示法, 定制由get()方法处理,它删除包装并且返回原始对象,所以保留的功能都授权给对象的本地属性,在必要时,可又__getattr__()获得。

    >>> class WrapMe(object):
      def __init__(self, obj):
        self.__data = obj
      def get(self):
        return self.__data
      def __repr__(self):
        return `self.__data`
      def __str__(self):
        return str(self.__data)
      def __getattr__(self, attr):
        return getattr(self.__data, attr)

    我们使用复数来举第一个例子,因为复数有数据属性及conjugate()内建方法。

    >>> wrappedComplex = WrapMe(3.3+1.2j)
    >>> wrappedComplex
    (3.3+1.2j)
    >>> wrappedComplex.real
    3.3
    >>> wrappedComplex.imag
    1.2
    >>> wrappedComplex.conjugate()
    (3.3-1.2j)
    >>> wrappedComplex.get()
    (3.3+1.2j)

    一旦我们创建了包装的对象类型,只要由交互解释器调用repr(),就可以得到一个字符串表示,然后我们访问了复数的三种属性,而这在类中一种都没有定义。对这种属性的访问,是通过__getattr__()方法,授权给对象。最终调用的get()方法没有授权,因为它是为我们对象单独定义的。
    然后是一个列表的例子

    >>> wrappedList = WrapMe([123, 'foo', 45.67])
    >>> wrappedList.append('bar')
    >>> wrappedList.append(123)
    >>> wrappedList
    [123, 'foo', 45.67, 'bar', 123]
    >>> wrappedList.index(45.67)
    2
    >>> wrappedList.count(123)
    2
    >>> wrappedList.pop()
    123
    >>> wrappedList
    [123, 'foo', 45.67, 'bar']


    注意,尽管我们正在我们的例子中使用实例,它们展示的行为与他们包装的数据类型非常相似。然后需要明白,只有已存在的属性是在此代码中授权的。
    但是也不是所有的行为都能被访问,比如:

    >>> wrappedList[3]
    
    Traceback (most recent call last):
    File "<pyshell#29>", line 1, in <module>
    wrappedList[3]
    TypeError: 'WrapMe' object does not support indexing

    对于列表的索引切片操作,它是内建于类型中的,而不是像append()方法那样作为属性存在的。所以不能被访问。从另一个角度来说,切片操作符是序列类型的一部分,并不是通过__getitem__()这样的特殊方法来实现的。
    我们有一种“作弊”的方法,访问实际对象,然后用它的切片能力:

    >>> realList = wrappedList.get()
    >>> realList[3]
    'bar'

    这就是我们实现get()方法的原因了,我们可以从访问调用中直接访问对象的属性,而忽略局部变量。
    这里是简单包装类的例子。我们还刚开始接触使用类型模拟来进行类自定义。你将会发现你可以进行无限多的改进,来进一步增加代码的用途。

    更新简单的包裹类
    创建时间,修改时间,及访问时间是文件的几个常见属性。我们将给一个类添加时间属性,创建时间('ctime')是实例化的时间,修改时间('mtime')是核心数据升级的时间,而访问时间('atime')是最后一次对象数据值被获取或者属性被访问的时间戳。
    我们更新前面定义的类,创建一个模块twrapme.py
    代码如下:

    from time import time, ctime
    class TimeWrapMe(object):
        def __init__(self, obj):
            self.__data = obj
            self.__ctime = self.__mtime = 
                   self.__atime = time() 
    
        def get(self):
            self.__atime = time()
            return self.__data
    
        def gettimeval(self, t_type):
            if not isinstance(t_type, str) or 
               t_type[0] not in 'cma':
                raise TypeError, 
                      "argument of 'c', 'm', or 'a' req'd"
            return getattr(self, '_%s__%stime' % 
                           (self.__class__.__name__, t_type[0]))
            
        def gettimestr(self, t_type):
            return ctime(self.gettimeval(t_type))
    
        def set(self, obj):
            self.__data = obj
            self.__mtime = self.atime = time()
        
        def __repr__(self):
            self.__atime = time()
            return `self.__data`
    
        def __str__(self):
            self.__atime = time()
            return str(self.__data)
    
        def __getattr__(self, attr):
            self.__atime = time()
            return getattr(self.__data, attr)

    测试如下:

    >>> TimeWrappedObj = TimeWrapMe(123)
    >>> TimeWrappedObj.gettimestr('c')
    'Tue Sep 29 17:25:11 2015'
    >>> TimeWrappedObj.gettimestr('m')
    'Tue Sep 29 17:25:11 2015'
    >>> TimeWrappedObj.gettimestr('a')
    'Tue Sep 29 17:25:11 2015'
    >>> TimeWrappedObj
    123
    >>> TimeWrappedObj.gettimestr('c')
    'Tue Sep 29 17:25:11 2015'
    >>> TimeWrappedObj.gettimestr('m')
    'Tue Sep 29 17:25:11 2015'
    >>> TimeWrappedObj.gettimestr('a')
    'Tue Sep 29 17:25:26 2015'
    >>> TimeWrappedObj.set('Update time!')
    >>> TimeWrappedObj.gettimestr('m')
    'Tue Sep 29 17:28:06 2015'
    >>> TimeWrappedObj
    'Update time!'
    >>> TimeWrappedObj.gettimestr('c')
    'Tue Sep 29 17:25:11 2015'
    >>> TimeWrappedObj.gettimestr('m')
    'Tue Sep 29 17:28:06 2015'
    >>> TimeWrappedObj.gettimestr('a')
    'Tue Sep 29 17:28:19 2015'
  • 相关阅读:
    C# Xamarin For Android自动升级项目实战
    C# Xamarin移动开发基础进修篇
    .NET轻量级ORM框架Dapper入门精通
    ASP.NET WebApi技术从入门到实战演练
    (简单、可靠的安装方法)在Windows Server2016中安装SQL Server2016
    ASP.NET (Core) WebAPI IIS PUT和DELETE请求失败 405的解决办法
    js中判断对象是否为空的三种实现方法
    windows10如何设置只显示时间不显示日期
    NuGet微软官方中国国内镜像
    Win10找不到hosts文件解决方法
  • 原文地址:https://www.cnblogs.com/SRL-Southern/p/4848590.html
Copyright © 2020-2023  润新知