前面有提到由于 Python 基于 C 语言编写的解释器中设置了一个 GIL 全局变量锁,该锁使得 Python 的多线程在处理 CPU 计算密集型任务时,同一时刻只能有一个线程在运行,这也是为什么说 Python 的多线程是一个「假性多线程」的原因。
解决 GIL 的办法在处理 CPU 计算密集型任务时,使用多进程 + 协程,发挥计算机多核的威力,而处理 I/O 密集型,则可以使用多线程。
16.说一说 Python 中的垃圾回收机制(GC)
引用计数清零
标记-清除
隔代回收
这里有一篇文章可以参考一下。垃圾回收机制
17.说一说 Python 中的函数式编程
lambda
lambda 匿名函数,使用匿名函数可以帮助我们简化代码,且节省程序由于显示创建函数的消耗。
示例:
# 冒号左边是参数, 右边是表达式,lambda 返回的是右边表达式的结果
L2 = lambda x : x * x
print(L2(5))
# 结果为 25
map
map(func,*Iterable),map 函数通过将可迭代对象传入 func 函数,全部执行。
示例:
# 相当于 [1*1, 2*2, 3*3, 4*4]
L2 = list(map(lambda x: x * x, range(1, 4)))
print(L2)
# 结果为
[1, 4, 9, 16]
reduce
reduce(func, *sequence),reduce 函数可以将函数 func 迭代调用。
示例:
from functools import reduce
# 相当于 ((1*2)*3)
L2 = reduce(lambda x, y: x * y, range(1, 4))
print(L2)
# 结果为
6
filter
filter(func,*Iterable),filter 函数通过将可迭代对象当做参数传递给 func 函数,并返回 func 函数中返回结果为 True 的值。
示例:
# 相当于 g = [x for x in range(10) if x < 3]
L2 = list(filter(lambda x: x < 3, range(10)))
print(L2)
# 结果为
[0, 1, 2]
18.is 和 == 的区别
a is b 比较的是 a 和 b 的「id」是否相同
a == b 比较的是 a 和 b 的「值」是否相同
19.range 和 xrange 的区别
对于现如今的 Python 3.x,这个问题其实已经没有什么意义了,由于其在各大面经中都出现过,这里简单解释下。在 Python 2.x 中,range和xrange最大的区别就是xrange生成的不是一个list,而是一个生成器,这两者的区别在问题 13 已经讲过,这里不在赘述。
Python 3.x 中已经移除了range()方法,并将xrange()方法更名为range()。
20.新式类和旧式类的区别
新式类和旧式类的变化主要讲的是 Python 中类的「继承」问题。
旧式类中,类的继承按照「从左往右,深度优先遍历」的原则。
新式类中,类的继承按照「C3 算法」的原则:如果子类在继承来自不同父类的方法是不一样的时候,会从下而上,先左后右的合并继承方法。如果有一样的方法,则会继承第一个方法然后舍弃另一个相同的方法,接着合并其他不同的方法。解决了旧式类继承中存在的二义性和单调性。
示例:
# 新式类
class B:
def __init__(self,name):
self.name = name
def get_name(self):
return 'B'
class C:
def __init__(self,name):
self.name = name
def get_name(self):
return 'C'
class BC(B, C):
pass
b = B('B')
c = C('C')
bc = BC('C')
print(bc.name)
name = bc.get_name()
print(name)
# 查看 c 在搜索方法时的先后顺序
print(bc.__mro__)
# 结果为
C
B
(, , , )
另外,新式类相对于旧式类还多了这么些变化。
__slot__:限制实例属性的设置范围。
__getattrbute__:获取实例属性值。
type与__class__:返回类型统一。