叠加装饰器:

  叠加装饰器
    - 每一个新的功能都应该写一个新的装饰器
    - 否则会导致,代码冗余,结构不清晰,可扩展性差

  在同一个被装饰对象中,添加多个装饰器,并执行。
  @装饰1
  @装饰2
  @装饰3
  def 被装饰对象():
    pass

  注意: 装饰器在调用被装饰对象时才会执行添加的功能。

  - 叠加装饰器:
    - 装饰的顺序: 由下到上装饰
    - 执行的顺序: 由上往下

  注意: 无论inner中出现任何判断,最后都要返回“调用后的被装饰对象” func(*args, **kwargs)

def wrapper1(func):
def inner1(*args, **kwargs):
print('1---start')
# 被裝飾對象在調用時,如果還有其他裝飾器,會先執行其他裝飾器中的inner
# inner2
res = func(*args, **kwargs)
print('1---end')
return res
return inner1 def wrapper2(func):
def inner2(*args, **kwargs):
print('2---start')
res = func(*args, **kwargs)
print('2---end')
return res
return inner2 def wrapper3(func):
def inner3(*args, **kwargs):
print('3---start')
res = func(*args, **kwargs)
print('3---end')
return res
return inner3
'''
叠加裝飾器的裝飾順序與執行順序:
- 裝飾順序: 调用wrapper装饰器拿到返回值inner
由下往上裝飾 - 執行順序: 调用装饰过后的返回值inner
由上往下執行
''' @wrapper1 # index《---inner1 = wrapper1(inner2)
@wrapper2 # inner2 = wrapper2(inner3)
@wrapper3 # inner3 = wrapper3(index)
def index(): # 被裝飾對象 # inner1 ---》
print('from index...') # 正在装饰
index = wrapper3(index)
index = wrapper2(index)
index = wrapper1(index) '''
inner1()
inner2()
inner3()
index()
'''
index() # 此处执行 # inner1() --> inner2() ---> inner3()

有参装饰器

# 无参装饰器: 装饰在被装饰对象时,没有传参数的装饰器。
'''
# 以下是无参装饰器
@wrapper1 # inner1 = wrapper1(inner2)
@wrapper2 # inner2 = wrapper2(inner3)
@wrapper3
''' # 有参装饰器: 在某些时候,我们需要给用户的权限进行分类
'''
# 以下是有参装饰器
@wrapper1(参数1) # inner1 = wrapper1(inner2)
@wrapper2(参数2) # inner2 = wrapper2(inner3)
@wrapper3(参数3)
'''
def user_auth(user_role):  # 'SVIP'
def wrapper(func):
def inner(*args, **kwargs):
if user_role == 'SVIP':
# 添加超级用户的功能
res = func(*args, **kwargs)
return res
elif user_role == '普通用户':
print('普通用户')
# 添加普通用户的功能
res = func(*args, **kwargs)
return res return inner
return wrapper # 被装饰对象
user_auth('SVIP')
等价于下面两行代码
wrapper = user_auth('普通用户')
@wrapper # @user_auth('SVIP') # wrapper = user_auth('普通用户')
#@wrapper #<--- 返回结果(wrapper) <---- user_auth()
def index():
pass # wrapper = user_auth('普通用户') 等价于这两行
# index=wrapper(index) index()

wraps: (了解)
  是一个修复工具,修复的是被装饰对象的空间。不用显示外层装饰器注释
  from functools import wraps

from functools import wraps

def wrapper(func):

    @wraps(func)  # 修改名称空间: inner ---》 func   不写此处变为装饰器注释
def inner(*args, **kwargs):
'''
此处是装饰器的注释
:param func:
:return:
'''
res = func(*args, **kwargs)
return res
return inner # ---》 func @wrapper
def index():
'''
此处是index函数的注释
:return:
'''
pass print(index) # 函数对象 # 函数对象.__doc__: 查看函数内部的注释
print(index.__doc__) # inner.__doc__

迭代器

- 迭代器
  迭代的工具。

  迭代:
    迭代指的是重复迭代,每一次迭代都是基于上一次的结果而来的。

  迭代器:
    迭代器指的是迭代取值的工具,它可以迭代取值。

  - 如果想要知道python中迭代器是什么?必须先知道什么是可迭代对象?

  可迭代对象: 所有的序列类型: str, list, tuple, dict, set, f
    凡是内部有str.__iter__()方法的都是可迭代对象。

  - 获取迭代器:
    通过可迭代对象.__iter__(), 得到的返回值就是 “迭代器对象”。
    迭代器是迭代取值的工具,作用是迭代取值。

    - 如何迭代取值:
      迭代器对象.__next__() # “每一次执行”,都会从迭代器对象中取出一个值

  - 总结: 可迭代对象 VS 迭代器对象:
   - 获取可迭代对象: 定义序列类型(str, list, set, tuple, dict, f)
    - 特点:
      内置有__iter__()

  - 获取迭代器对象: 通过可迭代对象调用.__iter__()得到返回值
    - 特点:
      内置有__next__()

  - 迭代器对象的优点:
    - 优点:
      1.不依赖于索引迭代取值。
      2.节省内存空间。

    - 缺点:
      1.取指定某个值麻烦
      2.每次取值都要从第一个值开始,无法同过索引取值。

      3.不能通过len计算长度

# 以下都是可迭代对象
'''
str1 = 'hello tank!'
str1.__iter__()
list1 = [1, 2, 3] # list([1, 2, 3])
list1.__iter__()
set.__iter__()
dict.__iter__()
tuple.__iter__()
open('a.txt').__iter__()
'''
# list1是一个可迭代对象
list1 = ['tank', 'jason鸡哥', 'sean', '饼哥'] # 获取迭代器对象: iter_list1
iter_list1 = list1.__iter__() while True:
# 补充: try: 捕获异常
try:
print(iter_list1.__next__()) # 报错 # 立即触发此处代码 StopIteration
except StopIteration:
break
测试迭代文件
f = open('user.txt', 'r', encoding='utf-8')
iter_f = f.__iter__() while True:
try:
print(iter_f.__next__()) except StopIteration: break

for循环本质

for循环原理
  语法: for i in 可迭代对象:
  in: 可迭代对象 ----> 内部会自动调用.__iter__() ---> 迭代器对象

  for i in 可迭代对象:
  # 会自定帮你调用__next__
  print(i)
  # for循环内部也会有一个捕获异常机制,一旦遇到异常也会停止取值。

list1 = [1, 2, 3, 4]

for line in list1:  # list1是可迭代对象 ----> 内部会自动调用.__iter__() ---> 迭代器对象
# 迭代器对象.__next__()
print(line)

迭代器对象

- 迭代器本质上是一个可迭代对象

- 文件本质上既是迭代器对象,也是可迭代对象。

- 可迭代对象不一定是迭代器对象

iter_set1 = set1.__iter__()
iter_set1.__next__() # 确定: 迭代器对象也是一个可迭代对象
# 判断可迭代对象是否是迭代器对象
# print(iter_set1.__iter__() is iter_set1) # True # 唯独文件比较特殊: 因为文件从读取出来的时候就是一个迭代器对象
# f ---> 可迭代对象, 还是迭代器对象
# f = open('user.txt', 'r', encoding='utf-8')
# f.__next__()
#
# # # 确定: 文件既是可迭代对象,也是迭代器对象。
# # # iter_f ---> 迭代器对象
# iter_f = f.__iter__()
# iter_f.__next__()
'''
list1 = [1, 2, 3]
# 迭代器对象
iter_list = list1.__iter__() str1 = '123'
# 迭代器对象
iter_str1 = str1.__iter__() print(iter_list is iter_str1) # False
''' # 可迭代对象
list1 = [1, 2, 3, 4] # iter_list1 ---> 迭代器对象
iter_list1 = list1.__iter__()
print(iter_list1 is list1) # False
# 可迭代对象不一定是迭代器对象

python基础语法8 叠加装饰器,有参装饰器,wraps补充,迭代器的更多相关文章

  1. python基础语法及知识点总结

    本文转载于星过无痕的博客http://www.cnblogs.com/linxiangpeng/p/6403991.html 在此表达对原创作者的感激之情,多谢星过无痕的分享!谢谢! Python学习 ...

  2. Python 基础语法_Python脚本文件结构

    目录 目录 前言 软件环境 Python Script文件结构 导入模块的流程 Python的包package 最后 前言 Python基础语法这一章,主要记录了Python的文件结构.逻辑运算符.算 ...

  3. python之最强王者(2)——python基础语法

    背景介绍:由于本人一直做java开发,也是从txt开始写hello,world,使用javac命令编译,一直到使用myeclipse,其中的道理和辛酸都懂(请容许我擦干眼角的泪水),所以对于pytho ...

  4. Python 基础语法(三)

    Python 基础语法(三) --------------------------------------------接 Python 基础语法(二)------------------------- ...

  5. Python 基础语法(四)

    Python 基础语法(四) --------------------------------------------接 Python 基础语法(三)------------------------- ...

  6. Python 基础语法(二)

    Python 基础语法(二) --------------------------------------------接 Python 基础语法(一) ------------------------ ...

  7. Python 基础语法

    Python 基础语法 Python语言与Perl,C和Java等语言有许多相似之处.但是,也存在一些差异. 第一个Python程序 E:\Python>python Python 3.3.5 ...

  8. 吾八哥学Python(四):了解Python基础语法(下)

    咱们接着上篇的语法学习,继续了解学习Python基础语法. 数据类型大体上把Python中的数据类型分为如下几类:Number(数字),String(字符串).List(列表).Dictionary( ...

  9. python学习第五讲,python基础语法之函数语法,与Import导入模块.

    目录 python学习第五讲,python基础语法之函数语法,与Import导入模块. 一丶函数简介 1.函数语法定义 2.函数的调用 3.函数的文档注释 4.函数的参数 5.函数的形参跟实参 6.函 ...

随机推荐

  1. 了解编程语言 ----- c# 简介

    1.编程语言 编程语言: 为了实现人与机器的交互,计算机主要识别的就是 0 和 1 语言的发展过程主要分为: 1.面向机器的语言:二进制,汇编 2.面向过程的语言:汇编语言,C语言,B语言, 3.基于 ...

  2. Docker-Compose通过文件声明默认的环境变量

    写文目的 在写本文之前,我在做一个docker-compose项目,这里需要在docker-compose.yml配置中引用到宿主机的ip,然而docker-compose并没有命令行一个输入的选项, ...

  3. Net操作RabbitMQ

    原文:Net操作RabbitMQ 文章目录 1 安装 2 管理界面 3 RabbitMQ的基本概念 4 RabbitMQ的六种工作模式 4.1 简单模式 4.2 工作模式 4.3 发布/订阅模式 4. ...

  4. 推送一个docker 使用阿里docker hub

    阿里docker hub 地址 打开容器镜像服务页https://cr.console.aliyun.com/cn-qingdao/namespaces 这个是我的私有库 配置加速 我这边用的也是阿里 ...

  5. Java自学-接口与继承 多态

    Java的多态 操作符的多态 +可以作为算数运算,也可以作为字符串连接 类的多态 父类引用指向子类对象 示例 1 : 操作符的多态 同一个操作符在不同情境下,具备不同的作用 如果+号两侧都是整型,那么 ...

  6. HTML中marquee标签的使用

    HTML5之前,marquee作为滚动标签被经常使用,具体用法如下: <marquee>默认向左滚动</marquee> 选项: 1.behavior 滚动的方式 altern ...

  7. windows 下安装 node 以及安装 git 指令使用

    一.先百度搜索 node 找 Download | Node.js 二.下载 Node 根据你自己的操作系统下载 三.安装 node (注意:如果安装失败,使用管理员身份打开安装) 四.安装成功之后 ...

  8. VsCode中编写python环境配置

    1. VsCode中编写python环境配置 1.1. 前言 有过开发经验都知道idea一系列的软件虽然功能比较多,但比较容易卡,电脑不好还真容易上火,这里我想要入门python,还是选了款vscod ...

  9. python 之 sys.stdout输出不换行

    在处理程序打进度条时,希望不换行显示进度,可以使用sys.stdout相关函数来进行处理. 1.print 输出不换行 首先可以使用print函数来整体输入,利用,结尾就可以在同一行内显示: # py ...

  10. mongo数据库的一些命令(对于新同学,按照我的步骤连一遍即可)

    进入mongo mongo 查看数据库 show dbs;/show databases(); 第一个命令简单(admin和local数据库是系统自带的) 进入/创建数据库 use 数据库名字(如果有 ...