zoukankan      html  css  js  c++  java
  • 封装、property特性及绑定与非绑定方法

    1、封装

    (1)什么是封装?

    封:属性对外是隐藏的,但对内是开放的;

    装:申请一个名称空间,往里面装入一系列名字/属性

    (2)为什么要封装?

    封装数据属性的目的

           首先定义属性的目的就是为了给类外部的使用者使用的

            隐藏之后是为了不让外部使用直接使用,需要类内部开辟一个接口

            然后让类外部的使用通过接口来间接的操作隐蔽的属性

            精髓在于:我们可以在接口之上附加任意逻辑,从而严格严格控制使用者对属性的操作;

    封装函数属性

             首先定义属性的目的就是为了给类外部的使用使用的

              隐藏函数属性是为了让使用者不直接使用,在类内部开辟一个接口,在接口内部调用隐蔽的功能

              精髓在于:隔离了复杂度

    (3)如何封装

    如何隐藏:在属性前加上__开头

    1、这种隐藏仅仅只是一种语法上的变形操作;

    2、这种语法上的变形只是在类定义阶段发生一次,因为类体代码仅仅只在类定义阶段检测一次;

    3、这种隐藏是对外不对内的,即在类的内部可以直接访问

    ,而在类的外部则无法直接访问,原因是在类定义阶段,类体代码统一发生了一次变形

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

    如何隐藏
    # class people:
    # __country= 'china' # 在要隐藏的属性前加上__
    # def __init__(self,name,age,sex):
    # self.__name= name #在定义阶段代码发生了变形 self._people__name= name
    # self .age= age
    # self .sex= sex
    #
    # def eat(self):
    # print('eat....')
    # print(people .__country ) #在类内部可以访问

    # peo= people('egon',18,'male')
    #people .__country #在类外部无法访问
    # print(people.__dict__ )
    #隐藏的原理:在类定义阶段属性名__country 变成了_people__country
    # print(people._people__country) #这样是可以访问到隐藏后的属性的


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

    class Foo:
    def __f1(self): #_Foo_f1
    print('Foo.f1')

    def f2(self):
    print('Foo.f2')
    self .__f1() #self._Foo_f1

    class Bar(Foo):
    def __f1(self): #_Bar_f1
    print('Bar.f1')
    #print(Bar.mro())
    obj = Bar()
    obj.f2()

    2、Property 装饰器

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

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

    @property #伪装成数据属性查看
    def name(self):
    return self .__name

    @name.setter #在原来函数name上进行修改操作
    def name(self,name):
    if type(name) is not str:
    print('名字必须是str类型')
    self .__name= name #把原来的属性self.__name改成name

    @name.deleter #删去操作
    def name(self):
    #print('不让删')
    del self.__name


    peo= people ('qqc') #建对象

    # print(peo.name) #查看
    peo.name= 'QQC' #修改操作
    print(peo.name)
    # del peo.name #删除操作
    # print(peo.__dict__ )

    3、绑定方法与非绑定方法

    (1)绑定方法

    特性:绑定给谁就应该由谁来调用,谁里啊调用就会将谁当做第一个参数自定传入

    注:自动传值

    绑定方法分为两类:

    1.1 绑定给对象的方法

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

    1.2 绑定给类的方法

               在类内部定义的函数如果被装饰器@classmethod装饰,

               那么则是绑定给类的,应该由类来调用,类来调用就自动将类当做第一个参数自动传入

    (2)非绑定方法

    类中定义的函数如果被装饰器@staticmethod装饰,那么该函数就变成非绑定方法

    即不与类绑定,又不与对象绑定,意味着类与对象都可以来调用

    但是无论谁来调用,都没有任何自动传值的效果,就是一个普通函数

    应用:

    函数体代码需要用外部传入的类,则应该将该函数定义成绑定给类的方法;

    函数体代码需要用外部传入的对象,则应该将该函数定义成绑定给对象的方法;

    如果函数体代码即不需要外部传入的类也不需要外部传入的对象,则应该将该函数定义成非绑定方法/普通函数

    #绑定方法
    class Foo:
    @classmethod
    def f1(cls):
    print(cls)
    def f2(self):
    print(self)

    obj = Foo()
    # print(obj.f2)
    #1、f1是绑定给类的,应该由类来调用,但其实对象也可以调用,但自动传入的参数仍然是类
    # Foo.f1()
    #2、f2是绑定给对象的
    # obj.f2()

    例子:从配置文件中实例化

    import settings
    class mysql:
    def __init__(self,ip,port):
    self.ip= ip
    self.port= port

    def tell_info(self):
    print(self.ip ,self .port)

    @classmethod
    def from_conf(cls): #与类绑定,自动传入类
    return mysql(settings.IP,settings.PORT)

    @staticmethod #非绑定方法,无自动传值
    def func():
    print('不与任何人绑定')

    #默认实例化方式:类名(。。。)
    obj= mysql('10.10.0.9',3307)

    #一种新的实例化方式,从配置文件中读取配置完成实例化
    obj1= mysql.from_conf()
    obj1.tell_info()
    obj1.func()
  • 相关阅读:
    Struts2多文件上传
    Struts2单文件上传
    java验证码
    spring-day01
    Spring MVC篇一、搭建Spring MVC框架
    连接oracle数据库
    spring 核心技术
    Spring的特点
    spring连接数据库
    oracle学习第六天
  • 原文地址:https://www.cnblogs.com/quqinchao/p/9239879.html
Copyright © 2011-2022 走看看