面向对象

分类

  • 面像过程编程:初学者容易接受,从上往下依次执行。
  • 面向函数编程:将某功能的代码封装为一个函数,使用时仅调用函数。
  • 面向对象编程:对函数进行分类和封装
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这个父类;
    pass
p1 = People("张策",18)
p1.run()

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

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

from abc import  ABCMeta,abstractmethod
class Alert(object):
    __metaclass__ = ABCMeta
    @abstractmethod
    def send(self):
        pass
class 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-8
class 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.leader
print Province.memo
print hb.memo
hb.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-8

class 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-8

class 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-8
class 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 = value
hb = Province('陕西','西安','飞天',True)
print hb.show
hb.show = False
print hb.show 
#结果:
True
False
  • 经典类
class test1():
    def __init__(self):
        self.__pravite = 'feitian1'
    @property
    def show(self):
        return self.__pravite
t1 = test1()
print t1.show
t1.show = 'change 1'
print t1.show
#输出结果:
feitian1
change 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 = sid

class Teacher(People,Relation):
    def __init__(self, name, age, tid):
        #People.__init__(self,name,age)          
        super(Teacher, self).__init__(name, age) 
        self.tid = tid

s1 = 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.Bar
father.funk
father.抽烟喝酒烫头
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 D
this 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 D
this is A.save

多态

  • 类的属性
    在内存中只需要存储一次
    在构造函数中的属性,每实例化一个就要存储一次
class People(object):
    def __init__(self,name,country="China"):
        self.name = name
        self.country = country
p1 = People("飞天")
print p1.name
p2 = 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.name
p1=People("mj",18)
print p1.name ,p1.age,p1.country
print id(People.country),id(p1.country)

特殊属性

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

他就是装饰函数的一个函数
#该方法实现了装饰器的功能,但是调用函数的方式发生了改变。
import time
def 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)

python 面向对象赋值 python面向对象函数_实例化

#类中的几种方法
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-y
print 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 = 18
p2 = Person('n2','aa',50)
p1.talk()
p2.talk()
p2.fight(p1.Weight)
print p2.__dict__
#__dict__对象中的所有字段通过key_value的方法获取到并且输出(仅仅字段)
print dir(p2)
#dir(var也一样)获取类中的所有东西,包括__init__等
#输出结果:
XXXX
XXXX
跑
{'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']
"""