类的授权
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'

最新文章

  1. 应用服务器和Web服务器
  2. XML操作类
  3. C语言实现快速排序
  4. cookie与sessionID之间的关系实验
  5. CSS定义字体间距 字体行与行间距
  6. 栈和队列的面试题Java
  7. View和ViewGroup的区别 -- Touch事件处理
  8. asp.net MVC 学习笔记
  9. B-number
  10. 文件I/O实现cp复制功能
  11. opencv图像特征检测之斑点检测
  12. 安装vue-cli时出现的错误,cmd 卡住
  13. javascript 手势(swipeLeft,swipeRight)滑动中使用css3动画卡顿,开启硬件加速
  14. Navicat之MySQL连接(二)
  15. C++ 标准库之 iomanip 、操作符 ios::fixed 以及 setprecision 使用的惨痛教训经验总结
  16. cpp 区块链模拟示例(五) 序列化
  17. 遍历DOM树,获取父节点
  18. .net MVC框架里怎么写控件
  19. Java 中的定时任务(一)
  20. Android——ImageView的scaleType属性与adjustViewBounds属性 (转)二

热门文章

  1. 团队Alpha冲刺(二)
  2. 未能加载文件或程序集“log4net, Version=1.2.10.0, Culture=neutral, PublicKeyToken=1b44e1d426115821”或它的某一个依赖项。系统找不到指定的文件。
  3. C++ Primer Plus学习:第十章
  4. 关于解决乱码问题的一点探索之一(涉及utf-8和GBK)
  5. SpringMVC项目中获取所有URL到Controller Method的映射
  6. JS在当前页面插入&lt;script&gt;标签,并执行
  7. 【前端学习笔记05】JavaScript数据存储Cookie相关方法封装
  8. 【uoj#225】[UR #15]奥林匹克五子棋 构造
  9. 【明哥报错簿】之【 &quot;javax.servlet.http.HttpServlet&quot; was not found on the Java Build Path || HttpServletRequest/HttpServletResponse cannot be resolved to a type】
  10. 迁移数据到历史表SQL(转)