zoukankan      html  css  js  c++  java
  • Python封装与隐藏

     

    今日内容:

    1.封装与隐藏
    2.property
    3.绑定方法与非绑定方法

    知识点一:封装与隐藏

      

      1、什么封装:
          封:属性对外是隐藏的,但对内是开放的
          装:申请一个名称空间,往里装入一系列名字/属性

         2、为什么要封装:
         封装数据属性的目的
           首先定义属性的目的就是为了给类外部的使用者使用的,
           隐藏之后是为了不让外部使用者直接使用,需要类内部开辟一个接口
           然后让类外部的使用通过接口来间接地操作隐藏的属性。
           精髓在于:我们可以在接口之上附加任意逻辑,从而严格控制使用者对属性的操作

         封装函数属性:
           首先定义属性的目的就是为了给类外部的使用使用的,
           隐藏函数属性是为了不让外不直接使用,需要类内部开辟一个接口
           然后在接口内去调用隐藏的功能
         精髓在于:隔离了复杂度

      3、如何隐藏:

        1、 这种隐藏仅仅只是一种语法上的变形操作
          2、 这种语法上的变形只在类定义阶段发生一次,因为类体代码仅仅只在类定义阶段检测一次
          3、 这种隐藏是对外不对内的,即在类的内部可以直接访问,而在类的外则无法直接访问,原因是
             在类定义阶段,类体内代码统一发生了一次变形

          4、 如果不想让子类的方法覆盖父类的,可以将该方法名前加一个__开头

     

     1 class People:
     2     def __init__(self,name,age):
     3         self.__name=name
     4         self.__age=age
     5 
     6     def tell_info(self):
     7         print('%s:%s' %(self.__name,self.__age))
     8 
     9     def set_info(self,name,age):
    10         if type(name) is not str:
    11             # print('用户名必须为str类型')
    12             # return
    13             raise TypeError('用户名必须为str类型')
    14 
    15         if type(age) is not int:
    16             # print('年龄必须为int类型')
    17             # return
    18             raise TypeError('年龄必须为int类型')
    19         self.__name=name
    20         self.__age=age
    21 
    22 peo1=People('张三',34)
    23 # peo1.name=123
    24 # peo1.age
    25 peo1.tell_info()
    26 
    27 peo1.set_info('李四',19)
    28 peo1.tell_info()

     知识点二:property装饰

      property装饰器用于将被装饰的方法伪装成一个数据属性,
      在使用时可以不用加括号而直接引用

     1 》》》@propertyde思路演变的过程
     2 版本1:正常版本
     3 class People:
     4     def __init__(self,name,weight,height):
     5         self.name=name
     6         self.weight=weight
     7         self.height=height
     8 
     9     def bmi(self):
    10         return self.weight / (self.height**2)
    11 
    12 peo=People('张三',130,1.65)
    13 print(peo.name)   #张三
    14 print(peo.bmi())   #47.75022956841139
    15 
    16 问题:发现bmi的访问方式比访问对象属性多了个()
    17 思路变化1:正常情况下我们访问对象里面属性和类里面的函数方式为:
    18 思路变化2:为了方便用户调用,能不能让用户已访问对象属性的方式去访问bmi print(peo.bmi)
     1 》》》》》》版本2:@property初始版本
     2 #思考:如何实现改变访问bmi的方式为》》》print(peo.bmi)
     3 #      使用@property:作用是将本装饰的方法伪装成一个数据类型,在使用时可以不用加括号而直接使用
     4 
     5 class People:
     6     def __init__(self,name,weight,height):
     7         self.name=name
     8         self.weight=weight
     9         self.height=height
    10 
    11     @property        #将bmi伪装成一个数据类型
    12     def bmi(self):
    13         return self.weight / (self.height**2)
    14 
    15 peo=People('张三',130,1.65)
    16 print(peo.name)   #张三
    17 print(peo.bmi)    #这样可以直接调用而不用加括号
     1 》》》》》完整版:@property
     2 #property装饰器
     3 #是一种封装思想的体现,方便了查 删 改
     4 
     5 class people:
     6     def __init__(self,name):
     7         self.__name=name
     8 
     9     @property
    10     def name(self):            #查看obj.name
    11         return '名字是:%s'%self.__name
    12 
    13     @name.setter              #更改obj.name=name
    14     def name(self,name):
    15         if type(name) is not str:
    16             raise TypeError('名字必须是str类型')
    17         self.__name=name
    18     @name.deleter           #删除 del obj.name
    19     def name(self):
    20         print('不让删除')  #里面可以设定提示
    21         # del self.__name      #也可以设定直接删除功能
    22 
    23 peo=people('张三')
    24 #默认不加@property调用方法为:
    25 
    26 peo.name='李四'  #@name.setter 调用了更改
    27 print(peo.name)    #@property 调用了查看                  输出结果:名字是:李四
    28 del peo.name       #@name.deleter  调用了产出的需求       输出结果:不让删除

    知识点三:绑定方法

      1.绑定方法
        特性:绑定给谁就应该由谁来调用,谁来调用就会将谁当做第一个参数自动传入
        精髓:是在于自动传值

          1.1绑定对象的方法
            在类内部定义的函数(没有被任何装饰器修饰的),默认是绑定给对象用的

          1.2绑定给类的方法
            在类内部定义的函数如果被装饰器@classmethod装饰,
            那么则是绑定给类的,应该由类来调用,类来调用就自动将类当做第一个参数自动传入

        绑定给类的:自动传的是类的值
        绑定给对象的:自动传的是对象的值

      2.非绑定方法

        类中定义的函数如果被装饰器@staticmethod装饰,那么该函数就变成非绑定方法
        既不与类绑定,又不与对象绑定,意味着类与对象都可以来调用
        但是无论谁来调用,都没有任何自动化传值的效果,就是一个普通函数

      

      3.应用
        什么时候绑定给类? 什么时候绑定给对象?
        取决于函数体代码
        如果函数体代码需要用外部传入的类,则应该将该函数定义成绑定类的方法
        如果函数体代码需要用外部传入的对象,则应该将该函数定义成绑定给对象的方法
        如果函数体代码既不需要外部传入的类也不需要外部传入的对象,则应该将该函数定义成非绑定方法对象,则应该将该函数定义成非绑定方法

      

     1 #绑定方法初始实例:绑定给类@classmethod
     2 
     3 class Foo:
     4     @classmethod
     5     def f1(cls):
     6         print('Foo.f1')
     7         print(cls)
     8 
     9     def f2(self):
    10         print(self)
    11 obj=Foo()
    12 # print(obj.f2)   #默认绑定给对象用#<bound method Foo.f2 of <__main__.Foo object at 0x04BACD10>>
    13 # print(Foo.f1)   #绑定给类用 <bound method Foo.f1 of <class '__main__.Foo'>>
    14 Foo.f1()     #类调用
    15 print(Foo)   #所以传入的值就是类的值 <class '__main__.Foo'>
    16 
    17 obj.f2()    #对象调用,
    18 print(obj)  #所以传入的值就是对象的值 <__main__.Foo object at 0x04C50EB0>
    19 
    20 # f1绑定给类的应该由类来调用,但对象其实也可以使用,只不过自动传入的任然是类
    21 #f2绑定给对象的应该由对象来调用,但是类其实也可以使用
     1 # 绑定方法进阶实例2:新的实例化方式,绑定给类
     2 """
     3 将需要传入的信息写到settings.py文件里面,调用更加灵活方便
     4 IP='192.168.11.3'
     5 PORT=3306
     6 """
     7 import settings
     8 
     9 class Mysql:
    10     def __init__(self,ip,port):
    11         self.ip=ip
    12         self.port=port
    13     def tell_info(self):
    14         print('%s:%s'%(self.ip,self.port))
    15         # print(self)
    16 
    17     @classmethod
    18     def from_conf(cls):
    19         return cls(settings.IP,settings.PORT) #可以理解为 Mysql('1.1.1.1',23)即调用了init方法,给其传参数 cls名字可以随便起
    20     # print(settings.IP)
    21 
    22 #默认的实例化方式:类名(..)(实质给tell_info(self)里的self传的是对象res的值)
    23 res=Mysql('1.1.1.1',23)
    24 print(res)  #<__main__.Mysql object at 0x05717410>即为对象res也就是tell_info(self)的值
    25 res.tell_info()
    26 
    27 # 一种新的实例化方式,从配置文件中读取配置完成实例化(这里实质给cls传的是类Mysql的值)
    28 res=Mysql.from_conf()
    29 res.tell_info()
    30 print(res.ip)

     非绑定方法应用实例:

     1 #非绑定方法应用实例:uuid获取随机id值
     2 
     3 import uuid
     4 class UserInfo:
     5     def __init__(self,name,age):
     6 
     7         self.name=name
     8         self.age=age
     9         self.uid = self.create_uid()
    10     def tell_info(self):
    11         print('姓名:%s,年龄:%s,ID:%s'%(self.name,self.age,self.uid))
    12         # print(self)
    13 
    14     @staticmethod          #因为不需要传任何参数所以定义为非绑定方法,类和对象都可以取调用
    15     def create_uid():
    16         return uuid.uuid1()
    17 
    18 res=UserInfo('张三',23,)
    19 res.tell_info()
    20 print(res.uid)

     

  • 相关阅读:
    Java实现蓝桥杯 算法训练 Professor Monotonic's Network
    Java实现 蓝桥杯 算法训练 数字游戏
    Java实现 蓝桥杯 算法训练 数字游戏
    Java实现 蓝桥杯 算法训练 数字游戏
    Java实现 LeetCode 264 丑数 II(二)
    Java实现 LeetCode 264 丑数 II(二)
    linux下c/c++方式访问curl的帮助手册
    boost 1.56.0 编译及使用
    使用Boost.Asio编写通信程序
    webrtc教程
  • 原文地址:https://www.cnblogs.com/yangzhizong/p/9240056.html
Copyright © 2011-2022 走看看