我的日志
就在这里写写日志吧.
2004年9月
9月7日
指南针在这里星期算是启动了,当然目前还是做一些准备工作.翻译Twisted文档是第一件事情了.这几日忙着帮朋友写PHP,翻译的时间少了很多,不过怎么样也要在星期天之前找时间把Low-Level部分翻译完,拖的时间太久了.
写了几天的PHP觉得其实PHP也挺有意思的.
有一些地方和Python比较类似,可能脚本语言都是这样吧,不过我没有涉及过其它的脚本语言,也不好多说什么.教本语言一般都有自己的内存管理机制吧,让写惯C++的我感觉很是爽,少了很多的麻烦事.
昨天看到群里面有人问怎么安装STLPort和Boost.又想起我自己的计划,最近是没有时间弄这个事情了.再等等吧...
2004年10月
下面是转载limoduo的一些文章.仔细学习一下
Mix-in技术介绍
mix-in技术,中文不知道应该如何称呼,但意思好象是混入。它的作用是,在运行期间 ,动态改变类的基类或类的方法,从而使得类的表现可以发生变化。可以用在一个通用类接口中,?br>莶煌难≡袷褂貌煌牡筒憷嗍迪郑卟憷嗖挥梅⑸浠6艺庖皇迪挚梢栽谠诵泄讨卸 懈谋洹S捎谖乙彩歉湛吹剑蠹矣形侍饪梢杂胛医薪涣鳌?a href="http://www2.linuxjournal.com/lj-issues/issue84/4540.html">这就是我看到的文章的链 接。前面的一个贴子,dont cry out loud褪怯胫泄?
下面再详细向大家介绍一下:
有一个类,
class foo:
pass我可以定义另外一个类,
class foobase:
def hello(self):
print "hello"如果我直接调用:
>>> obj=foo() >>> obj.hello()
这时你会看到出错。那么我可以这样:
>>> foo.__bases__ +=(foobase,) >>> obj.hello() hello
成功了。原理是,每个类都有一个bases属性,它是一个tuple,用来存放所有的基类。而且在 运行中,可以动态改变。所以当我们向其中增加新的基类时,再次调用原来不存在的函数,由于基 类中的函数已经存在了,所以这次成功了。
这是一个最简单的应用,可以看到我们可以动态改变类的基类。有几个注意事项要说一下:
bases是一个tuple,所以增加一个值要使用tuple类型,而单个元素tuple的写法为(foobase,)
- 类必须先存在。所以,如果想使用这一技术,先要将相关的类的模块导入(import)。
由于mix-in是一种动态技术,以多继承,对象为基础,而python正好是这样的语言,使得在python 中实现这一技术非常容易。
Mix-in技术介绍(续一)
前一篇文章,简单地向大家介绍了一下mix-in技术,它实现了基类的动态 增加。这样 我们就可以在运行时,根据选择可以动态地增加基类,从而 实现不同的目的。现在 还有一个问题,就是,在基类与派生类中都有同 名的函数,要如何处理呢?
在Python中,如果派生类中有与基类同名的函数,那么调用函数时,会调 用派生类的函数,而不是基类的函数,可以测试一下:
>>> class foobase:
def a(self):
print "hello"
>>> class foo(foobase):
def a(self):
print "foo"
>>> c=foo()
>>> c.a()
foo可以看出,执行的是foo类的函数。这样在使用mix-in技术时,如果原来 的类中存在与Mix类中同名的函数,那么Mix类中的函数不会运行,如果 想对其进行替换怎么办呢?方法就是使用getattr()和setattr()函数。当然 还是最简单的,更复杂更通用的以后再讨论。
定义两个类:
>>> class foobase:
def a(self):
print "hello"
>>> class foo:
def a(self):
print "foo"
>>> f=getattr(foobase, "a")
>>> setattr(foo, "a", f.im_func) #f.im_func会得到真正的函数对象
>>> c=foo()
>>> c.a()
hello可以看到,函数被替换了。
注意,使用dir(f)还会看到其它的属性im_class,它表示这个函数属于哪个类, im_self表示属于哪个实例
setattr(object,name,func) 这个函数会给object的name函数引用重新定义为func。即原来name所指的内容发生了变化。 你可以自已定义一个类如
class foo:
def a(self):pass
f=getattr(foo, "a")
dir(f) 你会看到有一些属性,其中有im_class,im_func,im_self属性。其中im_func是真正的函数引用(?br>部梢越泻韵螅蛭赑ython中都是对象) 使用setattr可以将类foo的函数a的实际函数对象替换成别的东西,象文章中所说的。 使用setattr(foo, "a", f.im_func)
如果foo不是一个类,而是一个模块,你使用getattr()后,再使用dir()不会看到im_func函数。
Mix-in技术介绍(续二)
前面讲了基本的实现技术,下面给大家介绍一个mix-in安装函数,这个函 数是从前面所说的文章copy下来的。
import types
def MixIn(pyClass, mixInClass, makeAncestor=0):
if makeAncestor:
if mixInClass not in pyClass.__bases__:
pyClass.__bases__ = (mixInClass,) + pyClass.__bases__
else:
# Recursively traverse the mix-in ancestor
# classes in order to support inheritance
baseClasses = list(mixInClass.__bases__)
baseClasses.reverse()
for baseClass in baseClasses:
MixIn(pyClass, baseClass)
# Install the mix-in methods into the class
for name in dir(mixInClass):
if not name.startswith('__'):
# skip private members
member = getattr(mixInClass, name)
if type(member) is types.MethodType:
member = member.im_func
setattr(pyClass, name, member)这个函数可以将某个mix-in类安装为指定类的基类,同时可以通过关键 字参数指定在基类中的顺序,是最前还是最后。因为Python在处理基类 时,是安顺序进行的,所以安装在最前则优先级最高。同时对于指定类 的方法如果在mix-in类中存在,则将指定类中的方法替换成mix-in类中的方法。
if makeAncestor:
if mixInClass not in pyClass.__bases__:
pyClass.__bases__ = (mixInClass,) + pyClass.__bases__如果makeAncestor为1,表示是安装在最前,则首先判断在pyClass的基 类中是否存在mixInClass类,如果不存在,再进行安装。
else:
# Recursively traverse the mix-in ancestor
# classes in order to support inheritance
baseClasses = list(mixInClass.__bases__)
baseClasses.reverse()
for baseClass in baseClasses:
MixIn(pyClass, baseClass)如果makeAncestor为0,并不将mixInClass安装在最后,原作者说他在 实际中没有这样用的。那么它完成什么任务呢?它实际完成了一个递归, 即从mixInClass的最底层的基类开始(因为mixInClass也可能是多重继承 而来的),对pyClass中也存在的函数进行替换。这样执行完毕后, mixInClass类中,包含所有基类中的函数,如果有与pyClass类中的函 数重名的,都将pyClass中的函数替换成mixInClass相应的函数。(有 些复杂!)
# Install the mix-in methods into the class
for name in dir(mixInClass):
if not name.startswith('__'):
# skip private members
member = getattr(mixInClass, name)
if type(member) is types.MethodType:
member = member.im_func
setattr(pyClass, name, member)这步完成重名函数的替换。首先去掉私有方法(私有方法名前有''). 得到mixInClass类中的指定名字的方法对象,判断是否为方法类型。 因为还有可能取到属性。在types模块中包含了一些类型,可以用它 来判断是否为方法类型。对于方法对象,如果是类方法,实际的函数 应使用它的属性im_func。然后将pyClass相应的方法替换成 mixInClass中的方法。 这样就将mixInClass安装为pyClass的基类了。 使用例子如: from classa import classa
from classb import classb
MixIn(classa, classb) #将classb安装为classa的基类
