Python全栈day28(描述符应用)
描述符的使用
python是弱类型语言,及参数的赋值没有类型限制,下面通过描述符机制来实现类型限制功能
描述符应用1.py
class Typed:
def __get__(self, instance, owner):
print('get方法')
print('instance是[%s]'%instance)
print('owner是[%s]'%owner) def __set__(self, instance, value):
print('set方法')
#instance就是实例化出来的对象本身
print('instance是[%s]'%instance)
print('value是[%s]'%value) def __delete__(self, instance):
print('delete方法')
print('instance是[%s]' % instance) class People:
name = Typed()
def __init__(self,name,age,salary):
self.name = name
self.age = age
self.salary = salary #实例化触发了以下三个赋值操作
#p1.__dict__['name'] = 'zhangsan'
#p1.__dict__['age'] = 18
#p1.__dict__['salary'] = 999999
#但是数据属性被描述符描述了进行赋值操作调用的是描述类的set方法
#因为set方法没有进行实际赋值操作所以字典的name属性为None
p1 = People('zhangsan',18,999999)
# set方法
# instance是[<__main__.People object at 0x000001E527F67390>]
# value是[zhangsan]
print(p1)
#<__main__.People object at 0x000001BF1B047390>
print(p1.__dict__)
#调用数据属性触发__get__方法输出
p1.name
# get方法
# instance是[<__main__.People object at 0x000001F7F3ED8080>]
# owner是[<class '__main__.People'>]
#调用del删除数据属性,触发__delete__方法输出
del p1.name
# delete方法
# instance是[<__main__.People object at 0x0000018239F274E0>]
以上只是测试是否调用了描述符,但是对应的__get__,__set__,__delete__只是执行了打印操作没有进行返回值,设置值,删除值的操作所以只有打印输出
PS:根据优先级数据描述符大于实例所以优先调用数据描述符,假如进行了p1.name = ‘lisi’会执行set方法但是不会赋值,name依旧为空
下面在__get__,__set__,__delete__执行对应的返回值,设置值,删除值操作
描述符应用2.py
class Typed:
def __init__(self,key):
self.key=key def __get__(self, instance, owner):
print('get方法')
# print('instance是[%s]'%instance)
# print('owner是[%s]'%owner)
#使用对应的key返回值
return instance.__dict__[self.key] def __set__(self, instance, value):
print('set方法')
#instance就是实例化出来的对象本身
# print('instance是[%s]'%instance)
# print('value是[%s]'%value)
#对对应的key进行赋值设置操作
instance.__dict__[self.key] = value def __delete__(self, instance):
print('delete方法')
# print('instance是[%s]' % instance)
#使用对应的key删除操作
instance.__dict__.pop(self.key) class People:
name = Typed('name')
def __init__(self,name,age,salary):
self.name = name
self.age = age
self.salary = salary #实例化触发了以下三个赋值操作
#p1.name = 'zhangsan'
#p1.age = 18
#p1.salary = 999999
#其中p1.name = 'zhangsan'调用了描述符类Typed进行实例化
#name = Typed('name')运行初始化函数__init__
#self.name = 'name'
p1 = People('zhangsan',18,999999)
#set方法
#打印字典name也赋值成功
print(p1.__dict__)
#{'age': 18, 'name': 'zhangsan', 'salary': 999999}
p1.name
#get方法
del p1.name
#delete方法
#打印字典上一步的删除操作也成功删除了属性name
print(p1.__dict__)
#{'age': 18, 'salary': 999999}
通过定义描述符类的初始化__init__函数获取字典需要修改对应的key值然后执行相应的返回值,设置值,删除值的操作
下面对用户实例化输入的信息进行判断,比如输入姓名只能是字符串格式不能是数字或者其他格式
描述符应用3.py
class Typed:
def __init__(self,key):
self.key=key def __get__(self, instance, owner):
print('get方法')
# print('instance是[%s]'%instance)
# print('owner是[%s]'%owner)
#使用对应的key返回值
return instance.__dict__[self.key] def __set__(self, instance, value):
print('set方法')
#instance就是实例化出来的对象本身
# print('instance是[%s]'%instance)
# print('value是[%s]'%value)
#对应的key进行赋值设置操作
if not isinstance(value,str):
print('你输入是不是字符串类型,错误')
return
instance.__dict__[self.key] = value def __delete__(self, instance):
print('delete方法')
# print('instance是[%s]' % instance)
#使用对应的key删除操作
instance.__dict__.pop(self.key) class People:
name = Typed('name')
def __init__(self,name,age,salary):
self.name = name
self.age = age
self.salary = salary p1=People(18,18,999999)
#因为调用__set__方法的时候检测到输入的名字不是字符串,然后直接return了所以name没有赋值
print(p1.__dict__)
#set方法
#你输入是不是字符串类型,错误
#{'salary': 999999, 'age': 18}
可以把错误返回改的高端一点
描述符应用4.py
class Typed:
def __init__(self,key):
self.key=key def __get__(self, instance, owner):
print('get方法')
# print('instance是[%s]'%instance)
# print('owner是[%s]'%owner)
#使用对应的key返回值
return instance.__dict__[self.key] def __set__(self, instance, value):
print('set方法')
#instance就是实例化出来的对象本身
# print('instance是[%s]'%instance)
# print('value是[%s]'%value)
#对应的key进行赋值设置操作
if not isinstance(value,str):
# print('你输入是不是字符串类型,错误')
# return
raise TypeError('%s你传入的不是字符串'%value)
instance.__dict__[self.key] = value def __delete__(self, instance):
print('delete方法')
# print('instance是[%s]' % instance)
#使用对应的key删除操作
instance.__dict__.pop(self.key) class People:
name = Typed('name')
def __init__(self,name,age,salary):
self.name = name
self.age = age
self.salary = salary p1=People(18,18,999999)
运行直接报错

以上只是实现了输入的name必须是字符串,并没有对输入的age限制必须是整型
描述符应用5.py
class Typed:
def __init__(self,key,expected_type):
self.key=key
self.expected_type=expected_type
def __get__(self, instance, owner):
print('get方法')
# print('instance是[%s]'%instance)
# print('owner是[%s]'%owner)
#使用对应的key返回值
return instance.__dict__[self.key] def __set__(self, instance, value):
print('set方法')
#instance就是实例化出来的对象本身
# print('instance是[%s]'%instance)
# print('value是[%s]'%value)
#对应的key进行赋值设置操作
if not isinstance(value,self.expected_type):
# print('你输入是不是字符串类型,错误')
# return
raise TypeError('%s你传入的不是%s'%(value,self.expected_type))
instance.__dict__[self.key] = value def __delete__(self, instance):
print('delete方法')
# print('instance是[%s]' % instance)
#使用对应的key删除操作
instance.__dict__.pop(self.key) class People:
name = Typed('name',str)
age = Typed('age',int)
def __init__(self,name,age,salary):
self.name = name
self.age = age
self.salary = salary p1=People('zhangsan','abc',999999)
print(p1.__dict__)
因为age输入的是字符串所以报错

Python全栈day28(描述符应用)的更多相关文章
- Python全栈day28(类的装饰器)
类是一个对象也可以像函数一样加装饰器 类的装饰器.py def deco(obj): print('======',obj) obj.x=1 obj.y=2 obj.z=3 return obj # ...
- Python全栈--7.1--字符串的格式化
Python字符串格式化:(百分号/format) 1.百分号的方式: %[(name)][flags][width].[precision]typecode (name) 可选,用于选择指 ...
- Python全栈day28(上下文管理)
我们知道在操作文件对象的时候可以这么写 with open('a.txt',''r) as f: 代码 上述叫做上下文管理协议,即with语句,为了让一个对象兼容with语句,必须在这个对象的类中声明 ...
- Python全栈之路目录结构
基础 1.Python全栈之路-----基础篇 2.Python全栈之路---运算符与基本的数据结构 3.Python全栈之路3--set集合--三元运算--深浅拷贝--初识函数 4.Python全栈 ...
- Python全栈【Socket网络编程】
Python全栈[socket网络编程] 本章内容: Socket 基于TCP的套接字 基于UDP的套接字 TCP粘包 SocketServer 模块(ThreadingTCPServer源码剖析) ...
- Python全栈开发【面向对象进阶】
Python全栈开发[面向对象进阶] 本节内容: isinstance(obj,cls)和issubclass(sub,super) 反射 __setattr__,__delattr__,__geta ...
- python 全栈开发,Day43(python全栈11期月考题)
python全栈11期月考题 1.常用字符串格式化有哪些?并说明他们的区别 2.请手写一个单例模式(面试题) 3.利用 python 打印前一天的本地时间,格式为‘2018-01-30’(面试题) 4 ...
- python 全栈开发,Day30(第一次面向对象考试)
月考题: python 全栈11期月考题 一 基础知识:(70分) 1.文件操作有哪些模式?请简述各模式的作用(2分) 2.详细说明tuple.list.dict的用法,以及它们的特点(3分) 3.解 ...
- python全栈开发之正则表达式和python的re模块
正则表达式和python的re模块 python全栈开发,正则表达式,re模块 一 正则表达式 正则表达式(Regular Expression)是一种文本模式,包括普通字符(例如,a 到 z 之间的 ...
随机推荐
- atitit。gui 界面皮肤以及换肤总结 java .net c++
atitit.gui 界面皮肤以及换肤总结 java .net c++ 1. Swing 的皮肤 1 1.1. windows风格 1 1.2. Mac风格 ( liquid 框架) 1 2. 如何给 ...
- Netty的ByteBuf
https://blog.csdn.net/thinking_fioa/article/details/80795673 netty的ByteBuf知识点
- redis基础之订阅发布、主从复制和事务(四)
前面已经学习了redis的基本的命令行操作和数据类型,下面开始redis一些有趣的功能. 订阅和发布机制 定义:发布者相当于电台,订阅者相当于客户端,客户端发到频道的消息,将会被推送到所有订阅此频道的 ...
- Struts2 上传下载
一. 1.文件上传是web应用经常用到的一个知识.原理是,通过为表单元素设置enctype=”multipart/form-data”属性,让表单提交的数 据以二进制编码的方式提交,在接收此请求的Se ...
- 在windows中使用Navicat连接Linux虚拟机中的mysql数据库
今天想用navicat远程连接虚拟机中的MySQL数据库,一直连不上,在网上搜索了一下,发现原因是MySQL对远程用户登陆的授权问题.这里说一下我的解决方法.(本人小白) 首先,我用navicat去远 ...
- AM335x 添加 HUAWEI MU609 Mini PCIe Module,并用pppd 启动相关设备
kernel 的配置 kernel 3.2.0 make menuconfig Device Drivers ---> [*] USB support ---> <*> USB ...
- PHP uxf framework 在模版中加入url标签
1. 确保不修改discuz代码: 2. 继承discuz template类,重载parse_template 方法:由于discuz在模版引擎这一块没有考虑扩展性,对标签的解析全部写在一个方法中, ...
- [Shell Script]关于source和sh对于脚本执行不同
当我修改了/etc/profile文件,我想让它立刻生效,而不用重新登录:这时就想到用source命令,如:source /etc/profile对source进行了学习,并且用它与sh 执行脚本进行 ...
- Android——RelativeLayout(相对布局)
一.相对于父容器 1.居中 2.同方向 <?xml version="1.0" encoding="utf-8"?> <RelativeLay ...
- Spider Studio 新版本 (20140108) - 优化设置菜单 / 生成程序集支持版本号
本次更新包含两项改进: 1. 优化了设置菜单, 去掉了一些不必要的浏览器行为设置选项: 取而代之的是在脚本中由用户自行设置: public void Run() { Default.CaptureNe ...