数据的在内存中的地址就是数据的引用。
  如果两个变量为同一个引用,那么这两个变量对应的数据一定相同;
  如果两个变量对应的数据相同,引用不一定相同。
  通过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中引用的理解的更多相关文章

  1. Python中在脚本中引用其他文件函数的方法

    在导入文件的时候,Python只搜索当前脚本所在的目录,加载(entry-point)入口脚本运行目录和sys.path中包含的路径例如包的安装地址.所以如果要在当前脚本引用其他文件,除了将文件放在和 ...

  2. shell基础概念, if+命令, shell中引用python, shell脚本的几种执行方式

    说明: 虚拟机中shell_test目录用来练习shell, 其中有个test.log文件用来存放日志 #!/usr/bin/bash      # shell文件开头, 用来指定该文件使用哪个解释器 ...

  3. python函数定义中引用外部变量的一个问题

    如果在函数定义的默认值中引用了一个外部变量,如下所示 x = 3 def func(a = x): print(a, x) 那么a的默认值就会是3, 但是print语句中的x会是调用时的x值 lamb ...

  4. java和python细节总结和java中string 的+操作

    //JAVA中对arrayList的初始化,能够分配空间,不能之间让一个ArrayList赋值给另外一个ArrayList,这样是引用赋值,当一个改变时候,另外一个也改变 List<String ...

  5. (Python)导出指定文件夹中as文件的完全限定类名

    AS3程序在编译的过程中,有一个特点是这样的,不管是项目中的类,还是标准库或者第三方库的类,编译的时候只会把用到的那些类文件编译进去,也就是说,某一些类,只要没有被主程序引用到,那这个文件是不会被编译 ...

  6. [Python][flask][flask-wtf]关于flask-wtf中API使用实例教程

    简介:简单的集成flask,WTForms,包括跨站请求伪造(CSRF),文件上传和验证码. 一.安装(Install) 此文仍然是Windows操作系统下的教程,但是和linux操作系统下的运行环境 ...

  7. python调用shell, shell 引用python

    python 调用 shell get_line_num="wc -l as_uniq_info | awk '{print $1}'" ###get the lines of & ...

  8. python笔记之提取网页中的超链接

    python笔记之提取网页中的超链接 对于提取网页中的超链接,先把网页内容读取出来,然后用beautifulsoup来解析是比较方便的.但是我发现一个问题,如果直接提取a标签的href,就会包含jav ...

  9. 【Python 05】Python开发环境搭建

    Python3安装和使用 1.安装 Python管方下载地址 选择Customize installation安装,并且勾选Add Python 3.X to PATH. 勾选Documentatio ...

随机推荐

  1. Struts+Spring+Hibernate处理Lob(Blob,Clob)

    在使用struts+spring+hibernate的开发中,有些时候用户会有数据库存储文件的需求,在数据库中一般会采用Blob字段或Clob字段来存储二进制图片.流媒体或文件.现就将在实际开发中遇到 ...

  2. Centos 7.x 配置Gitlab

    GitLab 是一个用于仓库管理系统的开源项目,使用Git作为代码管理工具,并在此基础上搭建起来的web服务. 1. 安装并配置必要的依赖关系 如果你想使用 Postfix 发送邮件,请在安装过程中根 ...

  3. luogu P3371 & P4779 单源最短路径spfa & 最大堆优化Dijkstra算法

    P3371 [模板]单源最短路径(弱化版) 题目背景 本题测试数据为随机数据,在考试中可能会出现构造数据让SPFA不通过,如有需要请移步 P4779. 题目描述 如题,给出一个有向图,请输出从某一点出 ...

  4. CentOS 6.9 --Squid代理服务器

    主机名 IP地址  网关   DNS   服务类型  Master eth0:192.168.17.130(VMnet4) eth1:192.168.30.130(NAT) 192.168.30.2 ...

  5. page.php 引入js文件

    2种写法 <script type='text/javascript' src='<?php echo get_template_directory_uri().'/js/jquery-1 ...

  6. 【C#】将数据库读出的数据转换为DataTable类型集合

    return View(ConverDataReaderToDataTable(reader)); // 静态方法public static DataTable ConverDataReaderToD ...

  7. LinQ的使用

    LinQ:LinQ to Sql类 它是一个集成化的数据访问类,微软将原本需要我们自己动手去编写的一些代码,集成到了这个类中,会自动生成. LinQ的创建: 添加项——添加新项(LinQ to Sql ...

  8. Java GC机制简要总结(Java垃圾回收的基本工作原理)

    第一次编辑 2019-05-07 01:09:39 垃圾回收的对象 程序中的不可用对象(不存活的对象,没有任何引用),或者无用的变量信息等,在程序中长期存在会逐渐占用较多的内存空间,导致没有足够的空间 ...

  9. TabLayout.Tab(自定义)点击事件

    TabLayout是官方design包中的一个布局控件,这里不介绍它的基本使用,只是解决Tab(自定义)点击事件. //获取Tab的数量 Int tabCount = tabLayout.getTab ...

  10. QWidget标题栏双击事件

    widget.h virtual bool event(QEvent *event); widget.cpp bool Widget::event(QEvent *event) { if (event ...