zoukankan      html  css  js  c++  java
  • 学习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
  • 相关阅读:
    SQL SERVER 2012 第三章 使用INSERT语句添加数据
    SQL SERVER 2012 第三章 T-SQL 基本语句 having子句
    T4 模板生产 多文件
    RAC+DG修改sys密码
    数据泵导入,报错:ORA-12899: value too large for column "SCOTT"."TEST112"."JOIN" (actual: 9, maximum: 8)
    grep过滤空行和注释行
    char与varchar2字符类型的区别
    Oracle不知道用户密码情况下,如何在不更改密码的前提下解锁用户或者延期密码有效期
    数据泵导出报错ORA-31693 ORA-02354 ORA-01466
    安装ogg软件报错:[INS-75012]Sofware Location specified is already an existing Oracle
  • 原文地址:https://www.cnblogs.com/lvyipin1/p/9844802.html
Copyright © 2011-2022 走看看