关于Python装饰器内层函数为什么要return目标函数的一些个人见解

https://blog.csdn.net/try_test_python/article/details/80802199

前几天在学装饰器的时候,关于装饰器内层函数调用目标函数时是否return目标函数的调用产生了一点迷惑,事实是当被装饰的目标函数有返回值的时候,装饰器内层函数也必须返回该目标函数的调用。

我们都知道不带括号的函数名指向是函数代码所在的内存地址,加上括号之后就变成了一个执行命令,那么这个‘func( )’到底有什么意义呢?

上面这张图可以大概看出点东西,单独的函数名是 function类的一个实例对象,而函数名() 则没有任何类型,是个无类型。

然后看这张图,一个无返回值的函数调用指向的内存地址和 ‘None’的内存地址是同一个,说明什么呢,说明函数名()很可能也是个变量。那这个变量里面装的什么东西呢,从上面两张图可以看得出,它里面什么也没有。

同样的,print()作为解释器自带的函数同样也指向了空。

这三张图想说明什么呢,可能有点乱,但是结论也很清楚,func()仅仅是告诉系统要执行其函数名指向的那段代码,剩下的没有别的意义,任何函数名加上括号,那就是执行的信号,告诉系统,我要执行。(其实任何可调用类型都是这样)

然后开始进入关于return的一些研讨。

在装饰器里,闭包的内层函数对目标函数的处理是否return,结果很不一样。

先看上面这块代码,当函数往外有返回值的时候,func()的指向就不一样了,它此时指向的是返回值所在的内存地址,我们上面已经得到了一些结论,那就是函数没有返回值的时候,func()是空的,有了返回值,那么它就成了指向返回值的这么个变量。所以,func()除了作为执行代码的信号之外,还用来保管函数的返回值。

我们可以理解为,func()是一个容纳函数运行附加值的容器,return的作用就是将这个附加值往外传递,传递给谁呢,传递给func()这个变量。下面做个小实验:

如上图所示,return将函数的返回值赋值给所在的“函数名()”,而函数名()这个变量也因为所装载的数据而改变了类型:

如图,func1()的类型变成了整型。

也就是说,如果一个函数有了结果而不往外传递这个结果的话,那么这个结果就算是被丢弃了,作为这个函数的‘首席’接受者,函数名()将仅仅是函数运行的一道代表命令的字符,除此之外,什么也不是。

现在讨论装饰器中return的必要性。

上面三张图分别是未经过装饰器扩展时目标函数的执行结果,然后是加了装饰器,但是没有返回函数结果和返回了函数结果的三种情况。

我们知道,装饰器的作用就是给目标函数扩展功能,基本原则是不能影响原函数的功能,所以不加return的话,原函数的基本功能都受到影响了,必须对原函数的返回值进行再次return。

现在我们来说结论,函数名是指代的函数代码所在的空间,函数名加括号,指代的是函数往外传递的数据,而return就是往外传递数据的工具,如果函数名()没有接收到return出来的数据,那么“函数名()”除了执行命令外,什么也不是,函数往外传递的数据就丢失了。

装饰器的核心在于闭包的内层函数,外层函数将目标函数拿来加工,用内层函数包裹目标函数的代码,然后将这一整块代码返回给目标函数的函数名,现在目标函数的函数名(代号func)指向的是加工过的函数代码,原来的代码成了外来户,所以原来代码传递的数据如果没有中间人进行传递的话,func将不能接收那个数据,那个数据就丢失了。func()原来可以容放原函数返回值,现在就什么都没有了。所以,return是很有必要的。
---------------------  
作者:haiton大人  
来源:CSDN  
原文:https://blog.csdn.net/try_test_python/article/details/80802199?utm_source=copy  
版权声明:本文为博主原创文章,转载请附上博文链接!

原文地址:https://www.cnblogs.com/fengff/p/9780334.html

时间: 2024-10-05 05:00:39

关于Python装饰器内层函数为什么要return目标函数的一些个人见解的相关文章

5.初识python装饰器 高阶函数+闭包+函数嵌套=装饰器

一.什么是装饰器? 实际上装饰器就是个函数,这个函数可以为其他函数提供附加的功能. 装饰器在给其他函数添加功能时,不会修改原函数的源代码,不会修改原函数的调用方式. 高阶函数+函数嵌套+闭包 = 装饰器 1.1什么是高阶函数? 1.1.1函数接收的参数,包涵一个函数名. 1.1.2 函数的返回值是一个函数名. 其实这两个条件都很好满足,下面就是一个高阶函数的例子. def test1(): print "hamasaki ayumi" def test2(func): return t

python 装饰器 二 (装饰带参数的函数)

#!/usr/bin/env python #_*_coding=utf-8_*_ #Author: xieyixue #mail: [email protected] #Created Time: 2015年07月01日 星期三 00时28分03秒 def deco(func): '''定义装饰器接受参数,参数为被装饰的函数 ''' def _deco(*arg, **kwarg): '''定义方法执行被装饰的函数,接受2个参数''' #执行前-->执行 print("before&qu

python 装饰器 一 (简单不带参数的函数)

#coding=utf-8 #python 装饰器 '''     装饰器可以用def的形式来定义.装饰器接收一个可调用对象作为输入参数, 并返回一个新的可调用对象. 装饰器新建了一个可调用对象,也就是return 返回的函数funx, 在新增的函数中,可以添加我们需要的功能,并通过调用原有函数来实现原有函数的功能 ''' def deco(func):     '''         无参数装饰器,func为被装饰的函数     '''     def _deco():         pri

python装饰器中@wraps作用--修复被装饰后的函数名等属性的改变

Python装饰器(decorator)在实现的时候,被装饰后的函数其实已经是另外一个函数了(函数名等函数属性会发生改变),为了不影响,Python的functools包中提供了一个叫wraps的decorator来消除这样的副作用.写一个decorator的时候,最好在实现之前加上functools的wrap,它能保留原有函数的名称和docstring.废话不多说,上俩栗子就能搞明白! 实例一: 不加wraps # -*- coding=utf-8 -*- from functools imp

由浅入深,走进Python装饰器-----第二篇:进阶--函数装饰函数

上一篇:由浅入深,走进Python装饰器-----第一篇:基础 装饰器的使用种类: # 第一种 @函数 被装饰函数 # 第二种 @函数 被装饰类 # 第三种 @类 被装饰类 # 第四种 @函数 被装饰函数 本篇介绍第一种 @函数 被装饰函数 1.1 对带参数的原函数进行修饰 # 默认将old函数的参数传给outer里面的第一层函数 def outer(f): def inner(var): print("1 我是outer函数,接收外部传进来的old :",f) print("

由浅入深,走进Python装饰器-----第二篇:进阶--类装饰函数

**类装饰器** @类 函数 2.1 用类装饰器来扩展原函数 # 用类装饰器来扩展原函数, 通过对象函数化触发__call__方法,进行返回 class KuoZhan(): def __call__(self,f): return self.newfunc(f) def newfunc(self,f): def in_newfunc(): print("1") f() print("2") return in_newfunc @KuoZhan() #1. KuoZ

Python装饰器由浅入深

装饰器的功能在很多语言中都有,名字也不尽相同,其实它体现的是一种设计模式,强调的是开放封闭原则,更多的用于后期功能升级而不是编写新的代码.装饰器不光能装饰函数,也能装饰其他的对象,比如类,但通常,我们以装饰函数为例子介绍其用法.要理解在Python中装饰器的原理,需要一步一步来.本文尽量描述得浅显易懂,从最基础的内容讲起. (注:以下使用Python3.5.1环境) 一.Python的函数相关基础 第一,必须强调的是python是从上往下顺序执行的,而且碰到函数的定义代码块是不会立即执行它的,只

一个关于python装饰器参数的问题

看到廖雪峰python教程上,python装饰器一章 https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/0014318435599930270c0381a3b44db991cd6d858064ac0000 在最后的作业题上 再思考一下能否写出一个@log的decorator,使它既支持: @log def f(): pass 又支持: @log('execute') def f(

遥想大肠包小肠----python装饰器乱弹

说起装饰器就tm蛋疼,在老男孩学习python装饰器,结果第二天默写,全错了,一道题抄十遍,共计二十遍. 要是装饰器是一人,我非要约他在必图拳馆来一场...... 下面容我展示一下默写二十遍的成果 语法形式 def  mydec(wenwa): def inner(*args,**kwagrs): ret = wenwa(*args,**kwargs) return ret  #请务必别忘记这还有个该死的return,如果被执行的函数没有返回值return,则ret为None return in