学会这12个Python装饰器,让你的代码更上一层楼
学会这12个Python装饰器,让你的代码更上一层楼
Python 装饰器是个强大的工具,可帮你生成整洁、可重用和可维护的代码。某种意义上说,会不会用装饰器是区分新手和老鸟的重要标志。如果你不熟悉装饰器,你可以将它们视为将函数作为输入并在不改变其主要用途的情况下扩展其功能的函数。装饰器可以有效提高你的工作效率并避免重复代码。本文我整理了项目中经常用到的 12 个装饰器,值得每一个Python开发者掌握。
文章目录
- 1. @logger
- 2. @wraps
- 3. @lru_cache
- 4. @repeat
- 5. @timeit
- 6. @retry
- 7. @countcall
- 8. @rate_limited
- 9. @dataclass
- 10. @register
- 11. @property
- 12. @singledispatch
- 结论
1. @logger
我们从最简单的装饰器开始,手动实现一个可以记录函数开始和结束的装饰器。被修饰函数的输出结果如下所示:
some_function(args)# ----- some_function: start -----
# some_function executing
# ----- some_function: end -----
要实现一个装饰器,首先要给装饰器起一个合适的名称:这里我们给装饰器起名为logger
。
装饰器本质上是一个函数,它将一个函数作为输入并返回一个函数作为输出。 输出函数通常是输入的扩展版。 在我们的例子中,我们希望输出函数用start
和end
语句包围输入函数的调用。
由于我们不知道输入函数都带有什么参数,我们可以使用 *args
和 **kwargs
从包装函数传递它们。*args
和 **kwargs
允许传递任意数量的位置参数和关键字参数。
下面是logger
装饰器的示例代码:
def logger(function):def wrapper(*args, **kwargs):print(f"----- {function.__name__}: start -----")output = function(*args, **kwargs)print(f"----- {function.__name__}: end -----")return outputreturn wrapper
logger
函数可以应用于任意函数,比如:
decorated_function = logger(some_function)
上面的语句是正确的,但Python 提供了更 Pythonic 的语法——使用 @
修饰符。因此更通常的写法是:
@logger
def some_function(text):print(text)some_function("first test")
# ----- some_function: start -----
# first test
# ----- some_function: end -----some_function("second test")
# ----- some_function: start -----
# second test
# ----- some_function: end -----
2. @wraps
此装饰器更新wrapper
函数,使其看起来像一个原始函数,并继承其名字和属性。
要了解 @wraps
的作用以及为什么需要它,让我们将前面写的logger
装饰器应用到一个将两个数字相加的简单函数中。
下面的代码是未使用@wraps
装饰器的版本:
def logger(function):def wrapper(*args, **kwargs):"""wrapper documentation"""print(f"----- {function.__name__}: start -----")output = function(*args, **kwargs)print(f"----- {function.__name__}: end -----")return outputreturn wrapper@logger
def add_two_numbers(a, b):"""this function adds two numbers"""return a + b
如果我们用__name__
和 __doc__
来查看被装饰函数add_two_numbers
的名称和文档,会得到如下结果:
add_two_numbers.__name__
'wrapper'add_two_numbers.__doc__
'wrapper documentation'
输出的是wrapper
函数的名称和文档。这是我们预期想要的结果,我们希望保留原始函数的名称和文档。这时@wraps
装饰器就派上用场了。
我们唯一需要做的就是给wrapper
函数加上@wraps
装饰器。
from functools import wrapsdef logger(function):@wraps(function)def wrapper(*args, **kwargs):"""wrapper documentation"""print(f"----- {function.__name__}: start -----")output = function(*args, **kwargs)print(f"----- {function.__name__}: end -----")return outputreturn wrapper@logger
def add_two_numbers(a, b):"""this function adds two numbers"""return a + b
再此检查add_two_numbers
函数的名称和文档,我们可以看到该函数的元数据。
add_two_numbers.__name__
# 'add_two_numbers'add_two_numbers.__doc__
# 'this function adds two numbers'
3. @lru_cache
@lru_cache
是Python内置装饰器,可以通过from functools import lru_cache
引入。@lru_cache
的作用是缓存函数的返回值,当缓存装满时,使用least-recently-used(LRU)算法丢弃最少使用的值。
@lru_cache
装饰器适合用于输入输出不变且运行时间较长的任务,例如查询数据库、请求静态页面或一些繁重的处理。
在下面的示例中,我使用@lru_cache
来修饰一个模拟某些处理的函数。然后连续多次对同一输入应用该函数。
import random
import time
from functools import lru_cache@lru_cache(maxsize=None)
def heavy_processing(n):sleep_time = n + random.random()time.sleep(sleep_time)# 初次调用
%%time
heavy_processing(0)
# CPU times: user 363 µs, sys: 727 µs, total: 1.09 ms
# Wall time: 694 ms# 第二次调用
%%time
heavy_processing(0)
# CPU times: user 4 µs, sys: 0 ns, total: 4 µs
# Wall time: 8.11 µs# 第三次调用
%%time
heavy_processing(0)
# CPU times: user 5 µs, sys: 1 µs, total: 6 µs
# Wall time: 7.15 µs
从上面的输出可以看到,第一次调用花费了694ms,因为执行了time.sleep()
函数。后面两次调用由于参数相同,直接返回缓存值,因此并没有实际执行函数内容,因此非常快地得到函数返回。
4. @repeat
该装饰器的所用是多次调用被修饰函数。这对于调试、压力测试或自动化多个重复任务非常有用。
跟前面的装饰器不同,@repeat
接受一个输入参数,
def repeat(number_of_times):def decorate(func):@wraps(func)def wrapper(*args, **kwargs):for _ in range(number_of_times):func(*args, **kwargs)return wrapperreturn decorate
上面的代码定义了一个名为repeat
的装饰器,有一个输入参数number_of_times
。与前面的案例不同,这里需要decorate
函数来传递被修饰函数。然后,装饰器定义一个名为wrapper
的函数来扩展被修饰函数。
@repeat(5)
def hello_world():print("hello world")hello_world()
# hello world
# hello world
# hello world
# hello world
# hello world
5. @timeit
该装饰器用来测量函数的执行时间并打印出来。这对调试和监控非常有用。
在下面的代码片段中,@timeit
装饰器测量process_data
函数的执行时间,并以秒为单位打印所用的时间。
import time
from functools import wrapsdef timeit(func):@wraps(func)def wrapper(*args, **kwargs):start = time.perf_counter()result = func(*args, **kwargs)end = time.perf_counter()print(f'{func.__name__} took {end - start:.6f} seconds to complete')return resultreturn wrapper@timeit
def process_data():time.sleep(1)process_data()
# process_data took 1.000012 seconds to complete
6. @retry
当函数遇到异常时,该装饰器会强制函数重试多次。它接受三个参数:重试次数、捕获的异常以及重试之间的间隔时间。
其工作原理如下:
wrapper
函数启动num_retrys
次迭代的for循环。- 将被修饰函数放到try/except块中。每次迭代如果调用成功,则中断循环并返回结果。否则,休眠
sleep_time
秒后继续下一次迭代。 - 当for循环结束后函数调用依然不成功,则抛出异常。
示例代码如下:
import random
import time
from functools import wrapsdef retry(num_retries, exception_to_check, sleep_time=0):"""遇到异常尝试重新执行装饰器"""def decorate(func):@wraps(func)def wrapper(*args, **kwargs):for i in range(1, num_retries+1):try:return func(*args, **kwargs)except exception_to_check as e:print(f"{func.__name__} raised {e.__class__.__name__}. Retrying...")if i < num_retries:time.sleep(sleep_time)# 尝试多次后仍不成功则抛出异常raise ereturn wrapperreturn decorate@retry(num_retries=3, exception_to_check=ValueError, sleep_time=1)
def random_value():value = random.randint(1, 5)if value == 3:raise ValueError("Value cannot be 3")return valuerandom_value()
# random_value raised ValueError. Retrying...
# 1random_value()
# 5
7. @countcall
@countcall
用于统计被修饰函数的调用次数。这里的调用次数会缓存在wraps
的count
属性中。
from functools import wrapsdef countcall(func):@wraps(func)def wrapper(*args, **kwargs):wrapper.count += 1result = func(*args, **kwargs)print(f'{func.__name__} has been called {wrapper.count} times')return resultwrapper.count = 0return wrapper@countcall
def process_data():passprocess_data()
process_data has been called 1 times
process_data()
process_data has been called 2 times
process_data()
process_data has been called 3 times
8. @rate_limited
@rate_limited
装饰器会在被修饰函数调用太频繁时,休眠一段时间,从而限制函数的调用速度。这在模拟、爬虫、接口调用防过载等场景下非常有用。
import time
from functools import wrapsdef rate_limited(max_per_second):min_interval = 1.0 / float(max_per_second)def decorate(func):last_time_called = [0.0]@wraps(func)def rate_limited_function(*args, **kargs):elapsed = time.perf_counter() - last_time_called[0]left_to_wait = min_interval - elapsedif left_to_wait > 0:time.sleep(left_to_wait)ret = func(*args, **kargs)last_time_called[0] = time.perf_counter()return retreturn rate_limited_functionreturn decorate
该装饰器的工作原理是:测量自上次函数调用以来所经过的时间,并在必要时等待适当的时间,以确保不超过速率限制。其中等待时间=min_interval - elapsed
,这里min_intervalue
是两次函数调用之间的最小时间间隔(以秒为单位),已用时间是自上次调用以来所用的时间。如果经过的时间小于最小间隔,则函数在再次执行之前等待left_to_wait
秒。
⚠注意:该函数在调用之间引入了少量的时间开销,但确保不超过速率限制。
如果不想自己手动实现,可以用第三方包,名叫ratelimit。
pip install ratelimit
使用非常简单,只需要装饰被调用函数即可:
from ratelimit import limitsimport requestsFIFTEEN_MINUTES = 900@limits(calls=15, period=FIFTEEN_MINUTES)
def call_api(url):response = requests.get(url)if response.status_code != 200:raise Exception('API response: {}'.format(response.status_code))return response
如果被装饰函数的调用次数超过允许次数,则会抛出ratelimit.RateLimitException
异常。要处理该异常可以将@sleep_and_retry
装饰器与@limits
装饰器一起使用。
@sleep_and_retry
@limits(calls=15, period=FIFTEEN_MINUTES)
def call_api(url):response = requests.get(url)if response.status_code != 200:raise Exception('API response: {}'.format(response.status_code))return response
这样被装饰函数在再次执行之前会休眠剩余时间。
9. @dataclass
Python 3.7 引入了@dataclass
装饰器,将其加入到标准库,用于装饰类。它主要用于存储数据的类自动生成诸如__init__
, __repr__
, __eq__
, __lt__
,__str__
等特殊函数。这样可以减少模板代码,并使类更加可读和可维护。
另外,@dataclass
还提供了现成的美化方法,可以清晰地表示对象,将其转换为JSON格式,等等。
from dataclasses import dataclass, @dataclass
class Person:first_name: strlast_name: strage: intjob: strdef __eq__(self, other):if isinstance(other, Person):return self.age == other.agereturn NotImplementeddef __lt__(self, other):if isinstance(other, Person):return self.age < other.agereturn NotImplementedjohn = Person(first_name="John", last_name="Doe", age=30, job="doctor",)anne = Person(first_name="Anne", last_name="Smith", age=40, job="software engineer",)print(john == anne)
# Falseprint(anne > john)
# Trueasdict(anne)
#{'first_name': 'Anne',
# 'last_name': 'Smith',
# 'age': 40,
# 'job': 'software engineer'}
10. @register
如果你的Python脚本意外终止,但你仍想执行一些任务来保存你的工作、执行清理或打印消息,那么@register
在这种情况下非常方便。
from atexit import register@register
def terminate():perform_some_cleanup()print("Goodbye!")while True:print("Hello")
运行上面的代码会不断在控制台输出"Hello",点击Ctrl + C
强制终止脚本运行,你会看到控制台输出"Goodbye",说明程序在中断后执行了@register
装饰器装饰的terminate()
函数。
11. @property
@property
装饰器用于定义类属性,这些属性本质上是类实例属性的getter
、setter
和deleter
方法。
通过使用@property
装饰器,可以将方法定义为类属性,并将其作为类属性进行访问,而无需显式调用该方法。
如果您想在获取或设置值时添加一些约束和验证逻辑,使用@property
装饰器会非常方便。
下面的示例中,我们在rating
属性上定义了一个setter
,对输入执行约束(介于0和5之间)。
class Movie:def __init__(self, r):self._rating = r@propertydef rating(self):return self._rating@rating.setterdef rating(self, r):if 0 <= r <= 5:self._rating = relse:raise ValueError("The movie rating must be between 0 and 5!")batman = Movie(2.5)
batman.rating
# 2.5batman.rating = 4
batman.rating
# 4batman.rating = 10# ---------------------------------------------------------------------------
# ValueError Traceback (most recent call last)
# Input In [16], in <cell line: 1>()
# ----> 1 batman.rating = 10
# Input In [11], in Movie.rating(self, r)
# 12 self._rating = r
# 13 else:
# ---> 14 raise ValueError("The movie rating must be between 0 and 5!")
#
# ValueError: The movie rating must be between 0 and 5!
12. @singledispatch
@singledispatch
允许函数对不同类型的参数有不同的实现,有点像Java等面向对象语言中的函数重载。
from functools import singledispatch@singledispatch
def fun(arg):print("Called with a single argument")@fun.register(int)
def _(arg):print("Called with an integer")@fun.register(list)
def _(arg):print("Called with a list")fun(1) # Prints "Called with an integer"
fun([1, 2, 3]) # Prints "Called with a list"
结论
装饰器是一个重要的抽象思想,可以在不改变原始代码的情况下扩展代码,如缓存、自动重试、速率限制、日志记录,或将类转换为超级数据容器等。
装饰器的功能远不止于此,本文介绍的12个常用装饰器只是抛砖引玉,当你理解了装饰器思想和用法后,可以发挥创造力,实现各种自定义装饰器来解决具体问题。
最后给大家推荐一个很棒的装饰器列表,里面记录了大量实用的、有趣的装饰器,大家可以多多尝试使用。
相关文章:
学会这12个Python装饰器,让你的代码更上一层楼
学会这12个Python装饰器,让你的代码更上一层楼 Python 装饰器是个强大的工具,可帮你生成整洁、可重用和可维护的代码。某种意义上说,会不会用装饰器是区分新手和老鸟的重要标志。如果你不熟悉装饰器,你可以将它们视为将函数作为输…...
企业使用ERP的好处
ERP系统是企业管理信息系统的简称,它是以信息技术为手段,以物流、资金流、信息流为主线,以企业的核心业务流程为对象,建立的一套适用于企业管理的、高效的企业管理信息系统。它是通过科学方法和计算机信息技术,将企业运…...
【QT】如何获取屏幕(桌面)的大小或分辨率
目录1. QDesktopWidget 获取系统屏幕大小2. QScreen 获取系统屏幕大小3. geometry() 与 availableGeometry() 的区别1. QDesktopWidget 获取系统屏幕大小 QDesktopWidget 提供了详细的位置信息,其能够自动返回窗口在用户窗口的位置和应用程序窗口的位置 QDesktopW…...
ETL工具的选择
正确选择 ETL 工具,可以从 ETL 对平台的支持、对数据源的支持、数据转换功能、管理 和调度功能、集成和开放性、对元数据管理等功能出发,具体如下。 支持平台 随着各种应用系统数据量的飞速增长和对业务可靠性等要求的不断提高,人们对数据抽…...
SpringBoot仿天猫商城java web购物网站的设计与实现
1,项目介绍 基于 SpringBoot 的仿天猫商城拥有两种角色,分别为管理员和用户。 迷你天猫商城是一个基于SSM框架的综合性B2C电商平台,需求设计主要参考天猫商城的购物流程。 后端页面兼容IE10及以上现代浏览器,Chrome,Edge,Firebox…...
C#基础教程22 文件的输入与输出
C# 文件的输入与输出 一个 文件 是一个存储在磁盘中带有指定名称和目录路径的数据集合。当打开文件进行读写时,它变成一个 流。 从根本上说,流是通过通信路径传递的字节序列。有两个主要的流:输入流 和 输出流。输入流用于从文件读取数据(读操作),输出流用于向文件写入数…...
Ubuntu18.04 python 开发usb通信
一、安装环境 1.安装pip sudo python3 get-pip.py 或 sudo -i apt update apt install python3-pip 确定pip是否安装成功: xxx-desktop:~$ pip3 --versionpip 9.0.1 from /usr/lib/python3/dist-packages (python 3.6)2.安装pyusb pip3 install pyusb --use…...
RabbitMq 消息确认机制详解 SpringCloud
1 消息可靠性 消息从发送,到消费者接收,会经理多个过程,其中的每一步都可能导致消息丢失. #### 2 常见的丢失原因 发送时丢失: 生产者发送的消息未送达exchange 消息到达exchange后未到达queueMQ宕机,queue将消息丢失 consumer…...
后台导航布局
五、后台导航实例 参考链接: 页面后台导航制作 如何实现html后台导航iframe点击换url(代码) 如何消除html页面下边和右边的滚动条 html页面有多个滚动条时的优化 页面出现不必要的滚动条,怎么调试? 一个页面有两…...
设计模式——抽象工厂模式(创建型)
一、介绍抽象工厂模式是一种创建型设计模式, 它能创建一系列相关的对象, 而无需指定其具体类。问题假设你正在开发一款家具商店模拟器。 你的代码中包括一些类, 用于表示:一系列相关产品, 例如 椅子Chair 、 沙发Sofa和…...
Java面试题--SpringMVC的执行流程
概要 SpringMVC是一种基于MVC(Model-View-Controller)框架的Web应用开发框架。下面是SpringMVC的详细执行流程。 客户端向DispatcherServlet发送请求。DispatcherServlet收到请求后,根据HandlerMapping(处理器映射)找…...
c# 32位程序突破2G内存限制
起因 在开发过程中,由于某些COM组件只能在32位程序下运行,程序不得不在X86平台下生成。而X86的32位程序默认内存大小被限制在2G。由于程序中可能存在大数量处理,期间对象若没有及时释放或则回收,内存占用达到了1.2G左右ÿ…...
【C语言】指针详解总结
指针1. 指针是什么2. 指针和指针类型2.1 指针-整数2.2 指针的解引用3. 野指针3.1 野指针成因3.2 如何规避野指针4. 指针运算4.1 指针-整数4.2 指针-指针4.3 指针的关系运算5. 指针和数组6. 二级指针7. 指针数组1. 指针是什么 指针是什么? 指针理解的2个要点…...
Java加解密(八)工具篇
目录Java加解密实用工具1 OpenSSL2 Keytool3 XCA4 KeyStore ExplorerJava加解密实用工具 1 OpenSSL OpenSSL是一个开放源代码的软件库包,应用程序可以使用这个包来进行安全通信,避免窃听,同时确认另一端连接者的身份。 例如Apache 使用它加…...
Go框架三件套(Web/RPC/ORM)
🧡 三件套介绍 Gorm Gorm 是一个已经迭代了10年的功能强大的 ORM 框架,在字节内部被广泛使用并且拥有非常丰富的开源扩展。 Kitex Kitex 是字节内部的 Golang 微服务 RPC 框架,具有高性能、强可扩展的主要特点,支持多协议并且拥有…...
HR问:假如公司给不到你期望的薪资怎么办?这个问题该如何体面地回答?
对大多数人而言,跳槽就是为了涨薪,工作就是为了挣钱。但如果面试时hr问:假如公司给不到你期望的薪资呢?面对这种问题,该怎么回答才体面?来看一波网友的机智回答:有人委婉拒绝,说“愿…...
LearnOpenGL-高级OpenGL-2.模板测试
本人刚学OpenGL不久且自学,文中定有代码、术语等错误,欢迎指正 我写的项目地址:https://github.com/liujianjie/LearnOpenGLProject 文章目录简单理解模板测试模板介绍模板函数物体轮廓介绍代码给加载的模型添加轮廓简单理解 同深度测试一样…...
【Git从入门到精通】Git入门
什么是版本控制 版本控制是一套系统,按时间记录某一个或一系列文件的变更,查看以前的特定版本。 使用版本控制系统,你可以将文件或者整个项目恢复到先前的状态,还可以对以前的文件进行对比。 本地版本控制系统 本地版本控制系…...
软件测试18
在桌面上打开终端窗口, 执行如下操作: 查看当前系统中开放的端口有哪些查看哪个程序正在使用 3306 端口(需要 root 用户权限) 注意: 1.某些端口号具备固定用途: 例如: 远程访问常用端口号:22 默认情况下是mysql使用的端口号&…...
C语言实现快速排序(hoare法、挖坑法、前后指针法与非递归实现)——不看后悔系列
目录 1. hoare法 方法与步骤 代码实现 2. 挖坑法 方法与步骤 代码实现 3. 前后指针法 方法与步骤 代码实现 4. 快速排序的缺点与优化 1.快速排序的缺点 2.快速排序的优化 ① 三数取中法选 key 代码实现 ② 小区间优化 代码实现 5. 快速排序的非递归实现 附录…...
如何为系统可靠性的量化提供依据
SLA 即 Service Level Agreement,也就是服务等级协议,它指的是系统服务提供者(Provider)对客户(Customer)的一个服务承诺。 而 SLO 就是 SLA 的具体目标管理办法,它由一系列相关的指标 SLI &am…...
量化投资中的因子是什么?因子是如何分类的,包括哪些?
因子就是对个股有解释的因素。因子的种类很多,不同类别的因子从不同的维度对个股收益进行解释。比如基本面因子的数据来源方面有很大一部分是财务报表,从估值、成长、盈利能力等多个方面对股票收益进行解释。量价因子是围绕价格、成交量等技术指标构建的…...
力扣-修复表中的名字
大家好,我是空空star,本篇带大家了解一道简单的力扣sql练习题。 文章目录前言一、题目:1667. 修复表中的名字二、解题1.正确示范①提交SQL运行结果2.正确示范②提交SQL运行结果3.正确示范③提交SQL运行结果4.正确示范④提交SQL运行结果5.其他…...
【博客633】linux vxlan设备工作原理
linux vxlan设备工作原理 vxlan处理包的原理:以k8s cni flannel组件创建的vxlan设备为例 1、k8s cni组件创建flannel设备flannel.1,且这个设备为vxlan类型的设备 root10.10.10.12:/home/ubuntu# ethtool -i flannel.1 driver: vxlan version: 0.1 fi…...
3.12学习周报
文章目录前言文献阅读摘要简介方法介绍讨论结论相关性分析总结前言 本周阅读文献《Streamflow and rainfall forecasting by two long short-term memory-based models》,文献主要提出两种基于长短时记忆网络的混合模型用于对水流量和降雨量进行预测。小波-LSTM&am…...
电力电子中逐波限流控制以及dsp实现
逐波限流是指在电力系统运行中,对电力设备进行电流保护的一种措施。它的实现方式是通过对电力系统的电流进行逐波监测和控制,每一波电流都可以独立地进行限制,从而保护电力系统设备不受过载损坏或短路故障的影响。 逐波限流的作用是提高电力…...
【数据结构】 顺序表
文章目录1 线性表2 顺序表2.1 概念及结构2.2 接口实现2.3 数组相关面试题2.4 顺序表的问题与思考1 线性表 线性表(linear list)是n个具有相同特性的数据元素的有限序列。线性表是一种在实际中广泛使用的数据结构,常见的线性表:顺序…...
Elasticsearch 集群规划- 单台机器核心数计算公式
在做集群规划的时候,到底需要给集群的每个节点多少个核心数?这个问题一直困扰了我很久。最近一段时间做千亿数据,PB存储量集群规划的时候,突然想明白了这件事,大致可以用一个公式来计算!我觉得这是一个非常…...
Tesla都使用什么编程语言?
作者 | 初光 出品 | 车端 备注 | 转载请阅读文中版权声明 知圈 | 进“汽车电子与AutoSAR开发”群,请加微“cloud2sunshine” 总目录链接>> AutoSAR入门和实战系列总目录 带着对更美好未来的愿景,特斯拉不仅成为有史以来最有价值的汽车公司&…...
1143. 最长公共子序列——【Leetcode每日刷题】
1143. 最长公共子序列 给定两个字符串 text1 和 text2,返回这两个字符串的最长 公共子序列 的长度。如果不存在 公共子序列 ,返回 0 。 一个字符串的 子序列 是指这样一个新的字符串:它是由原字符串在不改变字符的相对顺序的情况下删除某些…...
个人可以做导购网站吗/百度浏览器官网
为什么mysql要使用主从模型发布时间:2020-05-09 10:38:05来源:亿速云阅读:172作者:三月本文主要给大家介绍为什么mysql要使用主从模型,文章内容都是笔者用心摘选和编辑的,具有一定的针对性,对大…...
wordpress css3/有什么平台可以推广
//PS2键盘测试程序,可换行,按shift不放接着输入//可输出大写,按下CAPS输出大写,再次按下输出小写//此程序只用来测试,代码冗余,仅供参考,可根据需要自行删减//PA13->PS2.CLK PA15->PS2.D…...
株洲网站建设/淘宝数据分析工具
首先引用ICSharpCode.SharpZipLib.dll,没有在这里下载:http://files.cnblogs.com/KenBlove/ICSharpCode.SharpZipLib.rar 压缩打包代码 /// <summary>/// 生成压缩文件/// </summary>/// <param name"strZipPath">生成的zip文件的路径</param&g…...
网站制作过程合理步骤是什么/口碑营销案例及分析
1. 关联 首先,先将多级列表与各个标题相关联。 开始→\rightarrow→多级列表→\rightarrow→定义新的多级列表 设置一级标题 设置二级标题 同理,设置三级标题 2. 设置各标题格式 2.1 显示三级标题 附: 上述设置成功之后,样式栏…...
劳务合同免费模板下载/广州seo运营
四周静悄悄的,只有墙上的钟表在喳喳喳地响个不停,似乎只有它没有疲倦的感觉。手头的山东一级OFFICE开发模块估计做了一半多了,今晚终于又克服了一个难题。终于可以停下来休息了。四周静悄悄的,我的心却显得尤为激动......记得一个…...
吴江城乡住房和城乡建设局网站/拼多多关键词排名查询
document.addEventListener(blur,function(){document.body.scrollIntoView(false)},true) })复制代码...