使用描述符为python实现类型检测

class Typed:
    def __get__(self, instance, owner):
        print(instance)
        print(owner)

    def __set__(self, instance, value):
        pass

class Girl:
    name = Typed()

    def __init__(self, name, age):
        self.name = name  # 我希望传入的name是str
        self.age = age   # 我希望传入的age是int

g = Girl("satori", 18)
g.name
'''
<__main__.Girl object at 0x03D28430>
<class '__main__.Girl'>
'''
# 可以看到当我们获取被代理的值,会触发get方法,instance就是Girl的实例对象
# owner则是Girl这个类

  

class Typed:
    def __get__(self, instance, owner):
        pass

    def __set__(self, instance, value):
        print(instance)
        print(value)

class Girl:
    name = Typed()

    def __init__(self, name, age):
        self.name = name  # 我希望传入的name是str
        self.age = age   # 我希望传入的age是int

g = Girl("satori", 18)
'''
<__main__.Girl object at 0x03D28430>
satori

'''
# 可以看到当我们给被代理的值赋值的时候,会触发set方法,instance就是Girl的实例对象
# value则是我们赋的值

  

class Typed:
    def __init__(self, key):
        self.key = key

    def __get__(self, instance, owner):
        return instance.__dict__[self.key]

    def __set__(self, instance, value):
        instance.__dict__[self.key] = value

    def __delete__(self, instance):
        instance.__dict__.pop(self.key)

class Girl:
    name = Typed("name")
    age = Typed("age")

    def __init__(self, name, age):
        self.name = name
        self.age = age

g = Girl("satori", 18)
print(g.__dict__)
g.name = "mashiro"
print(g.__dict__)
del g.name
print(g.__dict__)

'''
{'name': 'satori', 'age': 18}
{'name': 'mashiro', 'age': 18}
{'age': 18}
'''

# 等于说  我饶了一圈,并没有直接将属性添加的字典里面,而是向上找了代理,让代理帮我把属性添加的字典里面去

  那么,接下来就可以实现类型检测了

class Typed:
    def __init__(self, key, except_type):
        self.key = key
        self.except_type = except_type

    def __get__(self, instance, owner):
        return instance.__dict__[self.key]

    def __set__(self, instance, value):
        if isinstance(value, self.except_type):
            instance.__dict__[self.key] = value
        else:
            raise Exception(f"{self.key} must be type {self.except_type}")

    def __delete__(self, instance):
        instance.__dict__.pop(self.key)

class Girl:
    name = Typed("name", str)
    age = Typed("age", int)

    def __init__(self, name, age):
        self.name = name
        self.age = age

g = Girl("satori", 18)
print(g.__dict__)  # {'name': 'satori', 'age': 18}

try:
    import traceback
    g = Girl("satori", "18")

except Exception:
    print(traceback.format_exc())

'''
Exception: age must be type <class 'int'>
'''

# 于是我们便手动实现了python的类型检测

  

类的装饰器

# 装饰器不仅可以给函数用,还可以给类用

def deco(obj):
    print("======")
    return obj

@deco
class Girl:
    def __init__(self, name, age):
        self.name = name
        self.age = age

g = Girl("satori", 18)
print(g.__dict__)

'''
======
{'name': 'satori', 'age': 18}
'''

  

描述符实现property

class LazeProperty:

   def __init__(self, func):
       self.func = func

   def __get__(self, instance, owner):

       # 如果是类调用,那么instance为空
       if instance is None:
           return self
       res = self.func(instance)
       setattr(instance, self.func.__name__, res)  # 将值设进去,如果有从字典里面找,没有再计算
       return res

class Girl:

    def __init__(self, name, age):
        self.name = name
        self.age = age

    @LazeProperty  # info = LazeProperty(info)
    def info(self):
        return f"my name is {self.name},age is {self.age}"

g = Girl("satori", 18)
print(g.info)  # my name is satori,age is 18

print(Girl.info)  # <__main__.LazeProperty object at 0x04EF3910>

  

python描述符的应用的更多相关文章

  1. 杂项之python描述符协议

    杂项之python描述符协议 本节内容 由来 描述符协议概念 类的静态方法及类方法实现原理 类作为装饰器使用 1. 由来 闲来无事去看了看django中的内置分页方法,发现里面用到了类作为装饰器来使用 ...

  2. python描述符(descriptor)、属性(property)、函数(类)装饰器(decorator )原理实例详解

     1.前言 Python的描述符是接触到Python核心编程中一个比较难以理解的内容,自己在学习的过程中也遇到过很多的疑惑,通过google和阅读源码,现将自己的理解和心得记录下来,也为正在为了该问题 ...

  3. 【转载】Python 描述符简介

    来源:Alex Starostin 链接:www.ibm.com/developerworks/cn/opensource/os-pythondescriptors/ 关于Python@修饰符的文章可 ...

  4. python描述符descriptor(一)

    Python 描述符是一种创建托管属性的方法.每当一个属性被查询时,一个动作就会发生.这个动作默认是get,set或者delete.不过,有时候某个应用可能会有 更多的需求,需要你设计一些更复杂的动作 ...

  5. python描述符 descriptor

    descriptor 在python中,如果一个新式类定义了__get__, __set__, __delete__方法中的一个或者多个,那么称之为descriptor.descriptor通常用来改 ...

  6. Python描述符的使用

    Python描述符的使用 前言 作为一位python的使用者,你可能使用python有一段时间了,但是对于python中的描述符却未必使用过,接下来是对描述符使用的介绍 场景介绍 为了引入描述符的使用 ...

  7. Python描述符 (descriptor) 详解

    1.什么是描述符? python描述符是一个“绑定行为”的对象属性,在描述符协议中,它可以通过方法重写属性的访问.这些方法有 __get__(), __set__(), 和__delete__().如 ...

  8. python描述符和属性查找

    python描述符 定义 一般说来,描述符是一种访问对象属性时候的绑定行为,如果这个对象属性定义了__get__(),__set__(), and __delete__()一种或者几种,那么就称之为描 ...

  9. Iterator Protocol - Python 描述符协议

    Iterator Protocol - Python 描述符协议 先看几个有关概念, iterator 迭代器, 一个实现了无参数的 __next__ 方法, 并返回 '序列'中下一个元素,在没有更多 ...

  10. Python描述符以及Property方法的实现原理

    Python描述符以及Property方法的实现原理 描述符的定义: 描述符是什么:描述符本质就是一个新式类,在这个新式类中,至少实了__get__(),__set__(),__delete__()中 ...

随机推荐

  1. 大话循环神经网络(RNN)

      在上一篇文章中,介绍了 卷积神经网络(CNN)的算法原理,CNN在图像识别中有着强大.广泛的应用,但有一些场景用CNN却无法得到有效地解决,例如: 语音识别,要按顺序处理每一帧的声音信息,有些结果 ...

  2. 15.4,redis不重启,切换RDB备份到AOF备份

      确保redis版本在2.2以上 [root@pyyuc /data ::]#redis-server -v Redis server v= sha=: malloc=jemalloc- bits= ...

  3. Entity FrameWork和Dapper的使用

    EF是微软系列下的更正苗红的重量级的ORM框架,功能强大,操作数据库的时候几乎不用写sql,可以像写C#代码一样操作数据库,尤其支持多表关联操作的时候极为方便,但是生成的sql语句性能很差,实在不敢恭 ...

  4. linux udp c/s

    一.UDP C/S编程的步骤如下图所示 二.与TCP C/S通信的区别在于:服务端没有设置监听和等待连接的过程.客户端没有连接服务端的过程.基于UDP的通信时不可靠地,面向无连接的,发送的数据无法确切 ...

  5. 《Cracking the Coding Interview》——第5章:位操作——题目4

    2014-03-19 06:15 题目:解释(n & (n - 1)) == 0是什么意思? 解法:n&n-1是去掉最低位‘1’的方法.根据运算符优先级,貌似用不着加那个括号,但位运算 ...

  6. 【Kth Smallest Element in a BST 】cpp

    题目: Given a binary search tree, write a function kthSmallest to find the kth smallest element in it. ...

  7. Redis常用知识

    1.redis是什么 redis是一种支持Key-Value等多种数据结构的存储系统.可用于缓存.事件发布或订阅.高速队列等场景.该数据库使用ANSI C语言编写,支持网络,提供字符串.哈希.列表.队 ...

  8. [译]9-spring bean的生命周期

    spring中bean的生命周期比较容易理解.bean在实例化之后有时需要调用某个初始化方法进行一些初始化的工作.同样的 ,当bean在销毁之前有时需要做一些资源回收的工作. 尽管bean在实例化和销 ...

  9. 从今天开始学Python

    外部链接下载吧 1.  Python 3.63.chm   AIP 帮助文档  下载:https://pan.baidu.com/s/1lhpv8JTC3Z7B6aZ3qQi40g 2.  VMwar ...

  10. (笔记) RealTimeRender[实时渲染] C3

    @author: 白袍小道 转载表明,查看随缘 前言: 从历史上看,图形加速始于每个像素扫描线上的插值颜色重叠一个三角形,然后显示这些值.包括访问图像数据允许纹理应用于表面.添加硬件 插入和测试z深度 ...