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 Sleep休眠函数使用简单实例
Feb 02 Python
Python和GO语言实现的消息摘要算法示例
Mar 10 Python
使用简单工厂模式来进行Python的设计模式编程
Mar 01 Python
python遍历目录的方法小结
Apr 28 Python
Python+树莓派+YOLO打造一款人工智能照相机
Jan 02 Python
python验证码识别教程之滑动验证码
Jun 04 Python
Sanic框架基于类的视图用法示例
Jul 18 Python
详解python tkinter教程-事件绑定
Mar 28 Python
Django 批量插入数据的实现方法
Jan 12 Python
Python动态导入模块:__import__、importlib、动态导入的使用场景实例分析
Mar 30 Python
使用python实现名片管理系统
Jun 18 Python
快速一键生成Python爬虫请求头
Mar 04 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
php echo, print, print_r, sprintf, var_dump, var_expor的使用区别
2013/06/20 PHP
浅析php变量作用域的一些问题
2013/08/08 PHP
php页面缓存方法小结
2015/01/10 PHP
thinkPHP使用post方式查询时分页失效的解决方法
2015/12/09 PHP
PHP中类属性与类静态变量的访问方法示例
2016/07/13 PHP
php array_multisort 对数组进行排序详解及实例代码
2016/10/27 PHP
Laravel如何实现适合Api的异常处理响应格式
2020/06/14 PHP
Aster vs KG BO3 第一场2.19
2021/03/10 DOTA
关于js类的定义
2011/06/28 Javascript
jquery.Jwin.js 基于jquery的弹出层插件代码
2012/05/23 Javascript
js的onload事件及初始化按钮事件示例代码
2013/09/25 Javascript
JavaScript快速切换繁体中文和简体中文的方法及网站支持简繁体切换的绝招
2016/03/07 Javascript
js添加绑定事件的方法
2016/05/15 Javascript
javascript实现根据汉字获取简拼
2016/09/25 Javascript
angularjs利用directive实现移动端自定义软键盘的示例
2017/09/20 Javascript
vue项目实战总结篇
2018/02/11 Javascript
vue的全局提示框组件实例代码
2018/02/26 Javascript
Vue2.0实现调用摄像头进行拍照功能 exif.js实现图片上传功能
2018/04/28 Javascript
vue.js仿hover效果的实现方法示例
2019/01/28 Javascript
Webpack中loader打包各种文件的方法实例
2019/09/03 Javascript
vue中@change兼容问题详解
2019/10/25 Javascript
uni-app 支持多端第三方地图定位的方法
2020/01/03 Javascript
javascript设计模式 ? 模板方法模式原理与用法实例分析
2020/04/23 Javascript
微信小程序淘宝首页双排图片布局排版代码(推荐)
2020/10/29 Javascript
从零学Python之hello world
2014/05/21 Python
Django的信号机制详解
2017/05/05 Python
VScode编写第一个Python程序HelloWorld步骤
2018/04/06 Python
在tensorflow中设置使用某一块GPU、多GPU、CPU的操作
2020/02/07 Python
python itsdangerous模块的具体使用方法
2020/02/17 Python
CSS3与动画有关的属性transition、animation、transform对比(史上最全版)
2017/08/18 HTML / CSS
找到您丢失的钥匙、钱包和手机:Tile
2017/05/19 全球购物
优秀毕业生的求职信
2014/07/21 职场文书
五月的鲜花活动方案
2014/08/21 职场文书
小学生差生评语
2014/12/29 职场文书
5行Python代码实现一键批量扣图
2021/06/29 Python
解决Swagger2返回map复杂结构不能解析的问题
2021/07/02 Java/Android