一.函数名
函数名是一个变量,但它是一个特殊的变量,与括号配合可以执行函数的变量,单纯print()出的是一个内存地址.
def func():
print('你说你有点难追')
print(func) #<function func at 0x000001F812922EA0>
########################单纯打印函数名打印的是一串内存地址############################## def func():
print('花店玫瑰')
other=func
print(other) #<function func at 0x0000013F2D7B2EA0>
other() #花店玫瑰
########################函数名可以赋值给其他变量########################### def func1():
print('广东')
def func2():
print('十年')
def func3():
print('爱情')
def func4():
print('故事')
list=[func1,func2,func3,func4]
for i in list:
i()
####################函数名可以当做容器类元素########################## # def func1():
# print('大猪蹄儿')
# def func2(fn):
# print('哗哗流油')
# fn()
# func2(func1)
###############################函数名可以当做函数的参数############################### def func1():
print('哈哈哈')
def func2():
print('呵呵呵')
return func1
set=func2()
set()
#######################函数名可以当做函数的返回值##################
二.闭包
闭包就是内层函数对外层函数(非全局)的变量的引用
2.1 __closure__用来检验函数是否闭包,格式(print(func.__closure__)),有返回值就是闭包,返回none就不是闭包.
def func1():
a='小花妹妹'
def func2():
print(a)
return func2
set1=func2()
def func3():
print('无所谓')
return func3 #在此处return func3也算引用外层变量
ret2=func3()
print(func3.__closure__) #没有引用外层变量就不是闭包
print(func2.__closure__) #引用了外层的变量,是闭包
print(func1.__closure__) #可以查看上一层的函数是不是闭包
return ret2
# func1()
print(func1.__closure__)
# print(func2.__closure__) #不能在父层查看子层的函数是不是闭包
ret=func1()
ret()
# func2() #只有闭包外层的才可以在外层调用内层的函数
2.2 多层嵌套
##############################多层嵌套#################################
def func1():
def func2():
def func3():
print('哈哈')
return func3
return func2
func1()()()
func2() #不能直接调用func2,因为此时内存只加载了func1,没有加载func2
#为什么要用三个括号:func1()表示调用func1,func1中的return func2语句返回 了func2,
# func2加一个括号才可以调用,执行func2中的return func3,
# 还需要载加一个括号才可以调用func3,func3打印出了哈哈
2.3 闭包的好处
1.保护变量不受外界影响
2.可以让变量常驻内存(正常情况下,当一个函数运行完毕后,,这个函数的局部命名空间将会被销毁,所以,python中规定,如果内层函数中访问了外层函数中的变量,即闭包的情况下,这个变量将不会消亡,也就是说,使用闭包,可以保证外层函数中的变量常驻内存)
简易爬虫
def outer():
# 常驻内存
s = urlopen("http://www.xiaohua100.cn/index.html").read()
def getContent(): # 闭包
return s
return getContent
print("爬取内容.....")
pa = outer() #此时的pa是getContent的内存地址
print (pa()) #相当于getContent() ret = pa() #直接调用getContent()
print(ret) #此时的s已经常驻内存,所以再次调用时速度非常快
2.4 闭包的写法
def outer():
a=10
def inner():
print(a)
return inner
zx_inner=outer()
zx_inner()
三.迭代器
iterable 可迭代的对象,内部包含__iter__()函数
object 对象,内部包含__iter__()函数和__next__()函数
l=[1,2,3]
l_iter=l.__iter__()
# print(l_iter)
from collections import Iterable
from collections import Iterator
print(isinstance(l,Iterable)) #判断,前的东西是不是,后的东西的实例
print(isinstance(l,Iterator))
print(isinstance(l_iter,Iterator))
print(isinstance(l_iter,Iterable))
True False True True ###################for的工作原理######################## s='你是不是傻'
c=s.__iter__()
print(c.__next__())#依次打印出你是不是傻
print(c.__next__())
print(c.__next__())
print(c.__next__())
print(c.__next__())
print(c.__next__())#StopIteration #已上会报错,所以for循环的机制会采用以下方法 lst=[1,2,3]
lst_iter=lst.__iter__()
while True:
try:
i=lst_iter.__next__()
print(i)
except StopIteration:
break ###################for的工作原理########################
迭代器的特点:
1.节省内存
2.惰性机制
3.不能反复,只能向下执行