python 中的面向对象

来源:互联网 发布:什么软件租房子最靠谱 编辑:程序博客网 时间:2024/05/19 09:37

面向对象

分类

  • 面像过程编程:初学者容易接受,从上往下依次执行。
  • 面向函数编程:将某功能的代码封装为一个函数,使用时仅调用函数。
  • 面向对象编程:对函数进行分类和封装
class people:               # 经典类class People(object):       # 新式类,object类是所有类的基类/父类    def __init__(self,name,age):    # 构造函数,当实例化对象时自动调用,self是实例化出来的对象        self.name = name            # 属性(和字段好像叫法不同)        self.age = age    def run(self):                  # 方法        print "running...."    def __del__(self):        print "deleteing......"     # 析构函数,删除实例化对象时自动调用;class Student(People):              # Student是子类,继承People这个父类;    passp1 = People("张策",18)p1.run()

还有抽象类(除了经典类和新式类)

作用
- 一个普通的接口(相当于API接口)
- 一种规范
也就是抽象类+抽象方法

from abc import  ABCMeta,abstractmethodclass Alert(object):    __metaclass__ = ABCMeta    @abstractmethod    def send(self):        passclass Foo(Alert):    def __init__(self):        print '__init__'    def send(self):        print '这里就是一种规范'f = Foo()  f.send()    #输出结果:__init__这里就是一种规范#为什么说这里是一种规范,因为,在你前面定义的Alter类中send方法下面不能写任何方法,他只是一种规范,在Foo类中继承这个方法时,你需要和Alter类中有相同的方法,也就是send方法

析构函数 __del__

  • python虚拟机销毁对象是自动执行的方法,他不是销毁内存,销毁内存它是由Python自带的算法。
  • 都是在最后执行

__call__方法

class Foo:    def __init__(self):        pass    def __del__(self):        print "拜拜"    def __call__(self):        print "这里是__call__方法"f1 = Foo()f1()#这里相当于指向call方法#输出结果:这里是__call__方法拜拜#拜拜是在执行完释放内存

python中的静态和动态方法,静态和动态字段

#!/usr/bin/env python#coding:utf-8class Province:    memo = '你一定可以成为你想成为的人'    #这里的memo变量属于整个类,而不是一个类中的某个方法,但是他可以通过对象和类调用。    def __init__(self,name,capital,leader):          self.name = name            #self 相当于动态字段        self.capital = capital        self.leader = leader    def suport_meet(self):          #这个方法为动态方法        print self.name + "正在开运动会" #静态方法是由于面向对象编程才从java哪里引进,在编程时你必须要构造一个对象进行进行使用,如果进行构造,就会在内存中开辟空间,所以才使用了静态方法       @staticmethod   #加了这个装饰器,相当于将动态方法,转化为静态方法,并且将self参数去掉    def Foo():      #如果有很多对象,我们可以不用实例化,直接进行类.方法,进行操作,节省实例化使用的内存空间        print '要努力学习,称为想成为的人'     @property       #加了这个装饰器,就相当于将方法变成特性,访问的方法和访问字段一样    def Bar(self):        print '加油,努力'hb = Province('陕西','西安','飞天')print hb.leaderprint Province.memoprint hb.memohb.suport_meet()Province.Foo()hb.Bar#这里注意,self相当于hb,这里就先当于给hb.name等等赋值,hb就叫做对象#在这里我们引出如果属于类的字段(也就是这里的demo),这类叫做静态字段,而self.name则是动态字段#对象可以访问静态字段,建议在使用时,应禁止使用对象调用。#类不能够访访问动态方法,他需要实例化,但是类可以访问静态方法#输出结果:飞天你一定可以成为你想成为的人你一定可以成为你想成为的人陕西正在开运动会要努力学习,称为想成为的人加油,努力

私有方法和私有字段(外面可以访问和不可以访问)

私有字段

  • self.__name = name
    这是name这个字段就变成了,私有字段,这时你如果通过类.字段访问就会报错
    AttributeError: Province instance has no attribute ‘__flag’(说我们没有这个属性)
    这里我们只能通过内部的类进行间接访问
#!/usr/bin/env python#coding:utf-8class Province:    memo = '你一定可以成为你想成为的人'    def __init__(self,name,capital,leader,flag):          self.name = name                  self.capital = capital        self.leader = leader        self.__flag = flag    def suport_meet(self):                  print self.name + "正在开运动会"       @staticmethod      def Foo():             print '要努力学习,称为想成为的人'    @property          def Bar(self):        print '加油,努力'    def show(self):        print self.__flag     hb = Province('陕西','西安','飞天',True)hb.show()#输出:True还有一种方法:通过 @property,将内部的show方法修改    @property        def show(self):        return self.__flag#通过特性方法进行直接调用print hb.show

私有方法(和私有字段相同的概念)

#!/usr/bin/env python#coding:utf-8class Province:    memo = '你一定可以成为你想成为的人'    def __init__(self,name,capital,leader,flag):          self.name = name                  self.capital = capital        self.leader = leader        self.__flag = flag    def suport_meet(self):                  print self.name + "正在开运动会"       @staticmethod      def Foo():             print '要努力学习,称为想成为的人'    @property          def Bar(self):        print '加油,努力'    def show(self):        print self.__flag        def __sha(self):         print '我真的很努力'      hb = Province('陕西','西安','飞天',True)hb.__sha()#结果报错:Province instance has no attribute '__sha'同样,在类的内部进行方访:    def lala(self):        self.__sha()hb = Province('陕西','西安','飞天',True)hb.lala()同样第二种方法直接访问:hb._Province__sha()进行直接访问

只读特性和只写特性(新式类,经典类好像可以直接改)

  • 新式类
#!/usr/bin/env python#coding:utf-8class Province(object):    memo = '你一定可以成为你想成为的人'    def __init__(self,name,capital,leader,flag):          self.name = name                   self.capital = capital        self.leader = leader        self.__flag = flag    def suport_meet(self):                 print self.name + "正在开运动会"    @staticmethod       def Foo():            print '要努力学习,成为想成为的人'     @property           def Bar(self):        print '加油,努力'    #这里就是只读    @property        def show(self):        return self.__flag    #可以写的装饰器    @show.setter    def show(self,value):        self.__flag = valuehb = Province('陕西','西安','飞天',True)print hb.showhb.show = Falseprint hb.show #结果:TrueFalse
  • 经典类
class test1():    def __init__(self):        self.__pravite = 'feitian1'    @property    def show(self):        return self.__pravitet1 = test1()print t1.showt1.show = 'change 1'print t1.show#输出结果:feitian1change 1

什么是面向对象?

类 ===== 建房子的图纸 (三室一厅,两室一厅…….)
对象===== 实际建出来的房子(门牌号)
class ThreeRoom:
pass

seven_zero_one = ThreeRoom()
seven_zero_one.live()
seven_zero_one.clean()

面向对象的三个特性:

封装

  • 封装:把内容统一放在一个地方,看成一个整体,(实例化对象self和类的属性绑定在一起);
  • 访问封装内容的两种方式:
    通过self去访问封装的内容;(self.name)
    通过实例化的对象名去访问封装的内容;(p1 = People(“westos”,17) p1.age)

继承(子承父业)

  • 新名词:基类/派生类, 父类/子类, 新式类和经典类
  • 多继承:
    新式类: 广度优先继承;(python2.x和python3.x均支持)
    经典类:深度优先继承;(python2.x支持,python3.x没有经典类)
  • 注意: 类的方法中可以传递一个对象;
class People(object):       # 新式类,object类是所有类的基类/父类    def __init__(self,name,age):    # 构造函数,当实例化对象时自动调用;        self.name = name            # 属性        self.age = age        print "%s is create...." %(self.name)    def run(self):                  # 方法        print "%s running...." %(self.name)    def __del__(self):        print "deleteing......"     # 析构函数,删除实例化对象时自动调用;class Relation(object):    **def make_relation(self,obj):** #传入一个对象作为参数        print "%s is related with %s" %(self.name,obj.name)class Student(People,Relation):              # Student是子类,继承People这个父类;      def __init__(self,name, age, sid):        # People.__init__(self,name,age)            # 如果父类名更改,此处也需要更改;        super(Student, self).__init__(name,age)     # 更推荐使用        self.sid = sidclass Teacher(People,Relation):    def __init__(self, name, age, tid):        #People.__init__(self,name,age)                  super(Teacher, self).__init__(name, age)         self.tid = tids1 = Student("lijian", 18, "007")t1 = Teacher("westos", 18, "001")s1.make_relation(t1)

继承中重写父类方法

class Father:    def __init__(self):        self.fname = 'fff'    def Func(self):        print 'father.funk'    def Bad(self):        print 'father.抽烟喝酒烫头'class Son(Father):    def __init__(self):        self.Sname = 'sss'    def Bar(self):        print 'Son.Bar'    #这里相当于重写了父类的bad方法    def Bad(self):        Father.Bad(self)        print 'son.学习上网'      s1 = Son()s1.Bar()s1.Func()#输出结果:Son.Barfather.funkfather.抽烟喝酒烫头son.学习上网

继承中调用父类构造方法

class Father(object):    def __init__(self):        self.fname = 'fff'        print 'father.__init__'    def Func(self):        print 'father.funk'    def Bad(self):        print 'father.抽烟喝酒烫头'class Son(Father):    def __init__(self):        self.Sname = 'sss'        print 'son.__init__'        #Father.__init__(self)       #经典类中的调用方法        super(Son, self).__init__()  #新式类调用方法(建议使用这种方式)    def Bar(self):        print 'Son.Bar'    def Bad(self):        Father.Bad(self)        print 'son.学习上网'     s1 = Son()#输出结果son.__init__father.__init__

多类继承

-新式类(广度优先)

class A(object):    def __init__(self):        print 'this is a'    def save(self):        print 'this is A.save'class B(A):    def __init__(self):        print 'this is B'class C(A):    def __init__(self):        print 'this is C'    def save(self):        print  'this is C.save'class D(B,C):    def __init__(self):        print 'this is D' c1 = D()c1.save()#输出结果:this is Dthis is C.save
  • 经典类
class A:    def __init__(self):        print 'this is a'    def save(self):        print 'this is A.save'     class B(A):    def __init__(self):        print 'this is B'class C(A):    def __init__(self):        print 'this is C'    def save(self):        print  'this is C.save'class D(B,C):    def __init__(self):        print 'this is D'        c1 = D()c1.save()#输出结果:this is Dthis is A.save

多态

  • 类的属性
    在内存中只需要存储一次
    在构造函数中的属性,每实例化一个就要存储一次
class People(object):    def __init__(self,name,country="China"):        self.name = name        self.country = countryp1 = People("飞天")print p1.namep2 = People("lala")print p2.name#结果为:飞天lala
  • 多态
    如果子类调用的方法,子类没有,父类有,运行父类;
    如果子类调用的方法,子类有,父类也有,只运行子类的;

面向对象进阶

  • 类变量,全局变量在内存中只存储一份
  • 普通的对象属性,每个对象中都要存储一份
class People(object):    # def __init__(self,name,age,country="china"):  #每创建一个实例就会在内存中重新存储一次    #     self.name=name    #     self.age=age    #     self.country=country    country="China"      #全局变量,类的属性,只需要存储一次    def __init__(self,name,age):        self.name=name        self.age=age    def run(self):        print "%s is running..."%self.namep1=People("mj",18)print p1.name ,p1.age,p1.countryprint id(People.country),id(p1.country)

特殊属性

装饰器property(前面已经介绍过,这里就不在说了)

他就是装饰函数的一个函数
#该方法实现了装饰器的功能,但是调用函数的方式发生了改变。import timedef dtimer(fun):    def timer(*args,**kwargs):  # 高阶函数       #args = (1,2,3,4)        start_time = time.time()        fun(*args,**kwargs)                      # args =(1,2,3,4), 解包*(1,2,3,4)        stop_time = time.time()        return "run %s" % (stop_time - start_time)    return timer  # 返回的是timer的地址,要执行该函数需要timer()@dtimer  # fun1 = dtimer(fun1)def fun1(*args,**kwargs):    print "in the fun1....."    print args    time.sleep(1)def fun2():    print "in the fun2....."    time.sleep(0.5)print fun1(1,2,3,4)# print timer(fun2)

这是输出结果

#类中的几种方法class Date(object):    def __init__(self,year, month,day):        self.year = year        self.month = month        self.day = day    def echo_date(self):        print "Year:",self.year        print "Month:",self.month        print "Day:",self.day    @classmethod    def from_str(cls,s):      # class,类方法的第一个参数是类本身,cls = Date        year, month, day = map(int, s.split("-"))        d = cls(year,month,day)     # d = Date(year,month,day)        return d        # 对象    @staticmethod·    def is_date_legal(s):      # 静态方法,第一个参数既不是对象,也不是类本身,        year, month, day = map(int, s.split("-"))        return year >= 1970 and 0 < month <= 12 and 0 < day <= 31#d = Date(2017,9,9)#d.echo_date()#d1 = Date.from_str('2018-10-19')#d1.echo_date()print  "legal" if Date.is_date_legal('2017-13-16') else "illegal"

匿名函数

  • lambda 形参 : 返回值
#这里的形参,可以赋默认值,形参和返回值都可以使用*args和**kwargs。f = lambda  x,y:x-yprint f (1,2)#返回结果为1

断言

  • assert(1 == 1)
  • 作用:也就是判断条件

练习

#!/usr/bin/env python#coding:utf-8"""class Person(object):    def __init__(self,name,gene,weight):        self.Name = name        self.__Gene = gene        if name != 'feitian':            self.Gender = '男'        self.Weight =weight        self.Age = None    def talk(self):        print 'XXXX'    def fight(self,weight):        if self.Weight > weight:            print '打'        else:            print '跑'p1 = Person('n1','a',190)p1.Age = 18p2 = Person('n2','aa',50)p1.talk()p2.talk()p2.fight(p1.Weight)print p2.__dict__#__dict__对象中的所有字段通过key_value的方法获取到并且输出(仅仅字段)print dir(p2)#dir(var也一样)获取类中的所有东西,包括__init__等#输出结果:XXXXXXXX跑{'Gender': '\xe7\x94\xb7', 'Age': None, '_Person__Gene': 'aa', 'Name': 'n2', 'Weight': 50}['Age', 'Gender', 'Name', 'Weight', '_Person__Gene', '__class__', '__delattr__', '__dict__', '__doc__', '__format__', '__getattribute__', '__hash__', '__init__', '__module__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'fight', 'talk']"""
原创粉丝点击