深入理解Python函数中的self

深入理解Python函数中的self

其实应该换个题目: 为什么类定义方法有self但实例调用方法没有self?

理解一等公民: 函数

在Python中一切皆对象, 类是对象, type是对象, 当然函数(方法)也是对象. 对象都有地址, 用id(对象)获得, 判断变量所指对象是不是同一个, 用表达式id(变量1)==id(变量2)判断.

看网上博客经常说, “在实例调用方法时, Python解释器内部把self替换成实例本身了”. 我们试着用函数是一等公民的思想来理解这句话, “实例的方法和类中定义的方法是用一个, 当方法执行是, 解释器会判断是不是实例, 如果是实例那么把方法中的第一个参数默认换成它本身”. 下面来验证这样对不对

class Foo:
    def echo(x):
        print(x)

f1 = Foo()
f2 = Foo()

print(id(f1.echo)==id(Foo.echo))
# False

print(id(f1.echo)==id(f2.echo))
# False

从结果发现, 类定义的函数和实例的函数根本不是同一个, 甚至每个实例的函数都不是同一个. 不是同一个代表了什么? 来看一下特殊的实例方法staticmethod

class Foo:
    @staticmethod  # 静态方法
    def sm(): print('sm')
f1 = Foo()
f2 = Foo()
print(id(Foo.sm) == id(f1.sm) == id(f2.sm)) # True

得到了截然相反的结果, 静态方法在类中和实例中都是同一个. 因为这它的执行不需要实例的参与. 所以, 为了实现实例中普通方法默认第一个参数是实例本身这个语法糖(完全可以看作是一种语法糖), 在类生成实例的同时, 根据类函数生成对应的实例函数(因为函数对象改变了, 所以一定是生成了一个新的函数对象), 并组装在实例身上. 制造一个新函数, 将实例本身保存起来, 将函数作为一等公民的Python实现起来并不难, 一个闭包就可以, 下面来模拟一下

模拟实例组装

from functools import wraps
# 定义一个空壳类
class Foo: pass
# 假设echo是Foo的函数
def echo(self,a):
    print(self.__class__.__name__, a)

# 获取实例函数, 用闭包把实例本身存起来
def get_instance_func(new_instance, class_func):
    @wraps(class_func):
    def instance_func(*arg,**kwargs)
        class_func(new_instance, *arg, **kwargs)
    return instance_func

# 生成实例
def new_foo():
    # 通过object生成新实例
    new_foo_instance = object.__new__(Foo)
    instance_echo = get_instance_func(new_foo_instance, echo)
    # 将新方法组装上
    setattr(new_foo_instance, instance_echo.__name__,instance_echo)
    # 执行初始化方法
    Foo.__init__(new_foo_instance)
    return new_foo_instance

# 测试
foo = new_foo()
foo.echo(10)
# Foo 10

后记

本文目的是探讨和理解Python实例方法省略self实现方法, 当然Python内部肯定不是这样实现的, 但实例方法是新方法, 那么实现思路大同小异, 都是将实例本身储存在实例方法中. 官方文档中写着实例方法用__self__属性保存实例本身, 同时还用__func__保存类中定义的方法. 只不过在上述模拟过程, 用闭包保存, 道理类似.

当然, 以上纯粹是根据代码运行结果的猜测, 如有你有不同的看法, 欢迎交流!

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值