1、函数名的应用html
# 1,函数名就是函数的内存地址,而函数名()则是运行这个函数。 def func(): return print(func) # 返回一个地址 # 2,函数名能够做为变量。 def func1(): print(666) f1 = func1 f2 = f1 f2() # 就等于func1() 此时执行函数 # 3,函数名能够做为函数的参数。 def func1(): print(666) def func2(x): x() func2(func1) # 输出666 func1做为实参传进func2中,此时在func2中x()等于func1() # 4,函数名能够当作函数的返回值。 def wraaper(): def inner(): print(666) return inner ret = wraaper() # 执行函数wraaper(),获得返回值inner赋值给ret,这个inner是函数名 ret() # 输出666,ret()等于inner() # (注意:这里要调用inner()只能是用ret(),由于inner()是wraaper()的嵌套函数,在全局命名空间中并无声明,因此直接用inner()不能执行,) def func2(): print('in func2') def func3(x): print('in func3') return x f = func3(func2) # 给func3传进一个参数func2,返回func2赋值给f f() # f()等于func2() # 5,函数名能够做为容器类类型的元素。 def f1(): print('f1') def f2(): print('f2') def f3(): print('f3') l = [f1, f2, f3] d = {'f1': f1, 'f2': f2, 'f3': f3} # 调用 l[0]() # 输出 f1 d['f2']() # 输出 f2 def func1(): print('in func1') def func2(): print('in func2') def func3(): print('in func3') def func4(): print('in func4') l1 = [func1,func2,func3,func4] #调用 for i in l1: i() # 像上面的函数名这种,称为第一类对象。 # 第一类对象(first-class object)指: # 1.可在运行期建立 # 2.可用做函数参数或返回值 # 3.可存入变量的实体。 # (若是不明白,那就记住一句话,就当普通变量用)
六、globals() locals()
globals() #返回全局变量(包含内置函数)的一个字典。
locals() #返回当前位置的变量的字典。
例如:闭包
def func1(): a = 2 b = 3 print(globals()) # 此时globals()在局部命名空间中,但也是返回全局命名空间的一个字典 print(locals()) # 此时locals()在局部命名空间中,因此返回的是局部命名空间的一个字典{'b': 3, 'a': 2} print(globals()) # 此时globals()在全局命名空间中,返回全局命名空间的一个字典 print(locals()) # 此时locals()在全局命名空间中,因此返回的是全局命名空间的一个字典 func1() # 结果: # {'__name__': '__main__', '__doc__': None(后面还有一大推东西,省略)...} # {'__name__': '__main__', '__doc__': None(后面还有一大推东西,省略)...} # {'__name__': '__main__', '__doc__': None(后面还有一大推东西,省略)...} # {'b': 3, 'a': 2}
总结:
globals()不管在哪一个命名空间,返回的都是全局命名空间的一个字典
locals()在哪一个命名空间就返回哪一个命名空间的字典函数
def func1(): a = 2 b = 3 def inner(): c = 5 d = 6 print(globals()) #{'__name__': '__main__', '__doc__': None(后面还有一大推东西,省略)...} print(locals()) #{'c': 5, 'd': 6} inner() func1()
2、闭包
一、内层函数引用外层函数的变量(非全局变量),这样该内部函数称就称为闭包函数。
(咱们都知道函数内的变量咱们要想在函数外部用,能够直接返回这个变量,那么若是咱们想在函数外部调用函数内部的函数呢?
就把这个函数的名字返回就行了,这是闭包函数最经常使用的用法)post
def wraaper(): name = '鬼见愁' def inner(): print(name) return inner f = wraaper() f()
二、判断闭包函数的方法__closure__ this
# 1,是闭包返回值会有cell元素 def wraaper(): name = '鬼见愁' def inner(): print(name) print(inner.__closure__) return inner f = wraaper() f() # 2,不是闭包就会返回None name = '鬼见愁' def wraaper(): def inner(): print(name) print(inner.__closure__) # None return inner f = wraaper() f() name = '番薯' def wraaper(n): n = '番薯' def inner(): print(n) print(inner.__closure__) # cell at 0x000002AD93BF76D8 inner() return inner wraaper(name)
'''
闭包做用:
当程序执行时,遇到了函数执行,他会在内存中开辟一个空间,局部名称空间,
若是这个函数内部造成了闭包,
那么它就不会随着函数的结束而消失。
'''url
何时用到闭包?
例如:爬虫,装饰器等
下面是一个爬虫的小案例:spa
from urllib.request import urlopen def index(): url = "http://www.xiaohua100.cn/index.html" def get(): return urlopen(url).read() return get xiaohua = index() # get content = xiaohua() # get() print(content.decode('utf-8'))
3、可迭代对象
for i in 'abc':
print(i)code
for i in 123:
print(i) # 'int' object is not iterable表示不是可迭代对象htm
对象内部含有__iter__方法就是可迭代对象.
可迭代对象知足可迭代协议。对象
可迭代对象:str list dict,tuple,set,range()
dir() :返回一个列表,列表里面包含了传入的参数的属性、方法
s1 = 'strs'
print(dir(s1))
判断一个对象是不是可迭代对象:
# 第一个方法 s1 = 'abc' dic = {'name':'xiaoming'} print('__iter__' in dir(s1)) print('__iter__' in dir(dic)) # 第二种方法 from collections import Iterable #判断是否为可迭代对象 from collections import Iterator #判断是否为迭代器 print(isinstance('xiaoming',Iterable)) # True print(isinstance('xiaoming',Iterator)) # False print(isinstance('xiaoming',str)) # True #isinstance() 应用比type()更普遍,isinstance()不只能够判断你是否是可迭代对象, #也可判断你是否迭代器,还能够判断你是什么类型的数据等等,而#type()只能判断你是什么数据类型
4、迭代器
一、对象内部含有__iter__方法且含有__next__方法就是迭代器。
#文件句柄是迭代器: f = open('register', encoding='utf-8') print('__iter__' in dir(f)) #True print('__next__' in dir(f)) #True #字典是可迭代对象,不是迭代器。 print('__iter__' in dir(dict)) #True print('__next__' in dir(dict)) #False
二、可迭代对象与迭代器的区别:
1,可迭代对象不能取值,迭代器是能够取值的。
(咱们知道字典列表都是可迭代对象,为何它们能取值呢?这是由于为了方便咱们操做,咱们设定了
索引(键)用来操做字典列表取值,若是没有人为为它们设置的索引(键),它们是不能取值的)
2, 迭代器很是节省内存。
3,迭代器每次只会取一个值。
4,迭代器单向的,一条路走到头。
三、可迭代对象 --->(转化成)迭代器
lis = [1, 2, 3] # 可迭代对象 ite1 = lis.__iter__() # 迭代器 <list_iterator object at 0x0000027A183BFFD0> ite1 = iter(lis) # 迭代器 <list_iterator object at 0x0000027A183BFFD0> print(ite1)
四、迭代器如何取值? next一次,取一个值
print(ite1.__next__()) print(ite1.__next__()) print(ite1.__next__())
五、while循环模拟for循环机制
1,将可迭代对象转化成迭代器。
2,调用__next__方法取值。
3,利用异常处理中止报错。
s1 = 'abcdefg' iter1 = s1.__iter__() while 1: try: print(iter1.__next__()) except StopIteration: break