python 05 关于对python中引用的理解
数据的在内存中的地址就是数据的引用。
如果两个变量为同一个引用,那么这两个变量对应的数据一定相同;
如果两个变量对应的数据相同,引用不一定相同。
通过id(数据)可以查看数据对应的地址,修改变量的值,其实是在修改变量的引用。
数据可以分为:可变类型与不变类型
可变类型:
如果修改了数据的内容,数据的地址没有发生改变.
有列表,字典,set集合
不可变类型:
如果修改了数据的内容,数据的地址发生改变.
有字符串,元组,数字
当python解释器首次启动时,会把小数字(-5~256)和短字符串(长度小于21)缓存到缓冲区中,当在程序中使用这些数字和字符串时,就直接从缓存区中取。
m = 300
n = 300
print(id(m)) # 1811121856
print(id(n))
不在小数字或者小字符串范围内的数据,会在第一次使用时缓存起来
m = 300
n = 300
print(id(m)) # 2345027009360
print(id(n)) # 2345027009360
演示:
1.
变量a实际存储的是1的引用(地址)(在程序执行过程中a被编译为一条指令)
a = 1
b = a
print(id(a)) # 1820749280
print(id(b)) # 1820749280
a =2 # 修改不可变类型(的引用)
print(id(a)) # 1820749312 a的id已经改变
print(id(b)) # 1820749280
print(a,b)
a = [1, 2,[3,5]]
b = a
print(id(a)) # 1758314288776
print(id(b)) # 1758314288776
# a.append(3)
a[0] = 6
a[2][0] = 1
b[2][1] = 6
print(id(a)) # 1758314288776 注意a与b始终指向同一个地址
print(id(b)) # 1758314288776
print(a) # [6, 2, [1, 6]]
print(b) # [6, 2, [1, 6]]
2.
list = []
dict = {"name":"wangjie","age":23}
print(id(dict))
a = 10
list.append(dict) # 把dict的引用加入到list中 0x111 的内容是(指向){"name":"wangjie","age":23}
list.append(a) # 把a的引用加入到list中 0x222 的内容是(指向)10
print(list) # list[0] 为0x111 ,内容是(指向)数据{"name":"wangjie","age":23},list[1]的内容0x222,内容是(指向)数据10
a = 20 # 修改了a的值 a的引用发生的变化 0x333
# 但不影响list中的引用指向的值 还是指向0x111 指向{"name":"wangjie","age":23}
# 和0x222 指向 10
print(list)
dict["name"] = "lili" #修改了dict的值 dict为可变数据类型,dict的引用不变,但0x111的内容已经变为{'name': 'lili', 'age': 23}
print(list) # list[0]的内容是 0x111 , 指向数据{'name': 'lili', 'age': 23},list[1]的内容为0x222指向数据 10
3.
list = []
list2 = []
m = 10
def func():
global m
m =20
list2 = [1,2] # 不属于修改,修改需要通过方法,这种是覆盖全局变量list2
list.append(1) # 通过append 方法修改,list的引用不变
print(list2) # [1, 2]
print(list) # []
print(m) # 10
print(id(m)) # 1811115776
func() # [1, 2]
print(list) # [1]
print(list2) # []
print(m) # 20
print(id(m)) # 1811116096
4.
def log2():
#info_dict = {} # 存储学生信息的字典若放在for循环外,则旧数据会被新数据覆盖
info_list = []
num = input("请输入要录入信息的数量")
num = int(num)
for i in range(num):
info_dict = {} # 重新直接对info_dict 赋值,info_dict的引用发生改变 存储学生信息的字典要放在for循环# 内
print("录入第%s 位信息" % (i + 1))
name = input("输入姓名:")
id = input("输入学号")
info_dict["姓名"] = name
info_dict["学号"] = id
info_list.append(info_dict)
print(info_list)
for info in info_list:
for k ,v in info.items():
print('%s = %s' % (k,v))
log2()
5.
a = [1, 2]
b = [3, 4]
a.append(b) # [1,2,[3,4]]
b.append(a)
print(a) # [1, 2, [3, 4, [...]]]
print(b) # [3, 4, [1, 2, [...]]]
6.传递数据,传递的是数据对应的地址.
a = [[]] * 5
print(a) [[],[],[],[],[]]
print(id(a)) # 2132205131400
print(id(a[0])) # 2132205131592
print(id(a[1])) # 2132205131592
a.append(1)
print(id(a)) # 2132205131400
print(a) # [[],[],[],[],[],1]
a[0].append(2)
print(id(a[0])) # 2132205131592
print(a) # [[2],[2],[2],[2],[2],1]
a[1].append(3)
print(id(a[1])) # 2132205131592
print(a) # [[2,3],[2,3],[2,3],[2,3],[2,3],1]
7.
>>> def selfAdd(a):
"""自增"""
a += a
>>> a_int = 1
>>> selfAdd(a_int)
>>> a_int
1
>>> a_list = [1,2]
>>> selfAdd(a_list)
>>> a_list
[1, 2, 1, 2]
总结:python中函数参数是引用传递(注意不是值传递)。对于不可变类型,因变量的值不能修改,所以运算不会影响到变量自身;而对于可变类型来说,函数体中的运算有可能
会更改传入参数的值
8.
+= 和append()方法等对数据引用的修改一样
9.
a = [1,2]
b = a * 2
print(b) # [1,2,1,2]
print(id(b)) # 2745079031688
b[0] = 55
print(b) # [55,2,1,2]
print(id(b)) # 2745079031688
print(a) # [1,2]
修改了b[0] ,则b[0] 的引用发生改变,但b的引用没发生变化
9.
第一种情况:
def func(x,l = []):
for i in range(x):
l.append(i+1)
print(l)
func(3) # [1,2,3]
func(2) # [1,2,3,1,2]
第二种情况:
def func(x,l = []):
for i in range(x):
l.append(i+1)
print(l)
list = [1,2]
func(3,list) # [1,2,1,2,3]
func(2) # [1,2]
函数有默认参数,定义时即为默认参数分配地址了,也只分配这一个地址,所以第一种情况,用的全是l这个列表;第二种调用func(3,list),将list变为了[1,2,1,2,3],然后调用func(2),l又指向了原先的已经定义好的空列表,所以结果为[1,2]。
python 05 关于对python中引用的理解的更多相关文章
- Python中在脚本中引用其他文件函数的方法
在导入文件的时候,Python只搜索当前脚本所在的目录,加载(entry-point)入口脚本运行目录和sys.path中包含的路径例如包的安装地址.所以如果要在当前脚本引用其他文件,除了将文件放在和 ...
- shell基础概念, if+命令, shell中引用python, shell脚本的几种执行方式
说明: 虚拟机中shell_test目录用来练习shell, 其中有个test.log文件用来存放日志 #!/usr/bin/bash # shell文件开头, 用来指定该文件使用哪个解释器 ...
- python函数定义中引用外部变量的一个问题
如果在函数定义的默认值中引用了一个外部变量,如下所示 x = 3 def func(a = x): print(a, x) 那么a的默认值就会是3, 但是print语句中的x会是调用时的x值 lamb ...
- java和python细节总结和java中string 的+操作
//JAVA中对arrayList的初始化,能够分配空间,不能之间让一个ArrayList赋值给另外一个ArrayList,这样是引用赋值,当一个改变时候,另外一个也改变 List<String ...
- (Python)导出指定文件夹中as文件的完全限定类名
AS3程序在编译的过程中,有一个特点是这样的,不管是项目中的类,还是标准库或者第三方库的类,编译的时候只会把用到的那些类文件编译进去,也就是说,某一些类,只要没有被主程序引用到,那这个文件是不会被编译 ...
- [Python][flask][flask-wtf]关于flask-wtf中API使用实例教程
简介:简单的集成flask,WTForms,包括跨站请求伪造(CSRF),文件上传和验证码. 一.安装(Install) 此文仍然是Windows操作系统下的教程,但是和linux操作系统下的运行环境 ...
- python调用shell, shell 引用python
python 调用 shell get_line_num="wc -l as_uniq_info | awk '{print $1}'" ###get the lines of & ...
- python笔记之提取网页中的超链接
python笔记之提取网页中的超链接 对于提取网页中的超链接,先把网页内容读取出来,然后用beautifulsoup来解析是比较方便的.但是我发现一个问题,如果直接提取a标签的href,就会包含jav ...
- 【Python 05】Python开发环境搭建
Python3安装和使用 1.安装 Python管方下载地址 选择Customize installation安装,并且勾选Add Python 3.X to PATH. 勾选Documentatio ...
随机推荐
- Struts+Spring+Hibernate处理Lob(Blob,Clob)
在使用struts+spring+hibernate的开发中,有些时候用户会有数据库存储文件的需求,在数据库中一般会采用Blob字段或Clob字段来存储二进制图片.流媒体或文件.现就将在实际开发中遇到 ...
- Centos 7.x 配置Gitlab
GitLab 是一个用于仓库管理系统的开源项目,使用Git作为代码管理工具,并在此基础上搭建起来的web服务. 1. 安装并配置必要的依赖关系 如果你想使用 Postfix 发送邮件,请在安装过程中根 ...
- luogu P3371 & P4779 单源最短路径spfa & 最大堆优化Dijkstra算法
P3371 [模板]单源最短路径(弱化版) 题目背景 本题测试数据为随机数据,在考试中可能会出现构造数据让SPFA不通过,如有需要请移步 P4779. 题目描述 如题,给出一个有向图,请输出从某一点出 ...
- CentOS 6.9 --Squid代理服务器
主机名 IP地址 网关 DNS 服务类型 Master eth0:192.168.17.130(VMnet4) eth1:192.168.30.130(NAT) 192.168.30.2 ...
- page.php 引入js文件
2种写法 <script type='text/javascript' src='<?php echo get_template_directory_uri().'/js/jquery-1 ...
- 【C#】将数据库读出的数据转换为DataTable类型集合
return View(ConverDataReaderToDataTable(reader)); // 静态方法public static DataTable ConverDataReaderToD ...
- LinQ的使用
LinQ:LinQ to Sql类 它是一个集成化的数据访问类,微软将原本需要我们自己动手去编写的一些代码,集成到了这个类中,会自动生成. LinQ的创建: 添加项——添加新项(LinQ to Sql ...
- Java GC机制简要总结(Java垃圾回收的基本工作原理)
第一次编辑 2019-05-07 01:09:39 垃圾回收的对象 程序中的不可用对象(不存活的对象,没有任何引用),或者无用的变量信息等,在程序中长期存在会逐渐占用较多的内存空间,导致没有足够的空间 ...
- TabLayout.Tab(自定义)点击事件
TabLayout是官方design包中的一个布局控件,这里不介绍它的基本使用,只是解决Tab(自定义)点击事件. //获取Tab的数量 Int tabCount = tabLayout.getTab ...
- QWidget标题栏双击事件
widget.h virtual bool event(QEvent *event); widget.cpp bool Widget::event(QEvent *event) { if (event ...