美文网首页互联网科技程序员Python学习分享
你真的理解Python中的赋值、传参吗?

你真的理解Python中的赋值、传参吗?

作者: 烟雨丿丶蓝 | 来源:发表于2019-02-23 14:19 被阅读9次

    在学习编程的过程我们都会遇到很多定义,之前在遇到这些定义的时候,我有一种强迫症。就是不搞清楚每一个字的含义,不善罢甘休。但是每次都会尽兴而来,失望而归。多次之后我学乖了,就是不纠结实际每一个字的含义,用自己能懂的方式理解他们,比如今天要说的引用传递和值传递

    官方的定义是这样的

    值传递:

    值传递是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数 —— via:百度百科

    引用传递:

    引用传递是指在调用函数时将实际参数的地址传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数—— via:百度百科

    这种每一个字我都认识,但连起来我就不知道啥意思的感觉,已经伴随了我这个9年+4年教育生涯,至此对它深恶痛绝。

    身为老司机,还是得分享些干货精品学习资料的,推荐下小编创建的Python学习交流群556370268,这里都是一群爱好Python的小伙伴,每天还会直播和大家交流分享经验哦。

    赋值

    回到正题,我们暂时抛开两个概念,我们先来说下 Python 中的赋值,以我的理解,其实就是下定义的步骤,如果大家看过《武林外传》中第二十九集《吕圣人智斗姬无命 佟掌柜火拼展红绫》,就更容易理解以下的概念了

    赋值这个操作,其实可以理解成给物体贴标签,或者可以理解为给物体命名,既然是名称,就像《武林外传》中的一样,你可以叫“姬无命”,我也可以叫“姬无命”。重要的是这个物体,而不是标签。其中,这个标签(或者名称),我们在计算机中把它叫做变量,物体就是实际的值

    我们知道,在计算机中,值会占用一定的空间去存储,而计算机为了方便找到它,则会给它一个地址,方便我们找到它

    我们写段代码理解下

    >>> zxj = "张小鸡"
    >>> jwm = zxj
    >>> print(id(zxj))
    4334207504
    >>> print(id(jwm))
    4334207504
    

    看图理解很简单,这里的物体是字符“张小鸡”,我们把它贴上标签“zxj”,第二个地方赋值就相当于再贴一个标签“jwm”。就是上面说的,你可以叫“张小鸡”,我也可以叫“张小鸡”,看后面,他们实际上的内存地址都是一样的

    参数传递

    在理解了上面的过程,我们再来看看 Python 中调用函数传递参数的过程。先说结论,Python 中参数的传递就是赋值的过程。我们来看下这段代码

    a = "张小鸡"
    def foo(b):
     print ">>> before id(b):", id(b)
     b = "姬无命"
     print ">>> after id(b):", id(b)
    print ">>> before id(a):", id(a)
    foo(a)
    print ">>> after id(a):", id(a)
    

    这一段的输出如下

    >>> before id(a): 4301385520
    >>> before id(b): 4301385520
    >>> after id(b): 4301385040
    >>> after id(a): 4301385520
    

    我们在赋值时,其实就相当于把函数的形参b这个标签又贴在了“张小鸡”物体上。后面我们再执行b=“姬无命”时,就相当于把b这个标签从“张小鸡”这个物体上撕下来,放到“姬无命”这个物体上

    可变和不可变对象

    Python 内部对对象进行了区分,即为可变对象和不可变对象,类型如下

    int、str、float、tuple等为不可变对象

    list、dict、set等为可变对象

    不可变对象我们上面已经说过他的赋值的特点,我们这里主要看可变对象。对于可变对象,我们可以简单的理解为做了个包装盒。我们在赋值的时候,这个标签是贴在了这个包装盒子上。计算机会记录这个盒子的地址,里面每一个物体的地址,计算机也仍然会记录

    a = ["张小鸡"]
    def foo(b):
     print(">>> before id(b):", id(b))
     b[0] = "姬无命"
     b.append("Tom")
     print(">>> after id(b):", id(b))
    print(">>> before value(a):", a)
    print(">>> before id(a):", id(a))
    foo(a)
    print(">>> after value(a):", a)
    print(">>> after id(a):", id(a))
    

    输出如下

    >>> before value(a): ['张小鸡']
    >>> before id(a): 4518129480
    >>> before id(b): 4518129480
    >>> after id(b): 4518129480
    >>> after value(a): ['姬无命', 'Tom']
    >>> after id(a): 4518129480
    

    我们将盒子里面的“张小鸡”替换为“姬无命”,又再盒子里面添加了“Tom”,自始至终,因为我们没有动过盒子本身,所以他的地址不会发生变化

    结合上图来看下,我们修改一下代码,再深入看下盒子和盒子里面物体的地址的变化

    a = ["张小鸡"]
    def foo(b):
     b[0] = "姬无命"
     b.append("Tom")
    print(">>> before id(a): ", id(a))
    print(">>> before id(a[:]): ", [id(_) for _ in a])
    foo(a)
    print(">>> after id(a): ", id(a))
    print(">>> after id(a[:]): ", [id(_) for _ in a])
    

    输出如下

    >>> before id(a): 4471127880
    >>> before id(a[:]): [4472230896]
    >>> after id(a): 4471127880
    >>> after id(a[:]): [4472230992, 4472127648]
    

    看,我们的盒子(即id(a))自始至终都没有变化,而内部因为更换过物体,所以里面的地址都不一样了

    拓展思考

    def foo(a, b=[]):
     b.append(a)
     return b
    print(foo(1))
    print(foo(1))
    print(foo(1))
    

    上面这段代码的输出结果是

    >>> [1]
    >>> [1, 1]
    >>> [1, 1, 1]
    

    相关文章

      网友评论

        本文标题:你真的理解Python中的赋值、传参吗?

        本文链接:https://www.haomeiwen.com/subject/nocpyqtx.html