python魔术方法

魔术方法:
在python中,所有以“__”双下划线包起来的方法,都统称为“magic method”,例如最常用的就是类的初始化方法init
既然叫魔法函数,一定是有他存在的意义,也就是说他可以怎么样....,可以创造奇迹,去完成一些正常的语法无法实现或者实现起来比较复杂的事情

下面我们就来具体的来看一下Python中常用的几个魔法函数

__init__: 构造函数(构造器)
当一个实例被创建的时候调用的初始化方法

__call__:
允许一个类的实例像函数一样被调用:x(a, b)调用 x.call(a, b)

__del__: 析构函数(析构器)
当一个实例被销毁的时候调用的方法

下面通过一个示例来了解下这几个方法的应用:

from time import sleep


class MagicTest:

    # 根据其自身特性,构造方法在自动化测试中可以用来做测试数据、测试资源的加载等初始化相关的场景
    def __init__(self):
        print("初始化构造函数")

    def invoke_fun(self):
        print("调用当前类方法")

    # 根据其特性,析构方法在自动化测试中可以用来处理如驱动对象关闭、全局变量静态写入等数据清理场景的场景
    def __del__(self):
        print("销毁当前对象")


if __name__ == '__main__':

    mt = MagicTest()
    mt.invoke_fun()
    sleep(1)
    mt.invoke_fun()
    sleep(3)
    mt.invoke_fun()

call 方法的功能类似于在类中重载()运算符,使得类实例对象可以像调用普通函数那样,以“对象名()”的形式使用

class CallTest:
    # 定义__call__方法

    def __call__(self, name, add):
        print("调用__call__()方法", name, add)


ct = CallTest()
ct("程序猿教育", "更高、更快、更强")
ct.__call__("程序猿教育", "更高、更快、更强")

执行结果:

调用__call__()方法 程序猿教育 更高、更快、更强

可以看到,通过在CallTest类中实现call()方法,使ct实例对象变为了可调用对象
python中,凡是可以将()直接应用到自身并执行,都称为可调用对象,可调用对象包括:

  • 自定义的函数
  • python内置函数
  • 以及类的实例对象
    对于可调用对象,实际上“名称()”可以理解为是“名称.call()”的简写,仍然以上面程序中定义的ct实例对象,
    其最后一行代码还可以改为如下形式:
ct.__call__("程序猿教育", "更高、更快、更强")

运行程序会发现结果跟改之前完全相同

自定义函数的例子

def call_test():
    print("我是一个自定义的可调用对象")


call_test()
call_test.__call__()

那么我们会有疑问,使用call方法的意义在哪里?
一般情况下我们类方法的调用是通过先创建类对象,再通过obj.func()的方式来调用,但是如果类中只有一个方法或者一个方法的使用频率非常高
那么这个就可以为这个方法命名为call来简化调用

上面我们讲到存在call方法的类对象被称为可调用对象,它的作用可以总结为一下几点:
1、简化了对象下方法的调用(当某方法调用频率很高的时候)
2、模糊了对象和函数调用时的区别(提高了代码的兼容性)
3、弥补反射函数hasattr()的短板(hasattr有一个缺陷,它无法判断类该指定的名称,到底是类属性还是类方法)

简化对象下方法调用示例

class NoneCall:
    """未实现call方法的类"""

    def check_position_current_number(self):
        print("查询用户当前持仓数量")

    def check_order_current_status(self):
        print("查询用户当前订单状态")


nc = NoneCall()
nc.check_position_current_number()
nc.check_order_current_status()
class HaveCall:
    """未实现call方法的类"""
    def check_position_current_number(self):
        print("查询用户当前持仓数量")

    def __call__(self):
        print("查询用户当前订单状态")


hc = HaveCall()
hc()
hc.check_position_current_number()

提高代码兼容性示例

class ServiceDBNoneCall:
    def get_order_info(self):
        print("获取订单号")

    def get_position_info(self):
        print("获取持仓数量")


sdbnc = ServiceDBNoneCall()
sdbnc.get_order_info()
class ServiceDBHaveCall:
    # 在当前类中,原本有get_order_info、get_position_info两个函数,但是由于get_order_info的使用频率高,因此我们使用call来简化该函数的调用,使其成为一个可调用对象
    def __call__(self):
        print("获取订单号")

    def get_position_info(self):
        print("获取持仓数量")


sdbhc = ServiceDBHaveCall()
sdbhc()

如果此时我有一个跟ServiceDBNoneCall类中的get_order_info方法功能类似的函数get_order_id,现在我们需要将ServiceDBNoneCall类的实例(对象)和这个新的函数作为参数分别传到另一个函数中去执行,我们看下此时应该怎么实现:

先创建一个与类中的get_order_info方法功能类似的函数

def get_order_id():
    print("生成订单号")

假如我现在需要将获取到的订单号从数字类型转换为字符串,那我是不是需要这样写

# def num_to_str(order_id):
#     str(order_id)
def num_to_str_by_no_call(func):
    """传入未实现call的类的对象"""
    str(func.get_order_info())
    print("订单号已经转换为字符串")


def num_to_str_by_func(func):
    """传入可调自定义函数对象"""
    str(func())
    print("订单号已经转换为字符串")


def num_to_str_by_call(func):
    """传入已经实现call的类的对象"""
    str(func())
    print("订单号已经转换为字符串")

print("测试实际使用效果")
num_to_str_by_no_call(sdbnc)
num_to_str_by_func(get_order_id)
num_to_str_by_call(sdbhc)

我们可以看到上面三个函数中,num_to_str_by_func、num_to_str_by_call在结构定义上完全相同,那么此时,我们可以怎么样,是不是就可以将这两个函数定义一次就够了

def num_to_str(func):
    """简化代码,定义通用方法"""
    str(func())
    print("订单号已经转换为字符串")


print("++++++++++++++++++++++++++++++++++++++++++++++")
print("测试实际使用效果")
num_to_str(get_order_id)
num_to_str(sdbhc)

弥补反射内置方法的不足
反射 4个内置函数分别为:getattr、hasattr、setattr、delattr 获取成员、检查成员、设置成员、删除成员
我们知道hasattr有一个缺陷,它无法判断类该指定的名称,到底是类属性还是类方法,那么要解决这个问题,我们可以借助今天学到的可调用对象的概念,要知道,类实例对象包含的方法,其实也属于可调用对象,但类属性缺不是
示例如下:

class ProgramAcademy:
    def __init__(self):
        self.name = "程序猿学院"
        self.age = 10

    def training_test_engineers(self):
        print("培养测试工程师,大牛辈出")


pa = ProgramAcademy()
if hasattr(pa, 'name'):
    print(hasattr(pa.name, "__call__"))

if hasattr(pa, 'training_test_engineers'):
    print(hasattr(pa.training_test_engineers, "__call__"))

预期执行结果:

# False
# True

可以看到,由于name是类属性,它没有以call为名的call()方法,而training_test_engineers是类方法,它是可调用对象,因此它有call()方法

魔法函数课程总结:
本次课程只介绍了python中几个常用的魔法函数,python中还有很多的魔法函数,每一个魔法函数都有其自身实现的特定的功能,在实际的自动化代码工程中会用到一些特定的魔术方法,我们在这里主要是让大家认识魔法函数这个概念、常用魔法函数的使用、以及在工作中实际遇到魔法函数不至于一脸懵,实际遇到了就去了解下具体函数的功能,能够学以致用

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容