【04】Python 深拷贝浅拷贝 函数 递归 集合
1 深拷贝浅拷贝
1.1 a==b与a is b的区别
- a == b 比较两个对象的内容是否相等(可以是不同内存空间)
- a is b 比较a与b是否指向同一个内存地址,也就是a与b的id是否相同
>>> a = 1
>>> b = 1
>>> a == b
True
>>> a is b
True
>>> a = 257
>>> b = 257
>>> a is b
False
出于对性能的考虑,Python内部做了很多的优化工作,对于整数对象,Python把一些频繁使用的整数对象缓存起来,保存到一个叫small_ints的链表中,在Python的整个生命周期内,任何需要引用这些整数对象的地方,都不再重新创建新的对象,而是直接引用缓存中的对象。Python把这些可能频繁使用的整数对象规定在范围[-5, 256]之间的小对象放在small_ints中,但凡是需要用些小整数时,就从这里面取,不再去临时创建新的对象。因为257不再小整数范围内,因此尽管a和b的值是一样,但是他们在Python内部却是以两个独立的对象存在的,各自为政,互不干扰。
>>> c = 257
>>> def foo():
... a = 257
... b = 257
... print a is b
... print a is c
...
>>> foo()
True
False
Python程序由代码块构成,代码块作为程序的一个最小基本单位来执行。一个模块文件、一个函数体、一个类、交互式命令中的单行代码都叫做一个代码块。在上面这段代码中,由两个代码块构成,c = 257作为一个代码块,函数foo作为另外一个代码块。Python内部为了将性能进一步的提高,凡是在一个代码块中创建的整数对象,如果存在一个值与其相同的对象于该代码块中了,那么就直接引用,否则创建一个新的对象出来。Python出于对性能的考虑,但凡是不可变对象,在同一个代码块中的对象,只有是值相同的对象,就不会重复创建,而是直接引用已经存在的对象。因此,不仅是整数对象,还有字符串对象也遵循同样的原则。所以 a is b就理所当然的返回True了,而c和a不在同一个代码块中,因此在Python内部创建了两个值都是257的对象。
- l2 = l1,相当于l2也指向与l1的内存地址,修改l1的值,l2也会跟着改变
l1 = [1,1,1,1,2,3,4,5]
l2 = l1 #浅拷贝, l1和l2指向同一个内存地址
print(id(l1)) #查看内存地址
print(id(l2))
for i in l2:
if i%2!=0:
l1.remove(i) #删除奇数
print(l1) #循环删list的时候,会导致下标错位,结果是不对的
运行结果如下:
42001160
42001160
[1, 1, 2, 4]
1.2 浅拷贝
import copy
l1 = [1,1,1,1,2,3,4,5]
l2 = l1 #浅拷贝, l和l2实际指向同一个内存地址
l3 = l1.copy() #浅拷贝
print(id(l1), id(l2), id(l3))
ll1 = [[1,2,3],[4,5,6]]
ll2 = ll1 #浅拷贝, l和l2实际指向同一个内存地址
ll3 = ll1.copy() #浅拷贝
print(id(ll1), id(ll2), id(ll3))
运行结果如下:
36164360 36164360 36164552
36165704 36165704 36165640
- 解析
1、b = a: 赋值引用,a 和 b 都指向同一个对象
2、b = a.copy(): 浅拷贝, a 和 b 是一个独立的对象,但他们的子对象还是指向统一对象(是引用)。
import copy
a = [1, 2, 3]
b = [4, 5, 6]
c = [a, b]
d = copy.copy(c)
print(id(c))
print(id(d))
42500296
42500232
import copy
a = [1, 2, 3]
b = [4, 5, 6]
c = (a, b)
d = copy.copy(c)
print(id(c))
print(id(d))
35510088
35510088
- 浅拷贝可变类型(比如列表),第一层的列表也会拷贝
- 浅拷贝不可变类型(比如元祖),d指向c的内存地址
1.3 深拷贝
import copy l = [1,1,1,2,3,4,5] l2 = copy.deepcopy(l)# 深拷贝
- 解析
b = copy.deepcopy(a): 深度拷贝, a 和 b 完全拷贝了父对象及其子对象,两者是完全独立的。
2 函数
2.1 函数定义
def db_connect(ip,port=3306): #1.位置参数,必填;2.默认值参数 print(ip,port) #函数体
def my3():
a = 1
b = 2
c = 3
return a,b,c b,c,d = my3()
s = my3()
print(b,c,d)
print(s)
运行结果:
1 2 3
(1, 2, 3)
2.2 return作用
1、结束函数,只要函数里面遇到return,函数立即结束运行
2、返回函数处理的结果
- 例:判断一个字符串是否是小数(含正小数和负小数)
def check_float(s):
'''
这个函数的作用就是判断传入的字符串是否是合法的小数
:param s: 传入一个字符串
:return: True/false
'''
s = str(s)
if s.count('.')==1:
s_split = s.split('.')
left,right = s_split
if left.isdigit() and right.isdigit():
return True
elif left.startswith('-') and left[1:].isdigit() \
and right.isdigit():
return True
return False #上面的几个条件未满足,则会走到这一步 print(check_float(1.3))
print(check_float(-1.3))
print(check_float('01.3'))
print(check_float('-1.3'))
print(check_float('-a.3'))
print(check_float('a.3'))
print(check_float('1.3a3'))
print(check_float('---.3a3'))
2.3 可变参数
def send_sms(*args): #可变参数,参数组
#1、不是必传的
#2、它把传入的元素全部都放到了一个元组里面
#3、不限制参数个数
#4、它用在参数比较多的情况下
for p in args:
print(p) send_sms()
print('-----------------------------')
send_sms(1861231231)
print('-----------------------------')
send_sms(1861231231,1232342,42342342)
运行结果:
-----------------------------
1861231231
-----------------------------
1861231231
1232342
42342342
2.4 关键字参数
def send_sms2(**kwargs):
#1、不是必传的
#2、不限制参数个数
print(kwargs) send_sms2()
send_sms2(name='xiaohei',sex='nan')
send_sms2(addr='北京',country='中国',c='abc',f='kkk')
运行结果:
{}
{'name': 'xiaohei', 'sex': 'nan'}
{'addr': '北京', 'country': '中国', 'c': 'abc', 'f': 'kkk'}
2.5 参数顺序
def my(name,country='China',*args,**kwargs):
#1、位置参数 2、默认值参数 3、可变参数 4、关键字
print(name)
print(country)
print(args)
print(kwargs)
my('xiaojun','Japan','beijing','天通苑',color='红色',
age=32)
运行结果:
xiaojun
Japan
('beijing', '天通苑')
{'color': '红色', 'age': 32}
2.6 全局变量
name = 'wangcan'#全局变量
names = []
def get_name():
names.append('hahaha')
name = 'hailong'
print('1、函数里面的name',name) def get_name2():
global name #声明name是全局变量
print('2、get_name2',name) get_name2()
get_name()
print(names)
print('3、函数外面的name',name)
#运行结果:
2、get_name2 wangcan
1、函数里面的name hailong
['hahaha']
3、函数外面的name wangcan
name = 'wangcan'#全局变量
def get_name3():
name = '我是谁' #不会对全局变量产生作用
print(name)
get_name3()
print(name)
#运行结果:
我是谁
wangcan
2.7 函数调用
def db_connect(ip, user, password, db, port):
print(ip)
print(user)
print(password)
print(db)
print(port) db_connect(user = 'abc', port= 3306, db = 1, ip = '', password = '') #记不住顺序可以这样调用
db_connect('','root', db = 2, password = '', port = 45)#这样也行, 混搭, 但'192','root'必须按顺序写在前面
3 递归
函数自己调用自己
递归最多999次
效率没有循环高
#阶乘 1 × 2 × 3 × ... × n
def fact(n):
if n==1:
return 1
return n * fact(n - 1)
4 集合
1、天生可以去重
2、集合是无序
- 集合定义
jihe = set()
- 集合例子
l=[1,1,2,2,3,3] res = set(l) print(res)
#运行结果:集合
{1, 2, 3}
4.1 交集、并集、差集、对称差集
xingneng =['tanailing','杨帆','liurongxin','小黑']
zdh = ['tanailing','杨帆','liurongxin','小军','海龙']
xingneng = set(xingneng)
zdh = set(zdh)
#取交集
res1 = xingneng.intersection(zdh)#取交集
res2 = xingneng & zdh #取交集
#取并集
res3 = xingneng.union(zdh) #取并集,把2个集合合并到一起,然后去重
res4 = xingneng | zdh
#取差集
res5 = xingneng.difference(zdh) #取差集,在a里面有,在b里面没有的
res6 = xingneng - zdh #取差集
#取对称差集
res7 =xingneng.symmetric_difference(zdh)#两个里不重复的值
res8 = xingneng ^ zdh print(res1)
print(res2)
print(res3)
print(res4)
print(res5)
print(res6)
print(res7)
print(res8)
#运行结果
{'liurongxin', '杨帆', 'tanailing'}
{'liurongxin', '杨帆', 'tanailing'}
{'tanailing', 'liurongxin', '小军', '海龙', '杨帆', '小黑'}
{'tanailing', 'liurongxin', '小军', '海龙', '杨帆', '小黑'}
{'小黑'}
{'小黑'}
{'海龙', '小军', '小黑'}
{'海龙', '小军', '小黑'}
4.2 其他方法
import string
l1 = set(string.ascii_lowercase)
l2 = {'a','b','c'}
print(l2.issubset(l1)) #子集
print(l1.issuperset(l2)) #父集
print(l1.isdisjoint(l2)) #有交集,返回False, 没有交集,返回True
#运行结果
True
True
False
l2 = {'a','b','c'}
l2.add('s') #增
print(l2) l2.remove('a') #删
print(l2) l2.pop()#随机删
print(l2)
#运行结果:
{'c', 'a', 'b', 's'}
{'c', 'b', 's'}
{'b', 's'}
【04】Python 深拷贝浅拷贝 函数 递归 集合的更多相关文章
- python深拷贝浅拷贝
python深拷贝和浅拷贝问题: 什么是深拷贝? (个人理解)深拷贝就是将原有的数据一模一样的拷贝一份,然后存到另一个地址中,而不是引用地址 什么是浅拷贝? (个人理解)就是引用地址 (1)用等于号的 ...
- 04 python学习笔记-函数、函数参数和返回值(四)
函数是组织好的,可重复使用的,用来实现单一,或相关联功能的代码段.函数能提高应用的模块性,和代码的重复利用率.Python提供了许多内建函数,比如print(),我们也可以自己创建函数,这叫做用户自定 ...
- Python开发【第一篇】Python基础之函数递归
函数递归 递归的本质: 就是一个函数调用另外一个函数. def d(): return '123' def c(): r = d() return r def b(): r = c() return ...
- Python基础(函数-递归)
本章内容: 深浅拷贝 函数(全局与局部变量) 内置函数 文件处理 三元运算 lambda 表达式 递归(斐波那契数列) 冒泡排序 深浅拷贝 一.数字和字符串 对于 数字 和 字符串 而言,赋值.浅拷贝 ...
- 【python】关于函数递归使用 return 后,收到数据为 None。
在写一个辗转相除求最小公因数的程序的时候,突然发现自己不管怎么写(除了两数恰巧可以整除),return 返回的值恒为 none. 代码为此: def gcd(a,b): if a%b==0: retu ...
- Python中的函数递归思想,以及对比迭代和递归解决Fibonacci数列
什么是递归?简单的说就是:函数自身调用自身. “普通程序员用迭代,天才程序员用递归” 虽然递归 在运行时会不断出栈压栈,调用底层的寄存器,造成空间上的占用以及时间上的缓慢, 但在一些算法上面仍然是递归 ...
- python学习-25 函数递归
递归 例如: def abc(n): print(n) if int(n/2) == 0: return n return abc(int(n/2)) abc(10) 运行结果: 10 5 2 1 P ...
- c# 内存的具体表现- 通用类型系统 深拷贝 浅拷贝 函数传参
c# 通用类型系统 及变量在 深拷贝 浅拷贝 函数传参 中的深层次的表现 在编程中遇到了一些想不到的异常,跟踪发现,自己对于c#变量在内存上的表现理解有偏差,系统的学习并通过代码实验梳理了各种情况下, ...
- python基础编程: 编码补充、文件操作、集合、函数参数、函数递归、二分查找、匿名函数与高阶函数
目录: 编码的补充 文件操作 集合 函数的参数 函数的递归 匿名函数与高阶函数 二分查找示例 一.编码的补充: 在python程序中,首行一般为:#-*- coding:utf-8 -*-,就是告诉p ...
随机推荐
- 在word中的表格指定位置插入一行
//创建一个Document类对象,并加载Word文档 Document doc = new Document(); doc.LoadFromFile(@"C:\Users\Administ ...
- 编译mysql时CMake Error at cmake/readline.cmake:85 (MESSAGE)
CMake Error at cmake/readline.cmake:85 (MESSAGE): Curses library not found. Please install appropr ...
- 惠普IPMI登陆不上
[问题描述] IPMI登陆不上(HP),点击无反应. 浏览器使用IE,java版本使用32位1.7版本. [问题原因] 保护此网站的证书使用弱加密,即 SHA1.此网站应该在 SHA1 被禁用之前将该 ...
- jmeter应用之批量插入数据
上一篇讲到如何在jmeter中配置并连接使用mysql数据库,这一节主要是讲数据库连接的简单应用——批量插入数据 总体步骤如下: 1)新建线程组和添加JDBC Connection Configura ...
- PhoneGap学习网址
官网:http://app-framework-software.intel.com/ 下载地址:http://download.csdn.net/download/haozq2012/7635951
- 【狗屁不通文章生成器】代码分析 (javaScript)
这几天在论坛上看到了一个很有意思的项目,一个生成"狗屁不通"的文章的程序.经过本人确定其的确是"狗屁不通"后,随后又好奇其实现,于是在其[GitHub]项目里( ...
- Linux菜狗入门(不停更新)
资料来源:<腾讯课堂> 1, 计算机硬件包括CPU,内存,硬盘,声卡等等 2, 没有安装操作系统的计算机,通常被称为裸机 如果想在裸机上运行自己所编写的程序,就必须用机器语言书写程序 如果 ...
- link标签中rel属性的作用
Link标签有两个作用:1. 定义文档与外部资源的关系:2. 是链接样式表.link标签是用于当前文档引用外部文档的 这个标签的rel属性用于设置对象和链接目标间的关系,说白了就是指明你链进来的对象是 ...
- datatable和dataset的区别
DataSet 是离线的数据源 DataTable 是数据源中的表.当然也可以自己建一张虚表.插入数据库中 DataSet是DataTable的容器DataSet可以比作一个内存中的数据库,DataT ...
- 1rem,1em,1vh,1px含义
rem:相对于页面根元素<html>元素,通常做法是给html元素设置一个字体大小,然后其他元素的大小就是相对于根元素的大小 em:相对于父元素字体大小,元素的width/height/p ...