Python笔记3|参数传递

Python笔记3|参数传递,第1张

Python笔记3|参数传递 Python参数传递

目录

Python参数传递

1.什么是参数的传递2.传递不可变对象的引用

2.1 传递不可变对象的引用的底层逻辑

2.1.1 传递不可变对象的底层步骤: 3.传递可变对象的引用

3.1 传递可变对象的引用的底层逻辑

3.1.1 传递可变对象的底层步骤:

1.什么是参数的传递

2.传递不可变对象的引用


不可变对象:int、float、字符串、元组、布尔值。

2.1 传递不可变对象的引用的底层逻辑


关键:把 a color{red}{a} a 的 i d color{orange}{id} id 传给 n color{purple}{n} n , n color{purple}{n} n 也指向了 a color{red}{a} a 所指向的 int 对象,但由于 int 对象不可改变,因此执行n=n+2时,会创建一个新对象存储 “n+2” 的计算结果,再让 n color{purple}{n} n 指向这个新的计算结果对象。但原本 a color{red}{a} a 和 n color{purple}{n} n 共同指向的 int对象未改变,现在 n color{purple}{n} n 不指向原本的 int对象了,又由 a color{red}{a} a 单独指向 int对象。

2.1.1 传递不可变对象的底层步骤:

①栈中创建全局变量 a color{red}{a} a,堆中创建了一个 v a l u e color{orange}{value} value 为 100 color{green}{100} 100 的 int对象。int对象 i d color{orange}{id} id 给了 全局变量 a color{red}{a} a, a color{red}{a} a 指向该列表对象。

②def f1(n):函数定义。

栈中产生全局变量 f 1 color{red}{f1} f1 ,堆中产生函数对象,并将函数对象的 i d color{orange}{id} id 给 f 1 color{red}{f1} f1。
此时 f 1 color{red}{f1} f1 指向函数对象。参数 n color{purple}{n} n 尚未用到。

③f1(a)调用函数,传递参数 a color{red}{a} a —> n color{purple}{n} n

栈中产生一个栈帧,在栈帧中产生局部变量 n color{purple}{n} n,由于实参 a color{red}{a} a 传给了形参 n color{purple}{n} n,此时局部变量 n color{purple}{n} n 也指向了 a color{red}{a} a 所指向的 int对象,此时 a color{red}{a} a 和 n color{purple}{n} n 共同指向 int对象。

④n=n+200计算结果并创建新对象保存结果,将 n color{purple}{n} n 指向新对象

由于 int对象是不可变对象,因此虽然 n color{purple}{n} n 此时直接指向了int对象也不能在int对象上直接修改。于是现在堆中新建一个 int对象,将n+200的计算结果保存到对象中作为 v a l u e color{orange}{value} value,再让 n color{purple}{n} n 指向这个新对象。随后,又只有 a color{red}{a} a 指向原本的 int对象, n color{purple}{n} n 不再指向原本的 int对象而是指向了新对象,所以 n color{purple}{n} n 在函数中的 *** 作就与原本的 int对象没有关系了。

简单测试:

浅拷贝


①test01()参数传递 a color{red}{a} a --> m color{purple}{m} m
a color{red}{a} a 和 m color{purple}{m} m共同指向元组对象。

②m[2][0] = 888修改元组对象的第三个子对象(即列表对象)的第一值为 i n t ( 888 ) color{green}{int(888)} int(888) 。

最终, a color{red}{a} a 和 m color{purple}{m} m 共同指向的元组对象的第二个子对象(列表对象)的第一个元素被修改为 i n t ( 888 ) color{green}{int(888)} int(888) 。

3.传递可变对象的引用


可变对象:列表、字典、自定义的其他可变对象。

3.1 传递可变对象的引用的底层逻辑


关键:由于列表是可变对象,那么把 b color{red}{b} b 列表 i d color{orange}{id} id 给了栈帧里的局部变量 m color{blue}{m} m, m color{blue}{m} m 也指向了列表对象。
于是 m color{blue}{m} m 可以对列表进行直接 *** 作。 m color{blue}{m} m 消失后, *** 作结果还在。

3.1.1 传递可变对象的底层步骤:

①栈中创建全局变量 b color{red}{b} b,堆中创建了一个列表对象,包含 10 color{green}{10} 10、 20 color{green}{20} 20两个值。列表对象 i d color{orange}{id} id 给了 全局变量 b color{red}{b} b, b color{red}{b} b 指向该列表对象。

②def f2(m):函数定义。

栈中产生全局变量 f 2 color{red}{f2} f2 ,堆中产生函数对象,并将函数对象的 i d color{orange}{id} id 给 f 2 color{red}{f2} f2。
此时 f 2 color{red}{f2} f2 指向函数对象。参数 m color{purple}{m} m 尚未用到。

③f2(b)调用函数,传递参数 b color{red}{b} b —> m color{purple}{m} m

栈中产生一个栈帧,在栈帧中产生局部变量 m color{purple}{m} m,由于实参 b color{red}{b} b 传给了形参 m color{purple}{m} m,此时局部变量 m color{purple}{m} m 也指向了 b color{red}{b} b 所指向的列表对象,此时 b color{red}{b} b 和 m color{purple}{m} m 共同指向列表对象。

④m.append(30)在列表对象中添加了一个新元素 i n t ( 30 ) color{green}{int(30)} int(30)

由于 m color{purple}{m} m 已经指向了列表对象,因此可以直接对列表对象进行 *** 作。于是成功在 b color{red}{b} b 和 m color{purple}{m} m 共同指向的列表对象上添加了元素 i n t ( 30 ) color{green}{int(30)} int(30)。
执行完函数后,栈内的栈帧消失,局部变量 m color{purple}{m} m 消失,此时又只有 b color{red}{b} b 指向列表对象。

此时栈堆情况:

b color{red}{b} b 指向的列表对象已经有了三个元素: 10 color{green}{10} 10、 20 color{green}{20} 20、 30 color{green}{30} 30 。

简单测试:

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/5720489.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-12-17
下一篇 2022-12-17

发表评论

登录后才能评论

评论列表(0条)

保存