1.  

今日内容:

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 printpeo.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)

Python封装与隐藏的更多相关文章

  1. 面向对象的封装与隐藏 this

    当我们创建一个对象的时候,我们可以通过‘对象.属性’的方式,对对象的属性进行赋值. 这里赋值操作要受到属性的数据类型和存储范围的制约,但是除此之外,没有其他制约条件. 但是实际问题中我们需要给这个属性 ...

  2. python 封装,隐藏属性,绑定方法classmethod和staticmethod

    [封装] 隐藏对象的属性和实现细节,仅对外提供公共访问方式. [好处] 1. 将变化隔离: 2. 便于使用: 3. 提高复用性: 4. 提高安全性: [封装原则] 1. 将不需要对外提供的内容都隐藏起 ...

  3. python 封装底层实现原理

    事实上,python封装特性的实现纯属"投机取巧",之所以类对象无法直接调用私有方法和属性,是因为底层实现时,python偷偷改变了它们的名称. python在底层实现时,将它们的 ...

  4. python 封装、绑定

    目录 python 封装.绑定 1.数据.方法的封装 2.隐藏属性 3.开放接口 4.绑定方法 1.对象的绑定 2.类的绑定(classmethod) 3.非绑定方法(staticmethod) 4. ...

  5. python封装configparser模块获取conf.ini值(优化版)

    昨天晚上封装了configparser模块,是根据keyname获取的value.python封装configparser模块获取conf.ini值 我原本是想通过config.ini文件中的sect ...

  6. 使用boost.python封装C++库

    使用boost.python封装C++库 C++以高性能著称,但是编写较为复杂.而简洁是Python的强项.如果能珠联璧合,就能发挥两家之长.本文尝试用boost库的python模块封装C++ 前期准 ...

  7. Python实现图像信息隐藏

    Python实现图像信息隐藏 之前学习密码学的时候老师有提到过『信息隐藏』,现在用图像的方法尝试一下.思想是:把信息藏到RGB通道中的B通道,然后利用奇偶性可以恢复过来 原理 从源图中提取文字图像信息 ...

  8. 孤荷凌寒自学python第二十四天python类中隐藏的私有方法探秘

    孤荷凌寒自学python第二十四天python类中隐藏的私有方法探秘 (完整学习过程屏幕记录视频地址在文末,手写笔记在文末) 今天发现了python的类中隐藏着一些特殊的私有方法. 这些私有方法不管我 ...

  9. 【AMAD】python-magic -- libmagic的python封装

    简介 动机 作用 用法 个人评分 简介 libmagic的python封装 动机 封装libmagic,使用python代码获取文件类型. 作用 libmagic通过文件头部,来确定文件的类型. 用法 ...

随机推荐

  1. cocos2d-x入门学习篇;切换场景

    手机游戏开发最近很火爆,鉴于一直在学习c++,看起来上手就比较快了.这篇文章来自皂荚花 cocos2d-x技术,我把我的想法分享给大家. 首先来看一段代码: CCScene* HelloWorld:: ...

  2. 用Python完成根据日期计算是星期几

    import datetime def week(year,month,day): someday=dayetime.date(year,month,day) result={ "0&quo ...

  3. Java 方法重载与引用数组类型

    1.方法重载  1)方法的签名 方法的签名包含方法名和参数列表 一个类中,不可以有两个方法的签名完全相同,即一个类中不能有两个方法的方法名和参数列表都一样. public class Test{ pu ...

  4. asp页面无法访问,可尝试开始SQL Server等服务

    存在问题 asp页面的英文提示,翻译后为: "一个错误发生在服务器在处理URL.请联系系统管理员(管理人).如果您是系统管理员,请点击这里了解更多关于这个错误."   解决方案 请 ...

  5. 融云红包全新升级,让App用户更便捷地用“钱”交流感情!

    随着移动互联网的飞速发展,如何增强社交关系.留住用户的心已成为移动社交化时代各类App持续探索的问题,除了接入即时通讯的能力,众多社交平台开始通过趣味性十足的红包功能为App中的社交场景赋能.当即时通 ...

  6. Array - RemoveDuplicatesfromSortedArray

    /** * 无额外空间,只要前n个是不重复的就行,不需要修改后面的数字 * @param nums 已排序的数组 * @return 去除重复数字后的长度 */ public int removeDu ...

  7. JS编程规范指南

    原文:github.com/ryanmcdermott/clean-code-javascript 说明:本文翻译自 github 上的一个项目,只取部分精华. 一.变量 用有意义且常用的单词命名 / ...

  8. python_105_类的特殊成员方法

    aa.py class C(): def __init__(self): self.name='QiZhiguang' 类的特殊成员方法: # 1. __doc__ 表示类的描述信息 class Do ...

  9. linux程序安装及包管理

    程序包的封装类型: RPM软件包:扩展名为“.rpm”,使用rpm命令安装. DEB软件包:扩展名为“.deb”,使用DPKG包管理器. 源代码软件安装:程序员开发完成的原始代码,一般制作成“.tar ...

  10. java中的同步与异步

    在多线程的环境中,经常会碰到数据的共享问题,即当多个线程需要访问同一个资源时,它们需要以某种顺序来确保该资源在某--时刻只能被-一个线程使用,否则,程序的运行结果将会是不可预料的,在这种情况下就必须对 ...