学习python课程第二十三天

 一.    组合 :

  1. 什么是组合 ?

    一个对象的属性是来自于另外一个类的对象, 称之为组合. (跟继承其实很相似.都是共用一个类里面的属性)

  2. 为何用组合 ?

    组合也是用来解决类与类代码冗余的问题.

  3. 如何用组合 ?

  

class Foo:
aaa=1111
def __init__(self,x,y):
self.x=x
self.y=y

def func1(self):
print('Foo内的功能')


class Bar:
bbb=2222
def __init__(self, m, n):
self.m = m
self.n = n

def func2(self):
print('Bar内的功能')

obj1=Foo(10,20)
obj2=Bar(30,40)

obj1.xxx=obj2 (把两个类组合到了一起.)


print(obj1.x,obj1.y,obj1.aaa,obj1.func1)
print(obj1.xxx.m,obj1.xxx.n,obj1.xxx.bbb,obj1.xxx.func2)


  

  小练习 :

class OldboyPeople:
school = 'Oldboy'
def __init__(self, name, age, gender):
self.name = name
self.age = age
self.gender = gender

class OldboyStudent(OldboyPeople):
def choose_course(self):
print('%s is choosing course' %self.name)

class OldboyTeacher(OldboyPeople):
def __init__(self, name, age, gender,level,salary):
OldboyPeople.__init__(self, name, age, gender)
self.level=level
self.salary=salary

def score(self,stu,num):
stu.num=num
print('老师%s给学生%s打分%s' %(self.name,stu.name,num))

class Course:
def __init__(self,course_name,course_price,course_period):
self.course_name=course_name
self.course_price=course_price
self.course_period=course_period

def tell_course(self):
print('课程名:<%s> 价钱:[%s] 周期:[%s]' % (self.course_name, self.course_price, self.course_period))

python_obj=Course('python开发',3000,'5mons')
linux_obj=Course('linux运维',5000,'3mons')



stu1=OldboyStudent('egon',18,'male')
stu1.courses=[]
stu1.courses.append(linux_obj)
stu1.courses.append(python_obj)
stu1.courses[0].tell_course()


stu2=OldboyStudent('kevin',38,'male')





  二. 封装 :
  1. 什么是封装 ?
    装,指的是把属性装进一个容器,
    封,指的是隐藏的意思,但是这种隐藏是对外不对内的.


  2. 为何要封装 ?
    封装不是单纯意义的隐藏,
    
    封装数据属性的目的 : 将数据属性封装起来, 类外部的使用就无法直接操作该数据属性了.
    需要类内部开一个接口给使用者,类的设计者可以在接口之上附加任意逻辑,从而严格控制使用者对属性的操作.

    封装函数属性的目的 :隔离复杂度.


  3. 如何封装 ?
    只需要在属性前加上__开头,该属性就会被隐藏起来,该隐藏具备的特点 :

    1. 只是一种语法意义上的变形, 即__开头的属性会在检测语法时发生变形为 _类名__属性名的形式

    2. 这种隐藏式对外不对内的, 因为在类内部检测语法时, 所有的代码统一都发生了变形.
  
    3. 这种变形只在检测语法时发生一次, 在类定义之后新增的__开头的属性并不会发生变形.

    4. 如果父类不想让子类覆盖自己的属性, 可以在属性前加__开头

  注意:
    这种机制也并没有真正意义上的限制我们从外部直接访问属性, 知道了类名和属性名就可以拼出名字 :
    _类名__属性名, 然后就可以访问了, 如a._A__N, 即这种操作并不是严格意义上的限制外部访问, 仅仅是
    一种语法意义上的变形, 主要用来限制外部的直接访问.



  示例:
    1. 封装数据 : 将数据隐藏起来并不是目的. 隐藏起来然后对外提供该数据的接口, 然后我们可以在接口附加上对该数据操作的
      限制, 以此完成对数据属性操作的严格控制.

    
    class Teacher:
        def __init__(self,name,age):
            # self.__name=name
            # self.__age=age
            self.set_info(name,age)

        def tell_info(self):
            print('姓名:%s,年龄:%s' %(self.__name,self.__age))
        def set_info(self,name,age):
            if not isinstance(name,str):
                raise TypeError('姓名必须是字符串类型')
            if not isinstance(age,int):
                raise TypeError('年龄必须是整型')
            self.__name=name
            self.__age=age


    t=Teacher('egon',18)
    t.tell_info()

    t.set_info('egon',19)
    t.tell_info()


    2. 封装函数 : 目的是隔离复杂度.
    
    提示 :在编程语言里, 对外提供的接口 (接口可理为一个入口), 可以是函数, 称为接口函数, 这与接口的概念还不一样,
    接口代表一组接口函数的集合体.

    
    
    #取款是功能,而这个功能有很多功能组成:插卡、密码认证、输入金额、打印账单、取钱
    #对使用者来说,只需要知道取款这个功能即可,其余功能我们都可以隐藏起来,很明显这么做
    #隔离了复杂度,同时也提升了安全性

    class ATM:
        def __card(self):
            print('插卡')
        def __auth(self):
            print('用户认证')
        def __input(self):
            print('输入取款金额')
        def __print_bill(self):
            print('打印账单')
        def __take_money(self):
            print('取款')

        def withdraw(self):
            self.__card()
            self.__auth()
            self.__input()
            self.__print_bill()
            self.__take_money()

    a=ATM()
    a.withdraw()


  封装与扩展性:

    封装在于明确区分内外, 使得类实现者可以修改封装内的东西,而不是影响外部调用者的代码. 而外部使用者只知道一个接口
    (函数), 只要接口(函数)名,参数不变,使用者的代码永远无需改变, 这就提供一个良好的合作基础, 或者说,只要这个基础约定
    不变,则代码改变不足为虑




  三. 特性(property)装饰器:

    什么是特性property
      property是一种特殊的属性,访问它时会执行一段功能(函数) 然后返回值



    为什么要用property :
      将一个类的函数定义后,对象再去使用的时候obj.name,根本无法察觉自己的name是执行了一个函数然后计算出来的,
      这种特性的使用方式遵循了统一访问的原则


    property 下面还有装饰器. @setter.(可以修改属性) @deleter.(可以删除属性)


    示例 :
      

      BMI指数(bmi是计算而来的,但很明显它听起来像是一个属性而非方法,如果我们将其做成一个属性,更

      便于理解)

    成人的BMI数值:
    过轻:低于18.5
    正常:18.5-23.9
    过重:24-27
    肥胖:28-32
    非常肥胖, 高于32
      体质指数(BMI)=体重(kg)÷身高^2(m)


  
    class People:
        def __init__(self,name,weight,height):
            self.name=name
            self.weight=weight
            self.height=height
        @property
        def bmi(self):
            return self.weight / (self.height**2)

    p1=People('egon',75,1.85)
    print(p1.bmi)


  property 的两种使用方法.

  
    class People:
    def __init__(self,name):
    self.__name=name

     @property
     def name(self):
     return '<name:%s>' %self.__name

     @name.setter
     def name(self,new_name):
     if type(new_name) is not str:
     print('名字必须是str类型')
     return
     self.__name=new_name

     @name.deleter
     def name(self):
     del self.__name
    
    obj=People('egon')
    print(obj.name)

    del obj.name
    print(obj.__dict__)


  

    class People:
     def __init__(self,name):
     self.__name=name

     def xxx_name(self):
     return '<name:%s>' %self.__name

     def yyy_name(self,new_name):
     if type(new_name) is not str:
     print('名字必须是str类型')
     return
     self.__name=new_name

     def zzz_name(self):
     del self.__name

     name=property(xxx_name,yyy_name,zzz_name)

    obj=People('egon')
    print(obj.name)

    del obj.name
    print(obj.__dict_




  四. 多态性.

    多态性分为静态多态性与动态多态性.

      静态多态性 :如任何类型都可以用运算符进行运算.

      动态多态性 : 如下
    

        peo=People()
        dog=Dog()
        pig=Pig()

        #peo、dog、pig都是动物,只要是动物肯定有talk方法
        #于是我们可以不用考虑它们三者的具体是什么类型,而直接使用
        peo.talk()
        dog.talk()
        pig.talk()

        #更进一步,我们可以定义一个统一的接口来使用
        def func(obj):
            obj.talk()

    多态性的好处 :

      1, 增加了程序的灵活性 :
          以不变应万变,不论对象千变万化,使用者都是同一种形式去调用, 如func(animal)

      2, 增加了程序的可扩展性
          通过集成animal类创建了一个新的类, 使用者无需更改自己的代码. 还是用func(animal)去调用


    示例:
      
      class Cat(Animal): #属于动物的另外一种形态:猫
        def talk(self):
          print('say miao')
      
      def func(animal): #对于使用者来说,自己的代码根本无需改动
        animal.talk()
      
      cat1=Cat() #实例出一只猫
      func(cat1) #甚至连调用方式也无需改变,就能调用猫的talk功能
      say miao

      这样我们新增了一个形态Cat,由Cat类产生的实例cat1,使用者可以在完全不需要修改自己代码的情况下。
      使用和人、狗、猪一样的方式调用cat1的talk方法,即func(cat1)


  鸭子类型 :
  
      Python崇尚鸭子类型,即如果看起来像,叫声像而且走起路来像鸭子, name它就是鸭子,
      python程序员通常根据这种行为来编写程序, 例如,如果想编写现有对象的自定义版本,可以继承该对象也可以
      创建一个外观和行为像,但与它无任何关系的全新对象,后者通常用于保存程序组件的松耦合度.


      示例一 :
          利用标准库中定义的各种'与文件类似'的对象, 尽管这些对象的工作方式像文件,但他们没有继承内置文件对象的方法.


             
        #二者都像鸭子,二者看起来都像文件,因而就可以当文件一样去用
        class TxtFile:
            def read(self):
                pass

            def write(self):
                pass

        class DiskFile:
            def read(self):
              pass
          def write(self):
              pass



      示例二. :
          其实大家一直在享受着多态性带来的好处,比如Python的序列类型有多种形态:字符串,列表,元组. 多态性体现如下


        #str,list,tuple都是序列类型
        s=str('hello')
        l=list([1,2,3])
        t=tuple((4,5,6))

        #我们可以在不考虑三者类型的前提下使用s,l,t
        s.__len__()
        l.__len__()
        t.__len__()

        len(s)
        len(l)
        len(t

猜你喜欢

转载自www.cnblogs.com/lvyipin1/p/9844802.html