Python学习面向对象类之进
1classDog(object):2def__init__(self,name):3self.name=name4
staticmethod#把eat方法变为静态方法5defeat(self):6print("%siseating"%self.name)78d=Dog("wangcai")9d.eat()上面的调用会出以下错误,1TypeError:eat()missing1requiredpositionalargument:self说是eat需要一个self参数,但调用时却没有传递,没错,当eat变成静态方法后,再通过实例调用时就不会自动把实例本身当作一个参数传给self了。想让上面的代码可以正常工作有两种办法调用时主动传递实例本身给eat方法,即d.eat(d)在eat方法中去掉self参数,但这也意味着,在eat中不能通过self.调用实例中的其它变量了
1classDog(object):2def__init__(self,name):3self.name=name4
staticmethod#把eat方法变为静态方法5defeat(self):6print("%siseating"%self.name)78d=Dog("wangcai")9d.eat(d)作用:只是相当于一个单纯函数,要传参数,就要把实例传进去。如果说和类有关系,就是必须有类名去调用。调用不了类或实例中的任何属性1.2类方法(classmethod)类方法通过classmethod装饰器实现,类方法和普通方法的区别是:类方法只能访问类变量,不能访问实例变量1classDog(object):2def__init__(self,name):3self.name=name4
classmethod5defeat(self):6print("%siseating"%self.name)78d=Dog("wangcai")9d.eat()执行报错如下,说Dog没有name属性,因为name是个实例变量,类方法是不能访问实例变量的1AttributeError:typeobjectDoghasnoattributename此时可以定义一个类变量,也叫name,看下执行效果
1classDog(object):2name="哮天犬"3def__init__(self,name):4self.name=name5
classmethod6defeat(self):7print("%siseating"%self.name)89d=Dog("wangcai")10d.eat()1.3属性方法(property)属性方法的作用就是通过property把一个方法变成一个静态属性(函数–变量)1classDog(object):2def__init__(self,name):3self.name=name4
property5defeat(self):6print("%siseating"%self.name)78d=Dog("wangcai")9d.eat()调用会出以下错误,说NoneTypeisnotcallable,因为eat此时已经变成一个静态属性了,不是方法了,想调用已经不需要加()号了,直接d.eat就可以了1TypeError:NoneTypeobjectisnotcallable正常调用如下
1d.eat2#输出3wangcaiiseating但是有个问题,如果eat有其他参数,没法传参数。而且即使变成了静态属性,也没法像普通变量那样“=”赋值怎么传参数呢?属性方法赋值
1classDog(object):2def__init__(self,name):3self.name=name4self.__food=None5
property6defeat(self):7print("%siseating%s"%(self.name,self.__food))8eat.setter#赋值调用属性,调这个方法9defeat(self,food):10print("settofood:",food)11self.__food=foodd=Dog("wangcai")14d.eat15d.eat="baozi"16d.eat如果传多个参数,‘d.eat=“baozi1”,”baozi2”’,接收为元组形式。删除属性方呢执行del删除1deld.eat报错:
1AttributeError:cantdeleteattribute默认不能删除,要删除也是在类里再写一个方法
1classDog(object):2def__init__(self,name):3self.name=name4self.__food=None5
property6defeat(self):7print("%siseating%s"%(self.name,self.__food))8eat.setter#赋值调用属性,调这个方法9defeat(self,food):10print("settofood:",food)11self.__food=food12eat.deleter#删除属性13defeat(self):14delself.__food15print("Deletethefinished")d=Dog("wangcai")18d.eat19d.eat="baozi"20d.eat#传完参数后调用21deld.eat22d.eat#删完后调用报错如下,说明已经删除了1AttributeError:Dogobjecthasnoattribute_Dog__food好吧,把一个方法变成静态属性有什么卵用呢?既然想要静态变量,那直接定义成一个静态变量不就得了么?well,以后你会需到很多场景是不能简单通过定义静态属性来实现的,比如,你想知道一个航班当前的状态,是到达了、延迟了、取消了、还是已经飞走了,想知道这种状态你必须经历以下几步:连接航空公司API查询对查询结果进行解析返回结果给你的用户因此这个status属性的值是一系列动作后才得到的结果,所以你每次调用时,其实它都要经过一系列的动作才返回你结果,但这些动作过程不需要用户关心,用户只需要调用这个属性就可以.2类的特殊成员方法2.1__doc__ 表示类的描述信息
1classFoo:2"""描述类信息"""3deffunc(self):4pass56printFoo.__doc__2.2__module__和__class____module__表示当前操作的对象在那个模块__class__表示当前操作的对象的类是什么aa.py
1classC:23def__init__(self):4self.name=fgfindex.py
1fromaaimportC23obj=C()4printobj.__module__#输出aa,即:输出模块5printobj.__class__#输出aa.C,即:输出类2.3__init__构造方法通过类创建对象时,自动触发执行。2.4__del__析构方法当对象在内存中被释放时,自动触发执行。2.5__call__方法对象后面加括号,触发执行。注:构造方法的执行是由创建对象触发的,即:对象=类名();而对于call方法的执行是由对象后加括号触发的,即:对象()或者类()()
1classFoo:2def__init__(self):3pass45def__call__(self,*args,**kwargs):6print(__call__,args,kwargs)78obj=Foo()#执行__init__9obj(1,2,3,a=4,b=5)#执行__call__2.6__dict__方法查看类或对象中的所有成员
1classProvince:2country=China3def__init__(self,name,count):4self.name=name5self.count=count6deffunc(self,*args,**kwargs):7print(func)89#获取类的成员,即:静态字段、方法、10print(Province.__dict__)11#输出:{country:China,__module__:__main__,func:functionfuncat0x10be30f50,__init__:function__init__at0x10be30ed8,__doc__:None}obj1=Province(HeBei,)14#获取对象obj1的成员15print(obj1.__dict__)16#输出:{count:,name:HeBei}2.7__str__方法如果一个类中定义了str方法,那么在打印对象时,默认输出该方法的返回值。
1classFoo:2def__str__(self):3returnfgf45obj=Foo()6print(obj)定以后,输入定义的值,不定义默认返回对象地址“main.Fooobjectat0xBFE9E8”2.8__getitem__、__setitem__、__delitem__用于索引操作,如字典。以上分别表示获取、设置、删除数据
1classFoo(object):2def__init__(self):3self.data={}4def__getitem__(self,key):5print(__getitem__,key)6returnself.data.get(key)7def__setitem__(self,key,value):8print(__setitem__,key,value)9self.data[key]=value10def__delitem__(self,key):11print(__delitem__,key)obj=Foo()obj[name]=fgf#设置,自动触发执行__setitem__16print(obj.data)17print(obj[name])#获取值,自动触发执行__getitem__18delobj[name]#触发__delitem__,只是调用那个方法,具体删不删看自己配置2.9类的起源__new__和__metaclass__
1classFoo(object):2def__init__(self,name):3self.name=name4f=Foo("fgf")上述代码中,f是通过Foo类实例化的对象,其实,不仅f是一个对象,Foo类本身也是一个对象,因为在Python中一切事物都是对象。如果按照一切事物都是对象的理论:obj对象是通过执行Foo类的构造方法创建,那么Foo类对象应该也是通过执行某个类的构造方法创建。
1printtype(f)#输出:class__main__.Foo表示,obj对象由Foo类创建2printtype(Foo)#输出:typetype表示,Foo类对象由type类创建所以,f对象是Foo类的一个实例,Foo类对象是type类的一个实例,即:Foo类对象是通过type类的构造方法创建。那么,创建类就可以有两种方式:普通方式(也是通过type创建的类,只是已经封装好了)
1classFoo(object):23deffunc(self):4printhellofgf特殊方式
1deffunc(self):2print(hellofgf)34Foo=type(Foo,(object,),{talk:func})5#type第一个参数:类名6#type第二个参数:当前类的基类7#type第三个参数:类的成员8f=Foo()9f.talk()f是Foo的对象,Foo又是type的对象,所以type又称类的类加上构造函数
1deffunc(self):2print("hello%s%s"%(self.name,self.age))34def__init__(self,name,age):5self.name=name6self.age=age7Foo=type(Foo,(object,),{func:func,8__init__:__init__})f=Foo("fgf",22)11f.func()So,记住,类是由type类实例化产生,那么问题来了,类默认是由type类实例化产生,type类中如何实现的创建类?类又是如何创建对象?答:类中有一个属性__metaclass__,其用来表示该类由谁来实例化创建,所以,我们可以为__metaclass__设置一个type类的派生类,从而查看类创建的过程。以下代码python2里运行看效果
1classMyType(type):2def__init__(self,what,bases=None,dict=None):3print("--MyTypeinit---")4super(MyType,self).__init__(what,bases,dict)5def__call__(self,*args,**kwargs):6print("--MyTypecall---")7obj=self.__new__(self,*args,**kwargs)8self.__init__(obj,*args,**kwargs)classFoo(object):11__metaclass__=MyType12def__init__(self,name):13self.name=name14print("Foo---init__")15def__new__(cls,*args,**kwargs):16#__new__是用来创建实例的,定制类,先运行new里调用init,这里写,对默认的重构17print("Foo--new--")18#print(object.__new__(cls))19returnobject.__new__(cls)#返回给init,cls这代表Foo,相当于对象的self20#调用父类的__new__方法#第一阶段:解释器从上到下执行代码创建Foo类23#第二阶段:通过Foo类创建obj对象24obj=Foo("Fgf")(默认之前应该还有个myType.new)先执行myType.init,再执行myType.call,再执行Foo.new,最后Foo.init3反射如果用户输入信息如”fgf”,通过输入字符串”fgf”去调用实例的属性,怎么实现。不可能”name=input()”,再用name去调用fgf属性,那样调用的是name而不是fgf。要想把用户输入字符串转为一个变量名,而不是一个值就需要用到:反射(实现用户输入字符串为类的方法)通过字符串映射或修改程序运行时的状态、属性、方法,有以下4个方法attr–attribute[??tr?bjut]属性;(人或物的)特征hasattr(obj,name_str)判断object中有没有一个name字符串对应的方法或属性
1classFoo(object):2#name=3def__int__(self):4self.name=fgf5deffunc(self):6return"func"78obj=Foo()print(hasattr(obj,name))#输出False11print(hasattr(obj,func))#输出Truegetattr(obj,name_str)根据字符串去获取obj对应方法的内存地址
1defgetattr(object,name,default=None):2"""3getattr(object,name[,default])-value4Getanamedattributefromanobject;getattr(x,y)isequivalenttox.y.5"""
1classFoo(object):2def__init__(self):3self.name=fgf4deffunc(self):5print(self.name,sayHi)6return"func"78obj=Foo()9print(getattr(obj,func))10getattr(obj,func)()##sameas:obj.func()setattr(obj,name_str,func)动态把一个函数装到类里面
1defsetattr(x,y,v):#realsignatureunknown;restoredfrom__doc__2"""3setattr(x,y,v)isequivalentto``x.y=v4"""
1deffunc1(self):2print(self.name,sayHi)3return"func"45classFoo(object):6def__init__(self):7self.name=fgf8deffunc2(self):9print(self.name,sayfu)10return"func"11obj=Foo()choice=input("请输入调用方法名[func1
func2]:")14ifhasattr(obj,choice):#实例中有这个方法,执行实例中的方法15print(getattr(obj,choice))16else:#动态加载函数封装到类中17setattr(obj,choice,func1)18func=getattr(obj,choice)19func(obj)delattr()删除set添加的属性
1defdelattr(x,y):#realsignatureunknown;restoredfrom__doc__2"""3delattr(x,y)isequivalentto``delx.y4"""
1classFoo(object):2def__init__(self):3self.name=fgf4deffunc(self):5print(self.name,sayHi)67obj=Foo()89setattr(obj,age,18)10print(obj.name,obj.age)11#delattr(obj,func)12delattr(obj,age)#只能删除setattr动态添加的,默认的不可以删除13print(obj.name,obj.age)动态导入模块
1importimportlib23__import__(import_lib.metaclass)#这是解释器自己内部用的4#importlib.import_module(import_lib.metaclass)#与上面这句效果一样,官方建议用这个isinstance(obj,cls):检查是否obj是否是类cls的对象issubclass(sub,super):检查sub类是否是super类的派生类4异常处理
在编程过程中为了增加友好性,在程序出现bug时一般不会将错误信息显示给用户,而是现实一个提示的页面
1try:2diction={}3diction[1]4names=[]5names[2]6exceptIndexErrorase:#python3.x里不是逗号,都是as7print(e)8except(KeyError,IndexError)ase:#采用统一处理办法9print(e)10exceptExceptionase:#抓住所有错误11print(e)12else:#没出错执行这个13pass14finally:#不管有没有错,都执行15passtry:18status=ifstatus!=0:20raiseException("自定义异常")21exceptExceptionase:22print(e)来源:
转载请注明:http://www.sonphie.com/jbby/14279.html