醉了,面个功能测试,还问我Python装饰器

news2024/9/23 6:24:49

 

Python 装饰器是个强大的工具,可帮你生成整洁、可重用和可维护的代码。某种意义上说,会不会用装饰器是区分新手和老鸟的重要标志。如果你不熟悉装饰器,你可以将它们视为将函数作为输入并在不改变其主要用途的情况下扩展其功能的函数。装饰器可以有效提高你的工作效率并避免重复代码。本文我整理了项目中经常用到的 12 个装饰器,值得每一个Python开发者掌握。

01 @logger

我们从最简单的装饰器开始,手动实现一个可以记录函数开始和结束的装饰器。

被修饰函数的输出结果如下所示:

  1. some_function(args)

  2. # ----- some_function: start -----

  3. # some_function executing

  4. # ----- some_function: end -----

要实现一个装饰器,首先要给装饰器起一个合适的名称:这里我们给装饰器起名为logger。

装饰器本质上是一个函数,它将一个函数作为输入并返回一个函数作为输出。 输出函数通常是输入的扩展版。在我们的例子中,我们希望输出函数用start和end语句包围输入函数的调用。

由于我们不知道输入函数都带有什么参数,我们可以使用 *args 和 **kwargs 从包装函数传递它们。*args 和 **kwargs 允许传递任意数量的位置参数和关键字参数。

下面是logger装饰器的示例代码:

  1. def logger(function):

  2.     def wrapper(*args, **kwargs):

  3.         print(f"----- {function.__name__}: start -----")

  4.         output = function(*args, **kwargs)

  5.         print(f"----- {function.__name__}: end -----")

  6.         return output

  7.     return wrapper

logger函数可以应用于任意函数,比如:

decorated_function = logger(some_function)

上面的语句是正确的,但Python 提供了更 Pythonic 的语法——使用 @ 修饰符。

因此更通常的写法是:

  1. @logger

  2. def some_function(text):

  3.     print(text)

  4. some_function("first test")

  5. # ----- some_function: start -----

  6. # first test

  7. # ----- some_function: end -----

  8. some_function("second test")

  9. # ----- some_function: start -----

  10. # second test

  11. # ----- some_function: end -----

02 @wraps

要了解 @wraps 的作用以及为什么需要它,让我们将前面写的logger装饰器应用到一个将两个数字相加的简单函数中。

下面的代码是未使用@wraps装饰器的版本

  1. def logger(function):

  2.     def wrapper(*args, **kwargs):

  3.         """wrapper documentation"""

  4.         print(f"----- {function.__name__}: start -----")

  5.         output = function(*args, **kwargs)

  6.         print(f"----- {function.__name__}: end -----")

  7.         return output

  8.     return wrapper

  9. @logger

  10. def add_two_numbers(a, b):

  11.     """this function adds two numbers"""

  12.     return a + b

如果我们用__name__ 和 __doc__来查看被装饰函数add_two_numbers的名称和文档,会得到如下结果

  1. add_two_numbers.__name__

  2. 'wrapper'

  3. add_two_numbers.__doc__

  4. 'wrapper documentation'

输出的是wrapper函数的名称和文档。这是我们预期想要的结果,我们希望保留原始函数的名称和文档。这时@wraps装饰器就派上用场了。

我们唯一需要做的就是给wrapper函数加上@wraps装饰器。

  1. from functools import wraps

  2. def logger(function):

  3.     @wraps(function)

  4.     def wrapper(*args, **kwargs):

  5.         """wrapper documentation"""

  6.         print(f"----- {function.__name__}: start -----")

  7.         output = function(*args, **kwargs)

  8.         print(f"----- {function.__name__}: end -----")

  9.         return output

  10.     return wrapper

  11. @logger

  12. def add_two_numbers(a, b):

  13.     """this function adds two numbers"""

  14.     return a + b

再此检查add_two_numbers函数的名称和文档,我们可以看到该函数的元数据。

  1. add_two_numbers.__name__

  2. # 'add_two_numbers'

  3. add_two_numbers.__doc__

  4. # 'this function adds two numbers'

03 @lru_cache

@lru_cache是Python内置装饰器,可以通过from functools import lru_cache引入。@lru_cache的作用是缓存函数的返回值,当缓存装满时,使用least-recently-used(LRU)算法丢弃最少使用的值。

@lru_cache装饰器适合用于输入输出不变且运行时间较长的任务,例如查询数据库、请求静态页面或一些繁重的处理。

在下面的示例中,我使用@lru_cache来修饰一个模拟某些处理的函数。然后连续多次对同一输入应用该函数。

  1. import random

  2. import time

  3. from functools import lru_cache

  4. @lru_cache(maxsize=None)

  5. def heavy_processing(n):

  6.     sleep_time = n + random.random()

  7.     time.sleep(sleep_time)

  8. # 初次调用

  9. %%time

  10. heavy_processing(0)

  11. # CPU times: user 363 µs, sys: 727 µs, total: 1.09 ms

  12. # Wall time: 694 ms

  13. # 第二次调用

  14. %%time

  15. heavy_processing(0)

  16. # CPU times: user 4 µs, sys: 0 ns, total: 4 µs

  17. # Wall time: 8.11 µs

  18. # 第三次调用

  19. %%time

  20. heavy_processing(0)

  21. # CPU times: user 5 µs, sys: 1 µs, total: 6 µs

  22. # Wall time: 7.15 µs

从上面的输出可以看到,第一次调用花费了694ms,因为执行了time.sleep()函数。后面两次调用由于参数相同,直接返回缓存值,因此并没有实际执行函数内容,因此非常快地得到函数返回。

04 @repeat

该装饰器的所用是多次调用被修饰函数。这对于调试、压力测试或自动化多个重复任务非常有用。

跟前面的装饰器不同,@repeat接受一个输入参数,

  1. def repeat(number_of_times):

  2.     def decorate(func):

  3.         @wraps(func)

  4.         def wrapper(*args, **kwargs):

  5.             for _ in range(number_of_times):

  6.                 func(*args, **kwargs)

  7.         return wrapper

  8.     return decorate

上面的代码定义了一个名为repeat的装饰器,有一个输入参数number_of_times。与前面的案例不同,这里需要decorate函数来传递被修饰函数。然后,装饰器定义一个名为wrapper的函数来扩展被修饰函数。

  1. @repeat(5)

  2. def hello_world():

  3.     print("hello world")

  4. hello_world()

  5. # hello world

  6. # hello world

  7. # hello world

  8. # hello world

  9. # hello world

05 @timeit

该装饰器用来测量函数的执行时间并打印出来。这对调试和监控非常有用。

在下面的代码片段中,@timeit装饰器测量process_data函数的执行时间,并以秒为单位打印所用的时间。

  1. import time

  2. from functools import wraps

  3. def timeit(func):

  4.     @wraps(func)

  5.     def wrapper(*args, **kwargs):

  6.         start = time.perf_counter()

  7.         result = func(*args, **kwargs)

  8.         end = time.perf_counter()

  9.         print(f'{func.__name__} took {end - start:.6f} seconds to complete')

  10.         return result

  11.     return wrapper

  12. @timeit

  13. def process_data():

  14.     time.sleep(1)

  15. process_data()

  16. # process_data took 1.000012 seconds to complete

06 @retry

其工作原理如下:

  • wrapper函数启动num_retrys次迭代的for循环。

  • 将被修饰函数放到try/except块中。每次迭代如果调用成功,则中断循环并返回结果。否则,休眠sleep_time秒后继续下一次迭代。

  • 当for循环结束后函数调用依然不成功,则抛出异常。

示例代码如下:

  1. import random

  2. import time

  3. from functools import wraps

  4. def retry(num_retries, exception_to_check, sleep_time=0):

  5.     """

  6.     遇到异常尝试重新执行装饰器

  7.     """

  8.     def decorate(func):

  9.         @wraps(func)

  10.         def wrapper(*args, **kwargs):

  11.             for i in range(1, num_retries+1):

  12.                 try:

  13.                     return func(*args, **kwargs)

  14.                 except exception_to_check as e:

  15.                     print(f"{func.__name__} raised {e.__class__.__name__}. Retrying...")

  16.                     if i < num_retries:

  17.                         time.sleep(sleep_time)

  18.             # 尝试多次后仍不成功则抛出异常

  19.             raise e

  20.         return wrapper

  21.     return decorate

  22. @retry(num_retries=3, exception_to_check=ValueError, sleep_time=1)

  23. def random_value():

  24.     value = random.randint(1, 5)

  25.     if value == 3:

  26.         raise ValueError("Value cannot be 3")

  27.     return value

  28. random_value()

  29. # random_value raised ValueError. Retrying...

  30. # 1

  31. random_value()

  32. # 5

07 @countcall

@countcall用于统计被修饰函数的调用次数。这里的调用次数会缓存在wraps的count属性中。

  1. from functools import wraps

  2. def countcall(func):

  3.     @wraps(func)

  4.     def wrapper(*args, **kwargs):

  5.         wrapper.count += 1

  6.         result = func(*args, **kwargs)

  7.         print(f'{func.__name__} has been called {wrapper.count} times')

  8.         return result

  9.     wrapper.count = 0

  10.     return wrapper

  11. @countcall

  12. def process_data():

  13.     pass

  14. process_data()

  15. process_data has been called 1 times

  16. process_data()

  17. process_data has been called 2 times

  18. process_data()

  19. process_data has been called 3 times

08 @rate_limited

@rate_limited装饰器会在被修饰函数调用太频繁时,休眠一段时间,从而限制函数的调用速度。这在模拟、爬虫、接口调用防过载等场景下非常有用

  1. import time

  2. from functools import wraps

  3. def rate_limited(max_per_second):

  4.     min_interval = 1.0 / float(max_per_second)

  5.     def decorate(func):

  6.         last_time_called = [0.0]

  7.         @wraps(func)

  8.         def rate_limited_function(*args, **kargs):

  9.             elapsed = time.perf_counter() - last_time_called[0]

  10.             left_to_wait = min_interval - elapsed

  11.             if left_to_wait > 0:

  12.                 time.sleep(left_to_wait)

  13.             ret = func(*args, **kargs)

  14.             last_time_called[0] = time.perf_counter()

  15.             return ret

  16.         return rate_limited_function

  17.     return decorate

该装饰器的工作原理是:测量自上次函数调用以来所经过的时间,并在必要时等待适当的时间,以确保不超过速率限制。其中等待时间=min_interval - elapsed,这里min_intervalue是两次函数调用之间的最小时间间隔(以秒为单位),已用时间是自上次调用以来所用的时间。如果经过的时间小于最小间隔,则函数在再次执行之前等待left_to_wait秒。

⚠注意:该函数在调用之间引入了少量的时间开销,但确保不超过速率限制。

如果不想自己手动实现,可以用第三方包,名叫ratelimit。

pip install ratelimit

使用非常简单,只需要装饰被调用函数即可:

  1. from ratelimit import limits

  2. import requests

  3. FIFTEEN_MINUTES = 900

  4. @limits(calls=15, period=FIFTEEN_MINUTES)

  5. def call_api(url):

  6.     response = requests.get(url)

  7.     if response.status_code != 200:

  8.         raise Exception('API response: {}'.format(response.status_code))

  9.     return response

如果被装饰函数的调用次数超过允许次数,则会抛出ratelimit.RateLimitException异常。要处理该异常可以将@sleep_and_retry装饰器与@limits装饰器一起使用。

  1. @sleep_and_retry

  2. @limits(calls=15, period=FIFTEEN_MINUTES)

  3. def call_api(url):

  4.     response = requests.get(url)

  5.     if response.status_code != 200:

  6.         raise Exception('API response: {}'.format(response.status_code))

  7.     return response

这样被装饰函数在再次执行之前会休眠剩余时间。

09 @dataclass

Python 3.7 引入了@dataclass装饰器,将其加入到标准库,用于装饰类。它主要用于存储数据的类自动生成诸如__init__, __repr__, __eq__, __lt__,__str__ 等特殊函数。这样可以减少模板代码,并使类更加可读和可维护。

另外,@dataclass还提供了现成的美化方法,可以清晰地表示对象,将其转换为JSON格式,等等。

  1. from dataclasses import dataclass, 

  2. @dataclass

  3. class Person:

  4.     first_name: str

  5.     last_name: str

  6.     age: int

  7.     job: str

  8.     def __eq__(self, other):

  9.         if isinstance(other, Person):

  10.             return self.age == other.age

  11.         return NotImplemented

  12.     def __lt__(self, other):

  13.         if isinstance(other, Person):

  14.             return self.age < other.age

  15.         return NotImplemented

  16. john = Person(first_name="John", 

  17.               last_name="Doe", 

  18.               age=30, 

  19.               job="doctor",)

  20. anne = Person(first_name="Anne", 

  21.               last_name="Smith", 

  22.               age=40, 

  23.               job="software engineer",)

  24. print(john == anne)

  25. # False

  26. print(anne > john)

  27. # True

  28. asdict(anne)

  29. #{'first_name': 'Anne',

  30. # 'last_name': 'Smith',

  31. # 'age': 40,

  32. # 'job': 'software engineer'}

10 @register

如果你的Python脚本意外终止,但你仍想执行一些任务来保存你的工作、执行清理或打印消息,那么@register在这种情况下非常方便。

  1. from atexit import register

  2. @register

  3. def terminate():

  4.     perform_some_cleanup()

  5.     print("Goodbye!")

  6. while True:

  7.     print("Hello")

运行上面的代码会不断在控制台输出"Hello",点击Ctrl + C强制终止脚本运行,你会看到控制台输出"Goodbye",说明程序在中断后执行了@register装饰器装饰的terminate()函数。

11 @property

@property装饰器用于定义类属性,这些属性本质上是类实例属性的getter、setter和deleter方法。

通过使用@property装饰器,可以将方法定义为类属性,并将其作为类属性进行访问,而无需显式调用该方法。

如果您想在获取或设置值时添加一些约束和验证逻辑,使用@property装饰器会非常方便。

下面的示例中,我们在rating属性上定义了一个setter,对输入执行约束(介于0和5之间)。

  1. class Movie:

  2.     def __init__(self, r):

  3.         self._rating = r

  4.     @property

  5.     def rating(self):

  6.         return self._rating

  7.     @rating.setter

  8.     def rating(self, r):

  9.         if 0 <= r <= 5:

  10.             self._rating = r

  11.         else:

  12.             raise ValueError("The movie rating must be between 0 and 5!")

  13. batman = Movie(2.5)

  14. batman.rating

  15. # 2.5

  16. batman.rating = 4

  17. batman.rating

  18. # 4

  19. batman.rating = 10

  20. # ---------------------------------------------------------------------------

  21. # ValueError                                Traceback (most recent call last)

  22. # Input In [16], in <cell line: 1>()

  23. # ----> 1 batman.rating = 10

  24. # Input In [11], in Movie.rating(self, r)

  25. #      12     self._rating = r

  26. #      13 else:

  27. # ---> 14     raise ValueError("The movie rating must be between 0 and 5!")

  28. #

  29. # ValueError: The movie rating must be between 0 and 5!

12 @singledispatch

@singledispatch允许函数对不同类型的参数有不同的实现,有点像Java等面向对象语言中的函数重载。

  1. from functools import singledispatch

  2. @singledispatch

  3. def fun(arg):

  4.     print("Called with a single argument")

  5. @fun.register(int)

  6. def _(arg):

  7.     print("Called with an integer")

  8. @fun.register(list)

  9. def _(arg):

  10.     print("Called with a list")

  11. fun(1)  # Prints "Called with an integer"

  12. fun([1, 2, 3])  # Prints "Called with a list"

结论

装饰器是一个重要的抽象思想,可以在不改变原始代码的情况下扩展代码,如缓存、自动重试、速率限制、日志记录,或将类转换为超级数据容器等。

装饰器的功能远不止于此,本文介绍的12个常用装饰器只是抛砖引玉,当你理解了装饰器思想和用法后,可以发挥创造力,实现各种自定义装饰器来解决具体问题。

总结:

感谢每一个认真阅读我文章的人!!!

作为一位过来人也是希望大家少走一些弯路,如果你不想再体验一次学习时找不到资料,没人解答问题,坚持几天便放弃的感受的话,在这里我给大家分享一些自动化测试的学习资源,希望能给你前进的路上带来帮助。

软件测试面试文档

我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。

 

          视频文档获取方式:
这份文档和视频资料,对于想从事【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!以上均可以分享,点下方小卡片即可自行领取。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1679590.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

走进开源,拥抱开源

走进开源&#xff0c;拥抱开源 一、开源文化1.1 什么是开源1.2 为什么要开源1.3 有哪些开源协议 二、选择开源2.1 开源社区的类型与特点2.2 如何选择开源社区2.3 如何选择开源项目 三、参与开源3.1 开源社区的参与方式3.2 开源项目的参与方式 四、Apache Doris 参与示例4.1 Dor…

随笔:棋友们

我是在小学二年级学会中国象棋的&#xff0c;准确说&#xff0c;是学会象棋的下棋规则的&#xff0c;师傅是二舅。我最早的对手就是同学波仔。波仔比我略早学会象棋&#xff0c;总用连珠炮欺负我&#xff0c;开局几步棋就把我将死。我不知道怎么破解。轮到我先走时&#xff0c;…

降Compose十八掌之『亢龙有悔』

公众号「稀有猿诉」 原文链接 降Compose十八掌之『亢龙有悔』 Jetpack Compose是新一代的声明式的UI开发框架&#xff0c;由Google在2019年推出&#xff0c;最初是作为Android的新式UI开发框架&#xff0c;但它本质是一个声明式UI开发框架&#xff0c;并不受制于底层的平…

机器人非线性系统反馈线性化与解耦

机器人非线性系统的反馈线性化和解耦是控制理论中的两个重要概念&#xff0c;它们分别用于简化系统分析和设计过程&#xff0c;提高控制系统的性能。 首先&#xff0c;反馈线性化是一种将非线性系统转化为线性系统的技术。在机器人控制中&#xff0c;由于机器人本身是一个强耦…

每日一日 kotori和气球

kotori和气球 (nowcoder.com) 题目描述&#xff0c;就是只要相邻的气球不相同即可&#xff0c; 解题思路 使用高中的排列组合&#xff1a;第一个位置 可以填n种情况 其次后推不可与前一个相同所以可以 填n -1中情况&#xff0c;结果相乘即可 可以使用bigInteger实现 或者说…

[Kubernetes] kube-proxy 详解

文章目录 1.kube-proxy概述2.userspace模式3.iptables模式4.ipvs模式 1.kube-proxy概述 kube-proxy组件是用来实现service的请求转发&#xff0c;具体实现方式是kube-proxy运行在每个node上&#xff0c;通过watch监听API Server 中service资源的create&#xff0c;update&…

Spring 各版本发布时间与区别

版本版本特性Spring Framework 1.01. 所有代码都在一个项目中 2. 支持核心功能IoC、AOP 3. 内置支持Hibernate、iBatis等第三方框架 4. 对第三方技术简单封装。如&#xff1a;JDBC、Mail、事务等 5. 只支持XML配置方式。6.主要通过 XML 配置文件来管理对象和依赖关系&#xff0…

【2024华为HCIP831 | 高级网络工程师之路】刷题日记(18)

个人名片&#xff1a;&#x1faaa; &#x1f43c;作者简介&#xff1a;一名大三在校生&#xff0c;喜欢AI编程&#x1f38b; &#x1f43b;‍❄️个人主页&#x1f947;&#xff1a;落798. &#x1f43c;个人WeChat&#xff1a;hmmwx53 &#x1f54a;️系列专栏&#xff1a;&a…

Kubernetes进阶对象Deployment、DaemonSet、Service

Deployment Pod 在 YAML 里使用“containers”就可以任意编排容器&#xff0c;而且还有一个“restartPolicy”字段&#xff0c;默认值就是 Always&#xff0c;可以监控 Pod 里容器的状态&#xff0c;一旦发生异常&#xff0c;就会自动重启容器。 不过&#xff0c;“restartPo…

达梦(DM) SQL数据及字符串操作

达梦DM SQL数据及字符串操作 数据操作字符串操作 这里继续讲解DM数据库的操作&#xff0c;主要涉及插入、更新、删除操作。 数据操作 插入数据&#xff0c;不指定具体列的话就需要插入除自增列外的其他列&#xff0c;当然自增列也可以直接指定插入 INSERT INTO SYS_USER VALU…

2024最新Kali Linux安装教程(非常详细)从零基础入门到精通(附安装包)!

什么是Kali Linux&#xff1f; Kali Linux是一个高级渗透测试和安全审计Linux发行版&#xff0c;其功能非常强大&#xff0c;能够进行信息取证、渗透测试、攻击WPA / WPA2保护的无线网络、离线破解哈希密码、将android、Java、C编写的程序反编译成代码等等&#xff0c;是黑客的…

iOS ------ 多线程基础

一&#xff0c;进程和线程 1&#xff0c;进程 定义&#xff1a; 进程是指在系统中正在运行的一个应用程序每个进程之间是独立的&#xff0c;每个进程均运行在其专有的且受保护的内存进程是系统进行资源分配和调度的一个独立单位 补充&#xff1a;iOS系统是相对封闭的系统&a…

cdn引入vue的项目嵌入vue组件——http-vue-loader 的使用——技能提升

最近在写MVC的后台&#xff0c;看到全是jq的写法&#xff0c;但是对于用惯了vue的我&#xff0c;真是让我无从下手。。。 vue的双向绑定真的很好用。。。 为了能够在cdn引入的项目中嵌入vue组件&#xff0c;则可以使用http-vue-loader了 步骤1&#xff1a;下载http-vue-loader…

电子邮箱是什么?付费电子邮箱和免费电子邮箱有什么区别?

注册电子邮箱前&#xff0c;有付费电子邮箱和免费电子邮箱两类选择。付费的电子邮箱和免费的电子邮箱有什么区别呢&#xff1f;区别主要在于存储空间、功能丰富度和售后服务等方面&#xff0c;本文将为您详细介绍。 一、电子邮箱是什么&#xff1f; 电子邮箱就是线上的邮局&a…

详解绝对路径和相对路径的区别

绝对路径和相对路径是用于描述文件或目录在文件系统中位置的两种不同方式。 绝对路径&#xff08;Absolute Path&#xff09;是从文件系统的根目录开始的完整路径&#xff0c;可以唯一地确定一个文件或目录的位置。在不同的操作系统中&#xff0c;根目录的表示方式可能略有不同…

SQL注入漏洞常用绕过方法

SQL注入漏洞 漏洞描述 Web 程序代码中对于用户提交的参数未做过滤就直接放到 SQL 语句中执行&#xff0c;导致参数中的特殊字符打破了原有的SQL 语句逻辑&#xff0c;黑客可以利用该漏洞执行任意 SQL 语句&#xff0c;如查询数据、下载数据、写入webshell 、执行系统命令以及…

ADS FEM 仿真设置

1、EM Simulator 选择FEM。 2、在layout界面打开的EM功能&#xff0c;这里不需要操作。 3、Partitioning 不需要操作。 4、没有叠层的话需要新建&#xff0c;过孔可以在叠层处右键添加。 5、端口需要设置GND layer。 6、设置仿真频率。 7、Output plan。 8、Options 设置 介质…

【企业宣传片】拍摄思维提升,专业影视质感核心揭密,一课搞定

课程下载&#xff1a;【企业宣传片】拍摄-课程网盘链接提取码下载.txt资源-CSDN文库 更多资源下载&#xff1a;关注我。 课程介绍 大量案例分析宣传片拍摄的痛点要点 根据案例告诉你解决方案&#xff0c;讲透概念 改变你对企业宣传片的思维层级与认知 归纳总结对比不同案…

【C -> Cpp】由C迈向Cpp (6):静态、友元和内部类

标题&#xff1a;【C -&#xff1e; Cpp】由C迈向Cpp &#xff08;6&#xff09;&#xff1a;静态、友元和内部类 水墨不写bug &#xff08;图片来源于网络&#xff09; 目录 &#xff08;一&#xff09;静态成员 &#xff08;二&#xff09;友元 &#xff08;三&#xff09…

GDPU 竞赛技能实践 天码行空 期末小测

1. 除法&#xff08;原题&#xff09; &#x1f468;‍&#x1f3eb; 实验二&#xff1a;1.简单枚举 输入正整数n&#xff0c;按从小到大的顺序输出所有形如abcde/fghij n的表达式&#xff0c;其中a&#xff5e;j恰好为数字0&#xff5e;9的一个排列&#xff08;可以有前导0&a…