今天的任务比较繁重,因为我们要一起来学习Python中比较重要比较牛逼比较难的装饰器。
我将会和大家一起通过代码的形式来迷你银行存款取款的功能,然后通过引入装饰器来一步一步优化代码。
废话不多说梦开始撸代码。我们去ATM存款取款的过程用代码来表示大概如下(不大准确,仅仅是为了掩饰需要)
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s1.51cto.com/oss/201807/23/520e10d1ae7630689511b2f27483d982.jpeg)
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s2.51cto.com/oss/201807/23/03ae3e1254f233e1b78dbd51a011c0c6.jpeg)
上面是按下按钮1就存款,否则则取款。
不对,存取款要输入密码啊!!!所以,我们要加密码验证代码。
可以看到,虽然实现了密码验证功能,但是代码冗余度比较高,而且现在只模拟了取款和存款功能,然而还有查询功能,转账功能等等,那么冗余度就更高了,而且相对于取款和存款函数来说,复用性没有那么高,所以我们要进一步优化代码,把验证函数写到取款和存款函数内部。
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s5.51cto.com/oss/201807/23/d32dce01fb1cc2c732bcafc0adb6af61.jpeg)
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s5.51cto.com/oss/201807/23/0d29f371e1f8ed32e0d0aef59cc0ac8d.jpeg)
有没有什么方法,可以在不改变原函数以及原函数的调用的情况下扩展原函数的功能呢?当然是有的,这就是python中著名的装饰器。还是使用上面的代码来演示
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s1.51cto.com/oss/201807/23/8d12fcc9b9ce35a050a40326c2523292.jpeg)
从现在开始,上面红框的代码不再进行更改,打死也不改,我们来实现有密码验证的存取款功能。
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s2.51cto.com/oss/201807/23/1840f6475bfcc32cd95d5d0268087f93.jpeg)
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s3.51cto.com/oss/201807/23/0823965c4815914e2adfbfb0ffb42875.jpeg)
装饰器装饰有参函数
看下面这段代码
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s4.51cto.com/oss/201807/23/bf1fe3917b19eab065b9f9b6a9344499.jpeg)
大家看看这段代码有没有什么问题?
运行结果
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s3.51cto.com/oss/201807/23/356a5717e00649ec20d4ea265e966360.jpeg)
看报错原因就知道,inner()函数多了一个参数。大家一定要注意一点,因为装饰器函数的返回值是inner,也就是说现在myprint是等同于inner的。
我们来改一下代码
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s3.51cto.com/oss/201807/23/80c27baa27200d6099dfb56d7a80b7e1.jpeg)
为了使装饰器能够装饰更多函数,我们进一步优化代码。
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s5.51cto.com/oss/201807/23/7669d7497cd71da7690d4f811468fdda.jpeg)
装饰器装饰有返回值函数
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s2.51cto.com/oss/201807/23/f01dce740b0c44f6abcc3a3c867839c5.jpeg)
如图可以看到,无论被装饰的函数有无返回值,其结果都无返回值,原因其实很简单,因为inner()函数根本就没有返回值。为了实现有返回值的函数被装饰之后仍然有返回值,需要inner函数与被装饰函数的返回值保持一致。
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s5.51cto.com/oss/201807/23/b634c4cd7f756a4ad44d4a0248d9fb26.jpeg)
可以看到,有返回值的函数被装饰之后依然有返回值,没有返回值的函数被装饰之后则没有返回值,符合我们想要的结果。
双重语法糖
为了更加直观地演示双重语法糖,我就不再使用上面的例子了。
我先来定义一个函数
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s3.51cto.com/oss/201807/23/04907a6a96e0ae745f246f736c1be2aa.jpeg)
现在我有一个需求,我想在'python小白联盟'的上面打印一行'='和一行'*',也就是要实现这种效果。
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s3.51cto.com/oss/201807/23/4f32e1df0b5b7bf5a66d15dd9fbe2d84.jpeg)
那么应该怎么写呢?大家可能都会想到语法糖的叠加,那叠加的方式是以下哪一种呢?
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s5.51cto.com/oss/201807/23/7e4cdabbd9d5cc39d21447f889a7227e.jpeg)
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s3.51cto.com/oss/201807/23/ca340e41da2e195c4f0adda146c09fd8.jpeg)
我们来运行一下就知道了
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s1.51cto.com/oss/201807/23/29ab245696f92b6d9e9b8043c55bdc6f.jpeg)
可以看到,叠加的方式应该是***种,也就是说最外层的语法糖先执行。
带参数的语法糖
为了演示带参数的语法糖,我再把上面的一张图搬下来
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s1.51cto.com/oss/201807/23/705bb90a84e236666dcc85cdf4bf073e.jpeg)
显然,红框部分的代码冗余度比较高,我们可以使用带参的语法糖来进行简化
![这是我见过最全面的Python装饰器详解!没有学不会这种说法!](https://s3.51cto.com/oss/201807/23/5de84d2acf553a8307445f7158211269.jpeg)
装饰器就讲到这里,祝大家有个愉快的一天!