day012 python 闭包和迭代器

一  函数名的运用:(函数名是一个变量,但它是一个特殊变量,与括号配合可以执行变量.

 (1)  数名可以赋值给其他变量
def chi():
    print("吃月饼")
fn=chi  # 函数名可以进行赋值
a=10
b=a
chi()
fn()    # 函数名可以像变量名一样进行使用

def func1(): print(
"我是一个单纯的函数") def func2(abc): abc() print("我是func2,",abc) func2(func1) #func1是一个内存地址,函数名可以像变量一样进行参数传递

例子二:
def func():
    print("呵呵")
print(func)
a = func # 把函数当成一个变量赋值给另一个变量
a() # 函数调用 func()
 

pass

def outer():
    # print('我i')
    def inner():
        print("我是里面")
    return inner
outer()()  #表示的是执行 它可以执行里面的函数

ret=outer()    #outer的结果是inner
ret()      #执行的是inner函数

(2) 函数可以当作容器类元素

def chi():
    print("吃饭")
def he():
    print("喝饮料")
def du():
    print("赌是不好的")
def chou():
    print("少抽点烟")

lst = [chi, he, du, chou]
for el in lst:
    el() #表示执行

核心业务逻辑

def panpan():
#     print("我是潘潘. 我喜欢毒丈夫 ")
#
# def xiaoping():
#     print("我是小萍萍. 我喜欢毒丈夫 ")
#
# def xiaohua():
#     print("我是小花花. 我喜欢毒丈夫 ")
#
# def daguanren():
#     print("大官人喜欢xxxx")
#
# def wangpo(nv, nan): # 核心业务逻辑
#     nv()
#     nan()
#
# wangpo(xiaohua, daguanren) # 王婆代理了大官人和潘潘

(3) 函数名 可以当作函数的参数

def func():
    print("吃了么")
def func2(fn):
    print("我是func2")
    fn() # 执⾏行行传递过来的fn
# print("我是func2")
func2(func) # 把函数func当成参数传递给func2的参数fn.

(4)函数名可以作为函数的返回值

def func_1():
    print("这⾥是函数1")
    def func_2():
        print("这里是函数2")
    print("这⾥是函数1")
    return func_2
fn = func_1() # 执⾏行行函数1. 函数1返回的是函数2, 这时fn指向的就是上⾯面函数2
fn() # 执⾏行行上⾯面返回的函数

二 闭包 (函数的嵌套. 返回函数名)

写法:再外层函数中声明一个变量,在内层函数中使用或者返回这个变量,这个结构叫做闭包

基本写法
            def outer():
                a = 10
                def inner():
                    return a
                return inner

简单例子 看看闭包作用 其一:  可以保护我们的变量

 def func():
#     a = 10
#     print(a)
# func()
# print(a) # 在外面你是访问不到局部变量的, 局部变量是安全的


# 全局变量可能会被修改, 全局变量是不安全的. 可能会被其他函数所更改
# a = 10
# def func():
#     global a
#     a = 20
#     print(a)
# func()
# print(a)

(2) 可以让一个变量常驻内存:

def outer():
#     a = 10 # 常驻内存
#     def inner():
#         print(a) # 在内部使用的外面的变量
#     return inner # 返回了内部函数
 ret是inner的地址. ret就是inner
ret=outer()
ret() # 这里执行的是inner()

       print("哈哈")
# #    print("哈哈")
# #    print("哈哈")

       ret() # inner的执行时间是不确定的

      print("哈哈")
# #   print("哈哈")
# #   print("哈哈")
# #   ret() # inner的执行时间是不确定的


print(ret.__closure__) # 有东西, 就是闭包. None就不是闭包 j查看是不是闭包

#     闭包的应用.保护变量, 常驻内存

检查是否为闭包
def func1(): name
= "alex" def func2(): print(name) # 闭包 func2() print(func2.__closure__) # (<cell at 0x10c2e20a8: str object at 0x10c3fc650>,) func1()

小问题 ;如何在函数外部调用内部函数

def outer():
name = "alex"
# 内部函数
def inner():
print(name)
return inner
fn = outer() # 访问外部函数, 获取到内部函数的函数地址
fn() # 访问内部函数

小拓展:j简单爬网站 (具体是常驻内存的应用 , 目的为下次打开更快 无序加载,主要对象为一些刷新率低的网站,

from urllib.request import urlopen

def func():
    # 闭包. content会常驻内存
    content = urlopen("http://www.xiaohuar.com/").read()
    def inner():
        return content
    return inner

print("加载中...")
g = func() # 网络请求
print("加载完毕")
print(g())
print(g())
print(g())

三 迭代器   ( 

特点:
1. 节省内存(见明天博客)
2. 惰性机制(只有执行__next__()才会取值)
3. 只能向前. 不能反复

可跌代对象:

str, list, tuple, dict, set. open() .range() 那为什什么我们可以称他们为可迭代对象呢? 因为他们都遵循了了可
迭代协议.

dir() 可以查看某数据类型中可以执行的方法

例子1

s = "alex"
# print(dir(s)) # 在字符串中发现了__iter__. 没有__next__
# a = 123
# print(dir(a)) # 在int中没有__iter__ 没有__next__

例子2

lst = [1, 2, 3,]

lst = [1, 2, 3,]
# print(dir(lst)) # 在list中也有__iter__

例子3

a=123
# for i in 123:
print(dir(a)) #"int"object is not iterable (整数类型不可迭代)也没有__iter__

小结论: 所有包含了__iter__的东西都可以使用for循环,都可以进行迭代.

for 循环机制:

lst = [1, 2, 3, 4, 5, 6]
#
it = lst.__iter__() # iterator 迭代器
while 1:
try: # 表示放手去干的意思 不然打印出来会有小瑕疵

print(it.__next__())
except StopIteration: #出了事我兜着 不然打印出来会有小瑕疵
print("结束了")
break

对比下:

lst = [1, 2, 3, 4, 5]

for el in lst:
    print(el)
else:
    print("结束了")  两个结果一样  

(2)  迭代器给所有数据提供了一种统一的遍历方式(可跌代协议) ,Iterable,__iter__()


看是否是可跌代的 两种情况:

lst=[1,2,3,4,5] print("__iter__"in dir(lst)) # trut print("__next__"in dir(lst)) false # 换没执行 所以没有 # it = lst.__iter__() # 执行 # print("__iter__" in dir(it)) # ture 迭代器里面是有__iter__的. 迭代器一定是可迭代的 # print("__next__" in dir(it)) # ture

for el in it: # 迭代器可以使用for循环

# print(el)

 小结  :Iterable: 可迭代对象. 里面包含了__iter__(),可以使用for循环

         Iterator: 迭代器. 里面包含了__iter__() 和 __next__(), 也可以使用for循环

看是否是可跌代的还有看是不是迭代器  两种情况:

from
collections import Iterable # 可迭代的 from collections import Iterator # 迭代器 lst = ["周润发","麻花藤","刘伟"] print(isinstance(lst, Iterable)) # instance 实例, 对象 print(isinstance(lst, Iterator)) #instance 实例 对象 it = lst.__iter__() print(isinstance(it, Iterable)) # instance 实例, 对象 print(isinstance(it, Iterator)) # instance 实例, 对象
只能向前不能 反复

lst = [1,2,3] it = lst.__iter__() print(it.__next__()) print(it.__next__()) print(it.__next__()) #重新拿一个迭代器 执行完第一个就没了 t = lst.__iter__() print(it.__next__()) print(it.__next__()) print(it.__next__())

我们可以把要迭代的内容当成⼦子弹. 然后呢. 获取到迭代器__iter__(), 就把⼦子弹都装在弹夹
中. 然后发射就是__next__()把每⼀一个⼦子弹(元素)打出来. 也就是说, for循环的时候. ⼀一开始的
时候是__iter__()来获取迭代器. 后⾯面每次获取元素都是通过__next__()来完成的. 当程序遇到
StopIteration将结束循环.

猜你喜欢

转载自www.cnblogs.com/systemsystem/p/9665165.html