Python封装与隐藏
今日内容:
1.封装与隐藏
2.property
3.绑定方法与非绑定方法
知识点一:封装与隐藏
1、什么封装:
封:属性对外是隐藏的,但对内是开放的
装:申请一个名称空间,往里装入一系列名字/属性
2、为什么要封装:
封装数据属性的目的
首先定义属性的目的就是为了给类外部的使用者使用的,
隐藏之后是为了不让外部使用者直接使用,需要类内部开辟一个接口
然后让类外部的使用通过接口来间接地操作隐藏的属性。
精髓在于:我们可以在接口之上附加任意逻辑,从而严格控制使用者对属性的操作
封装函数属性:
首先定义属性的目的就是为了给类外部的使用使用的,
隐藏函数属性是为了不让外不直接使用,需要类内部开辟一个接口
然后在接口内去调用隐藏的功能
精髓在于:隔离了复杂度
3、如何隐藏:
1、 这种隐藏仅仅只是一种语法上的变形操作
2、 这种语法上的变形只在类定义阶段发生一次,因为类体代码仅仅只在类定义阶段检测一次
3、 这种隐藏是对外不对内的,即在类的内部可以直接访问,而在类的外则无法直接访问,原因是
在类定义阶段,类体内代码统一发生了一次变形
4、 如果不想让子类的方法覆盖父类的,可以将该方法名前加一个__开头
class People:
def __init__(self,name,age):
self.__name=name
self.__age=age def tell_info(self):
print('%s:%s' %(self.__name,self.__age)) def set_info(self,name,age):
if type(name) is not str:
# print('用户名必须为str类型')
# return
raise TypeError('用户名必须为str类型') if type(age) is not int:
# print('年龄必须为int类型')
# return
raise TypeError('年龄必须为int类型')
self.__name=name
self.__age=age peo1=People('张三',34)
# peo1.name=123
# peo1.age
peo1.tell_info() peo1.set_info('李四',19)
peo1.tell_info()
知识点二:property装饰
property装饰器用于将被装饰的方法伪装成一个数据属性,
在使用时可以不用加括号而直接引用
》》》@propertyde思路演变的过程
版本1:正常版本
class People:
def __init__(self,name,weight,height):
self.name=name
self.weight=weight
self.height=height def bmi(self):
return self.weight / (self.height**2) peo=People('张三',130,1.65)
print(peo.name) #张三
print(peo.bmi()) #47.75022956841139 问题:发现bmi的访问方式比访问对象属性多了个()
思路变化1:正常情况下我们访问对象里面属性和类里面的函数方式为:
思路变化2:为了方便用户调用,能不能让用户已访问对象属性的方式去访问bmi print(peo.bmi)
》》》》》》版本2:@property初始版本
#思考:如何实现改变访问bmi的方式为》》》print(peo.bmi)
# 使用@property:作用是将本装饰的方法伪装成一个数据类型,在使用时可以不用加括号而直接使用 class People:
def __init__(self,name,weight,height):
self.name=name
self.weight=weight
self.height=height @property #将bmi伪装成一个数据类型
def bmi(self):
return self.weight / (self.height**2) peo=People('张三',130,1.65)
print(peo.name) #张三
print(peo.bmi) #这样可以直接调用而不用加括号
》》》》》完整版:@property
#property装饰器
#是一种封装思想的体现,方便了查 删 改 class people:
def __init__(self,name):
self.__name=name @property
def name(self): #查看obj.name
return '名字是:%s'%self.__name @name.setter #更改obj.name=name
def name(self,name):
if type(name) is not str:
raise TypeError('名字必须是str类型')
self.__name=name
@name.deleter #删除 del obj.name
def name(self):
print('不让删除') #里面可以设定提示
# del self.__name #也可以设定直接删除功能 peo=people('张三')
#默认不加@property调用方法为: peo.name='李四' #@name.setter 调用了更改
print(peo.name) #@property 调用了查看 输出结果:名字是:李四
del peo.name #@name.deleter 调用了产出的需求 输出结果:不让删除
知识点三:绑定方法
1.绑定方法
特性:绑定给谁就应该由谁来调用,谁来调用就会将谁当做第一个参数自动传入
精髓:是在于自动传值
1.1绑定对象的方法
在类内部定义的函数(没有被任何装饰器修饰的),默认是绑定给对象用的
1.2绑定给类的方法
在类内部定义的函数如果被装饰器@classmethod装饰,
那么则是绑定给类的,应该由类来调用,类来调用就自动将类当做第一个参数自动传入
绑定给类的:自动传的是类的值
绑定给对象的:自动传的是对象的值
2.非绑定方法
类中定义的函数如果被装饰器@staticmethod装饰,那么该函数就变成非绑定方法
既不与类绑定,又不与对象绑定,意味着类与对象都可以来调用
但是无论谁来调用,都没有任何自动化传值的效果,就是一个普通函数
3.应用
什么时候绑定给类? 什么时候绑定给对象?
取决于函数体代码
如果函数体代码需要用外部传入的类,则应该将该函数定义成绑定类的方法
如果函数体代码需要用外部传入的对象,则应该将该函数定义成绑定给对象的方法
如果函数体代码既不需要外部传入的类也不需要外部传入的对象,则应该将该函数定义成非绑定方法对象,则应该将该函数定义成非绑定方法
#绑定方法初始实例:绑定给类@classmethod class Foo:
@classmethod
def f1(cls):
print('Foo.f1')
print(cls) def f2(self):
print(self)
obj=Foo()
# print(obj.f2) #默认绑定给对象用#<bound method Foo.f2 of <__main__.Foo object at 0x04BACD10>>
# print(Foo.f1) #绑定给类用 <bound method Foo.f1 of <class '__main__.Foo'>>
Foo.f1() #类调用
print(Foo) #所以传入的值就是类的值 <class '__main__.Foo'> obj.f2() #对象调用,
print(obj) #所以传入的值就是对象的值 <__main__.Foo object at 0x04C50EB0> # f1绑定给类的应该由类来调用,但对象其实也可以使用,只不过自动传入的任然是类
#f2绑定给对象的应该由对象来调用,但是类其实也可以使用
# 绑定方法进阶实例2:新的实例化方式,绑定给类
"""
将需要传入的信息写到settings.py文件里面,调用更加灵活方便
IP='192.168.11.3'
PORT=3306
"""
import settings class Mysql:
def __init__(self,ip,port):
self.ip=ip
self.port=port
def tell_info(self):
print('%s:%s'%(self.ip,self.port))
# print(self) @classmethod
def from_conf(cls):
return cls(settings.IP,settings.PORT) #可以理解为 Mysql('1.1.1.1',23)即调用了init方法,给其传参数 cls名字可以随便起
# print(settings.IP) #默认的实例化方式:类名(..)(实质给tell_info(self)里的self传的是对象res的值)
res=Mysql('1.1.1.1',23)
print(res) #<__main__.Mysql object at 0x05717410>即为对象res也就是tell_info(self)的值
res.tell_info() # 一种新的实例化方式,从配置文件中读取配置完成实例化(这里实质给cls传的是类Mysql的值)
res=Mysql.from_conf()
res.tell_info()
print(res.ip)
非绑定方法应用实例:
#非绑定方法应用实例:uuid获取随机id值 import uuid
class UserInfo:
def __init__(self,name,age): self.name=name
self.age=age
self.uid = self.create_uid()
def tell_info(self):
print('姓名:%s,年龄:%s,ID:%s'%(self.name,self.age,self.uid))
# print(self) @staticmethod #因为不需要传任何参数所以定义为非绑定方法,类和对象都可以取调用
def create_uid():
return uuid.uuid1() res=UserInfo('张三',23,)
res.tell_info()
print(res.uid)
Python封装与隐藏的更多相关文章
- 面向对象的封装与隐藏 this
当我们创建一个对象的时候,我们可以通过‘对象.属性’的方式,对对象的属性进行赋值. 这里赋值操作要受到属性的数据类型和存储范围的制约,但是除此之外,没有其他制约条件. 但是实际问题中我们需要给这个属性 ...
- python 封装,隐藏属性,绑定方法classmethod和staticmethod
[封装] 隐藏对象的属性和实现细节,仅对外提供公共访问方式. [好处] 1. 将变化隔离: 2. 便于使用: 3. 提高复用性: 4. 提高安全性: [封装原则] 1. 将不需要对外提供的内容都隐藏起 ...
- python 封装底层实现原理
事实上,python封装特性的实现纯属"投机取巧",之所以类对象无法直接调用私有方法和属性,是因为底层实现时,python偷偷改变了它们的名称. python在底层实现时,将它们的 ...
- python 封装、绑定
目录 python 封装.绑定 1.数据.方法的封装 2.隐藏属性 3.开放接口 4.绑定方法 1.对象的绑定 2.类的绑定(classmethod) 3.非绑定方法(staticmethod) 4. ...
- python封装configparser模块获取conf.ini值(优化版)
昨天晚上封装了configparser模块,是根据keyname获取的value.python封装configparser模块获取conf.ini值 我原本是想通过config.ini文件中的sect ...
- 使用boost.python封装C++库
使用boost.python封装C++库 C++以高性能著称,但是编写较为复杂.而简洁是Python的强项.如果能珠联璧合,就能发挥两家之长.本文尝试用boost库的python模块封装C++ 前期准 ...
- Python实现图像信息隐藏
Python实现图像信息隐藏 之前学习密码学的时候老师有提到过『信息隐藏』,现在用图像的方法尝试一下.思想是:把信息藏到RGB通道中的B通道,然后利用奇偶性可以恢复过来 原理 从源图中提取文字图像信息 ...
- 孤荷凌寒自学python第二十四天python类中隐藏的私有方法探秘
孤荷凌寒自学python第二十四天python类中隐藏的私有方法探秘 (完整学习过程屏幕记录视频地址在文末,手写笔记在文末) 今天发现了python的类中隐藏着一些特殊的私有方法. 这些私有方法不管我 ...
- 【AMAD】python-magic -- libmagic的python封装
简介 动机 作用 用法 个人评分 简介 libmagic的python封装 动机 封装libmagic,使用python代码获取文件类型. 作用 libmagic通过文件头部,来确定文件的类型. 用法 ...
随机推荐
- Linux系统常用命令大全
一.系统信息操作(备注:红色标记为常用命令,以下类推,不再赘述) arch 显示机器的处理器架构(1) uname -m 显示机器的处理器架构(2) uname -r ...
- LeetCode:103Binary Tree Zigzag Level Order Traversal
真是不容易啊,做这道题的时候脑子一团乱,感觉还是得劳逸结合啊.这道题的思想不难,就是宽搜BFS.通过设置一个flag来判断是否需要逆序输出. 我的做法虽然AC,但是觉得代码还是不好,空间占用较多. / ...
- Python中的绝对路径和相对路径
大牛们应该对路径都很了解了,这篇文章主要给像我这样的入门小白普及常识用的,啊哈 下面的路径介绍针对windows,其他平台的暂时不是很了解. 在编写的py文件中打开文件的时候经常见到下面其中路径的表达 ...
- 正确配置Nginx+PHP
对很多人而言,配置Nginx+PHP无外乎就是搜索一篇教程,然后拷贝粘贴.听上去似乎也没什么问题,可惜实际上网络上很多资料本身年久失修,漏洞百出,如果大家不求甚解,一味的拷贝粘贴,早晚有一天会为此付出 ...
- Jquery库插件大全(工作中遇到总结)
Jquery UI所有插件下载:http://jqueryui.com/download/all/ Jquery layer灯箱等演示与帮助:http://sentsin.com/jquery/lay ...
- Beta冲刺(周四)
这个作业属于哪个课程 https://edu.cnblogs.com/campus/xnsy/SoftwareEngineeringClass1 这个作业要求在哪里 https://edu.cnblo ...
- 2018.4.11 Java为何用xml做配置文件? 理由如下
在Java世界里xml配置文件几乎是首选,xml有什么好的特性呢? 第一:xml能存储小量数据,仅仅是存储数据. 第二:xml可以跨平台,主流各种平台都对xml有支持, 真正的跨平台, 第三:xml读 ...
- AutoWidthInput
import React from 'react'; import PropTypes from 'prop-types'; class AutoWidthInput extends React.Co ...
- 关于List的remove方法我遇到的坑
结果是下标越界异常,原因是remove方法的参数不是value,而是index 唉~~~ 年少轻狂啊
- GIT在团队中的最佳实践
我们已经从SVN 切换到Git很多年了,现在几乎所有的项目都在使用Github管理, 本篇文章讲一下为什么使用Git, 以及如何在团队中正确使用. Git的优点 Git的优点很多,但是这里只列出我认为 ...