在函数以外建立的变量属于__main__,又被称为全局变量。它们能够在__main__中的任意函数中访问,与局部变量在函数结束时消失不一样,全局变量能够在不一样函数的调用之间持久存在。全局变量经常用做标志(Flags)。标志是一种布尔型变量,能够标志一个条件是否为真。html
verbose = True
def example():
if verbose:
print('你好,今每天气很好!')
else:
print('你好')
复制代码
若是在函数里尝试给全局变量赋值,必须先用global关键字进行声明,不然函数会建立一个同名局部变量而不是使用全局变量。python
verbose = True
def example():
global verbose
verbose = False
print('你好')
复制代码
在Python中,string、tuple和number是不可变对象,而list、dict等是可变对象。shell
先来看一段代码:bash
b = [1, 2, 3]
a = b
print(a is b) # True
b.append(4)
print(a) # [1, 2, 3, 4]
复制代码
a is b
返回True,说明这python内部a与b是相同的,变量a与变量b都指向同一个对象。此时称a和b为这个对象的别名。当对象的值发生改变时,a和b天然也会随之改变。若是a、b只是值相等而不指向同一个对象,咱们称a与b是相等的。相同一定相等,相等不必定相同。app
a = [1, 2, 3]
b = [1, 2, 3]
print(a is b) # False
复制代码
咱们平时说的【变量】其实只是标签,是对内存中对象的引用。赋值操做只是给变量一个指向对象的引用函数
写代码的时候经常用is和==来比较两个对象是否相等,可是它们有什么不一样呢?参考下面的例子:性能
a = 1
b = 1
a == b # True
a is b # True
a = 888
b = 888
a == b # True
a is b # False
a = 'hello'
b = 'hello'
a is b # True
a == b # True
a = 'hello world'
b = 'hello world'
a == b # True
a is b # False
复制代码
is和==的结果不一样!不是说好的都是比较两个对象是否相等吗?怎么到这里变了样了?不急,先介绍一下python内置的一个函数:id(),这个函数会打印参数的内存地址,让咱们来试试:优化
a = 888
b = 888
id(a) # 1939743592336
id(b) # 1939745557808
a = 'hello world'
b = 'hello world'
id(a) # 1939745897200
id(b) # 1939745912624
复制代码
能够看到,尽管a、b的值是相同的,可是其内存地址却不一样。那么答案就很显然了,is比较的是两个对象的内存地址是否相等,==比较的是两个对象的值是否相等。这样就能解释为何is和==的结果不一样了。ui
But wait,那么为何当a、b的值为1和'hello'时,is与==的结果是同样的呢?这就要说到python的小整数池和垃圾回收机制了。python为了让运行速度快些,在内存中专门开辟了一块区域放置-5到256,全部表明-5到256的对象都会指向这个区域。spa
相似的,字符串类型做为Python中最经常使用的数据类型之一,Python解释器为了提升字符串使用的效率和使用性能,作了不少优化。
例如:Python解释器中使用了intern(字符串驻留)的技术来提升字符串效率,什么是intern机制?即值一样的字符串对象仅仅会保存一份,放在一个字符串储蓄池中,是共用的,固然,确定不能改变,这也决定了字符串必须是不可变对象。
同时,若是字符串中有空格,默认不启用intern机制。对字符串储蓄池中的字符串使用is和==比较会获得相同的结果。:
a = 1
b = 1
id(a) # 1963327952
id(b) # 1963327952
a = 'hello'
b = 'hello'
id(a) # 1939745887600
id(b) # 1939745887600
复制代码
注意:在shell中,仅有如下划线、数字、字母组成的字符串会被intern。而pycharm中只要是同一个字符串不超过20个字符都被加入到池中
python函数参数传递是引用传递:
def test(n):
print(id(n))
k = "string"
id(k) # 2305161642256
test(k) # 2305161642256
复制代码
经常使用的拷贝方式有:
浅拷贝属于“新瓶装旧酒”,即生成了一个新的变量,而变量所指向的对象和原来的是同样的:
l = ["hello", [2, 3, 4]]
id(l) # 3048239386824
[id(i) for i in l] # [1524761040, 1524761072]
k = l.copy()
id(k) # 3048239387080,地址不一样,k是另外一个变量
[id(i) for i in k] # [1524761040, 1524761072],地址相同,指向同一个变量
复制代码
深拷贝属于“新瓶装新酒”,即生成了一个新变量,指向和原对象相等的新对象(不可变对象除外):
import copy
l = ["hello world", [2, 3, 4]]
id(l) # 3048239386824
[id(i) for i in l] # [3048239385040, 3048239387080]
k = copy.deepcopy(l)
id(k) # 3048240927048,地址不一样,k是另外一个变量
[id(i) for i in k] # [3048239385040, 3048240927304],字符串是不可变对象,因此仍指向原地址,对于list
复制代码