Python中用Descriptor实现类级属性(Property)详解


Posted in Python onSeptember 18, 2014

上篇文章简单介绍了python中描述器(Descriptor)的概念和使用,有心的同学估计已经Get√了该技能。本篇文章通过一个Descriptor的使用场景再次给出一个案例,让不了解情况的同学可以更容易理解。

先说说decorator

这两个单词确实是有些相似,同时在使用中也是形影不离。这也给人造成了理解上的困难,说装饰器和描述器到底是怎么回事,为什么非得用一个@符号再加上描述器才行。

很多文章也都把这俩结合着讲,我自己看完之后都会觉得很绕。其实学习一个知识点,和做项目开发一个功能是一样的。在功能拆分的时候我们都会尽量的把任务拆分的足够小,然后才分配到开发者头上。这样保证各个任务的独立性,完整性,并且易于做进度管理。在任务开发的时候也不能把你的任务都放到一个函数/接口中去做,以避免各功能间产生高耦合的状况,导致后期难以维护。

再说回到学习一个技术点,如果你总是尝试一下子就要掌握两个或多个技术点,结果可能是忙活了半天,发现还是晕头转向。

擦,好像扯远了。

说Descriptor是Descriptor, Decorator是Decorator,遇到不懂的地方,各个击破,哪里不懂点哪里。所以先说Decorator, 关键点是你要意识到这就是一个语法糖 。所谓语法糖就是让你可以用简单的方式写代码。本质上装饰器(Decorator)就是这样:

def decorator(func):

    def wrapper():

        print 'in decorator'

        func()

    return wrapper
def func():

    print 'in func'
# 把func装饰一下

func = decorator(func)  # 左边的func其实是那个wrapper, 你执行它的时候会,它会帮你执行func()

# 等同于你在定义func的时候加上@

@decorator

def func():

    print 'in func'

正题:通过Descriptor来做一个类级的Property

常见的Property是这样的:

class Foo(object):

    _name = 'the5fire'
    @property

    def name(self):

        return self._name

这中property的使用,是实例级的应用。因为只有在 foo = Foo() 之后,才可以 foo.name 。

但是如果我需要一个类级的属性应该怎么做呢,就像是 classmethod一样,不需要实例化类我就可以调用。对应的需求是这样的,定义了一个基类DBManage:

class DBManage(object):

    @classmethod

    def table_name(cls):

        return cls.__name__.lower()
    @classmethod

    def select_all(cls):

        sql = "SELECT * FROM %s""" % cls.table_name()

        # 执行这个语句的代码

        return result

这其实一个对应着数据库中某张表的基础模型,我希望其他的Model都来继承它,然后可以重用这个table_name的方法(目前还是方法)。

我只需要这么定义User模型即可:

class User(DBManage):

    pass

然后这么定义Post模型:
class Post(DBManage):

    pass

这样我如果需要查所有的User数据,只需要 User.select_all() 即可,同理Post也是如此 Post.select_all() 。但此时发现一个有点不爽的事情。那就是基类中的 cls.table_name() 这个代码,table_name看起来就是属性,却需要用调用方法的方式获取。不妥。

于是自定义了一个classproperty:

class classproperty(object):

    def __init__(self, func):

        self.func = func
    def __get__(self, instance, klass):

        return self.func(klass)

这需要这样,我在DBManage中的代码就可以改为:

class DBManage(object):

    @classproperty

    def table_name(cls):

        return cls.__name__.lower()
    @classmethod

    def select_all(cls):

        sql = "SELECT * FROM %s""" % cls.table_name  # 多么直观

这就是Descriptor另外的一个使用案例了。
可能有人或有一个小疑问:为毛你不是在sql赋值时直接 sql = "SELECT * FROM %s" % cls.__name__.lower() 。这个问题,问的非常好,原因就一个字:懒。懒得以后每次都得敲那么多代码。
Python 相关文章推荐
python简单实现基数排序算法
May 16 Python
python检查字符串是否是正确ISBN的方法
Jul 11 Python
Python的Django框架中的数据过滤功能
Jul 17 Python
Python中的FTP通信模块ftplib的用法整理
Jul 08 Python
详解opencv Python特征检测及K-最近邻匹配
Jan 21 Python
六行python代码的爱心曲线详解
May 17 Python
python实现飞机大战小游戏
Nov 08 Python
Python算法中的时间复杂度问题
Nov 19 Python
python__new__内置静态方法使用解析
Jan 07 Python
pycharm 2019 最新激活方式(pycharm破解、激活)
Sep 22 Python
python 5个顶级异步框架推荐
Sep 09 Python
Python urlopen()参数代码示例解析
Dec 10 Python
Python中的闭包总结
Sep 18 #Python
python的即时标记项目练习笔记
Sep 18 #Python
python脚本实现分析dns日志并对受访域名排行
Sep 18 #Python
python中的字典详细介绍
Sep 18 #Python
python中执行shell命令的几个方法小结
Sep 18 #Python
python处理PHP数组文本文件实例
Sep 18 #Python
Python threading多线程编程实例
Sep 18 #Python
You might like
隐藏X-Space个人空间下方版权方法隐藏X-Space个人空间标题隐藏X-Space个人空间管理版权方法
2007/02/22 PHP
深入理解PHP原理之Session Gc的一个小概率Notice
2011/04/12 PHP
关于php正则匹配汉字的方法介绍
2013/04/25 PHP
理解PHP中的stdClass类
2014/04/18 PHP
让ThinkPHP支持大小写url地址访问的方法
2014/10/31 PHP
PHP实现HTML页面静态化的方法
2015/11/04 PHP
Symfony2框架学习笔记之表单用法详解
2016/03/18 PHP
thinkPHP简单导入和使用阿里云OSSsdk的方法
2017/03/15 PHP
PHP编译configure时常见错误的总结
2017/08/17 PHP
jQuery实现的类flash菜单效果代码
2010/05/17 Javascript
jquery下将选择的checkbox的id组成字符串的方法
2010/11/28 Javascript
JavaScript中的ubound函数使用实例
2014/11/04 Javascript
AngularJS基础知识笔记之表格
2015/05/10 Javascript
理解JavaScript表单的基础知识
2016/01/25 Javascript
JavaScript中省略元素对数组长度的影响
2016/10/26 Javascript
jQuery简单实现对数组去重及排序操作实例
2017/10/31 jQuery
使用JavaScript通过前端发送电子邮件
2020/05/22 Javascript
python 从远程服务器下载东西的代码
2013/02/10 Python
利用Anaconda完美解决Python 2与python 3的共存问题
2017/05/25 Python
Python爬取附近餐馆信息代码示例
2017/12/09 Python
解决Pycharm中import时无法识别自己写的程序方法
2018/05/18 Python
Python数据可视化之画图
2019/01/15 Python
对python3中, print横向输出的方法详解
2019/01/28 Python
节日快乐! Python画一棵圣诞树送给你
2019/12/24 Python
Python终端输出彩色字符方法详解
2020/02/11 Python
利用 PyCharm 实现本地代码和远端的实时同步功能
2020/03/23 Python
python爬虫中的url下载器用法详解
2020/11/30 Python
Python+unittest+requests+excel实现接口自动化测试框架
2020/12/23 Python
美国翻新电子产品商店:The Store
2019/10/08 全球购物
土木工程专业大学毕业生求职信
2013/10/13 职场文书
什么是岗位职责
2013/11/12 职场文书
物流专业大学生的自我鉴定
2013/11/13 职场文书
理财计划书
2014/08/14 职场文书
2015年环卫处个人工作总结
2015/07/27 职场文书
2016年少先队活动总结
2016/04/06 职场文书
vue实现简单数据双向绑定
2021/04/28 Vue.js