python笔记

在学习python函数的参数传递部分时,发现了有趣的现象。

对于python来说,变量是没有类型的,只有对象才有类型,而变量只是对象的一个“标签”,那么,当传递不可变对象的时候(数字,字符串和元组),在函数中对参数作出的修改是不能被传递到函数外部的,而传递可变对象(列表,字典)时,对可变对象的修改也会影响到外部的对象。

学过c++的都知道,假如一个参数传递的是对象的地址,那么对象在函数中的改变就会被传递到外面来。在函数的参数复制阶段,参数被赋为外部传递的对象的地址,在函数中对这个地址中的值进行修改后,外部再引用这个地址时,就会发现地址内的东西已经被改变了。

python作为一个用c实现的语言,如何实现传不可变参数和可变参数?因为所有的变量都是指针,指向对应的对象,所以我猜测,当你对这个指针进行重新赋值的时候,它会指向一个新的地址,但是假如你对这个指针所指对象的一部分进行修改的话,就只有这一部分的内容或者是这一部分指向的地址(因为这一部分也有可能是个指针)发生改变。这样的话改变就会传递到函数的外面了,因为外部的变量指向的地址也是同一块。

如下:


def func(a):
    a = 6
    print a

a = 5
b = 10
print a,b
func(a)
print a,b

这一段代码的运行结果是:

5 10
6
5 10

说明a在函数内的改变并没有被传递到外面来。

再好比说字符串:

def func(a):
    a += "123456"
    print a

a = "ababab"
b = "cdcdcd"
print a,b
func(a)
print a,b

结果为:

ababab cdcdcd
ababab123456
ababab cdcdcd

改变也没有传递到外部。

数字,字符串和元组都是不能对其内部进行改变的,假如要改变,只能新建一个新的对象并让原有的变量指向这个对象(比如说在字符串的后面加上“123456”,其实是新建了一个新的字符串,而不是在原来的字符串的基础上添加),所以说,函数内部的a和函数外部的a已经不指向同一个地址了,改变就自然不会传递到外部。再看看字典:

def func(dic):
    dic["abc"] = "biubiubiu"
    print str(dic)
    return dic

dic = {"abc":"lalala",123:7302}
print dic
print func(dic)
print dic

结果是:

{123: 7302, 'abc': 'lalala'}
{123: 7302, 'abc': 'biubiubiu'}
{123: 7302, 'abc': 'biubiubiu'}
{123: 7302, 'abc': 'biubiubiu'}

改变的确传递到了外部,可是再看下面一段代码:

def func(dic):
    dic = {"nibuzhidao":"wodiquebuzhidao"}
    print str(dic)
    return dic

dic = {"abc":"lalala",123:7302}
print dic
print func(dic)
print dic

结果居然是:

{123: 7302, 'abc': 'lalala'}
{'nibuzhidao': 'wodiquebuzhidao'}
{'nibuzhidao': 'wodiquebuzhidao'}
{123: 7302, 'abc': 'lalala'}

改变没有传递到外部,怎么回事,不是说字典是可更改对象吗?
在这里,我没有更改字典内部的键值对,而是直接将一个新的字典赋给了dic,这个时候,程序实际上是对dic这个“标签”进行改写。就像是字符串或是数字的情况一样,dic选择改变自己指向的地址,这时里面的dic和外面的dic实际上已经不是指向同一个地址了,所以改变才不能传递到外部。

因为列表和字典的一些性质,我大概猜测,列表和字典的实现都是通过指针相连的松散结构,实际在内存中的位置不一定连续,所以当修改和添加内容的时候可以直接改变指针的指向,不用新建一个全新的对象。而字符串,数字,元组在内存中是相邻的,所以不可以对其作出局部修改。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值