pyxllib 0.3.197__py3-none-any.whl → 0.3.200__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- pyxllib/__init__.py +21 -21
- pyxllib/algo/__init__.py +8 -8
- pyxllib/algo/disjoint.py +54 -54
- pyxllib/algo/geo.py +541 -541
- pyxllib/algo/intervals.py +964 -964
- pyxllib/algo/matcher.py +389 -389
- pyxllib/algo/newbie.py +166 -166
- pyxllib/algo/pupil.py +629 -629
- pyxllib/algo/shapelylib.py +67 -67
- pyxllib/algo/specialist.py +241 -241
- pyxllib/algo/stat.py +494 -494
- pyxllib/algo/treelib.py +149 -149
- pyxllib/algo/unitlib.py +66 -66
- pyxllib/autogui/__init__.py +5 -5
- pyxllib/autogui/activewin.py +246 -246
- pyxllib/autogui/all.py +9 -9
- pyxllib/autogui/autogui.py +852 -852
- pyxllib/autogui/uiautolib.py +362 -362
- pyxllib/autogui/virtualkey.py +102 -102
- pyxllib/autogui/wechat.py +827 -827
- pyxllib/autogui/wechat_msg.py +421 -421
- pyxllib/autogui/wxautolib.py +84 -84
- pyxllib/cv/__init__.py +5 -5
- pyxllib/cv/expert.py +267 -267
- pyxllib/cv/imfile.py +159 -159
- pyxllib/cv/imhash.py +39 -39
- pyxllib/cv/pupil.py +9 -9
- pyxllib/cv/rgbfmt.py +1525 -1525
- pyxllib/cv/slidercaptcha.py +137 -137
- pyxllib/cv/trackbartools.py +251 -251
- pyxllib/cv/xlcvlib.py +1040 -1040
- pyxllib/cv/xlpillib.py +423 -423
- pyxllib/data/echarts.py +240 -240
- pyxllib/data/jsonlib.py +89 -89
- pyxllib/data/oss.py +72 -72
- pyxllib/data/pglib.py +1127 -1127
- pyxllib/data/sqlite.py +568 -568
- pyxllib/data/sqllib.py +297 -297
- pyxllib/ext/JLineViewer.py +505 -505
- pyxllib/ext/__init__.py +6 -6
- pyxllib/ext/demolib.py +246 -246
- pyxllib/ext/drissionlib.py +277 -277
- pyxllib/ext/kq5034lib.py +12 -12
- pyxllib/ext/old.py +663 -663
- pyxllib/ext/qt.py +449 -449
- pyxllib/ext/robustprocfile.py +497 -497
- pyxllib/ext/seleniumlib.py +76 -76
- pyxllib/ext/tk.py +173 -173
- pyxllib/ext/unixlib.py +827 -827
- pyxllib/ext/utools.py +351 -351
- pyxllib/ext/webhook.py +124 -119
- pyxllib/ext/win32lib.py +40 -40
- pyxllib/ext/wjxlib.py +88 -88
- pyxllib/ext/wpsapi.py +124 -124
- pyxllib/ext/xlwork.py +9 -9
- pyxllib/ext/yuquelib.py +1105 -1105
- pyxllib/file/__init__.py +17 -17
- pyxllib/file/docxlib.py +761 -761
- pyxllib/file/gitlib.py +309 -309
- pyxllib/file/libreoffice.py +165 -165
- pyxllib/file/movielib.py +148 -148
- pyxllib/file/newbie.py +10 -10
- pyxllib/file/onenotelib.py +1469 -1469
- pyxllib/file/packlib/__init__.py +330 -330
- pyxllib/file/packlib/zipfile.py +2441 -2441
- pyxllib/file/pdflib.py +426 -426
- pyxllib/file/pupil.py +185 -185
- pyxllib/file/specialist/__init__.py +685 -685
- pyxllib/file/specialist/dirlib.py +799 -799
- pyxllib/file/specialist/download.py +193 -193
- pyxllib/file/specialist/filelib.py +2829 -2829
- pyxllib/file/xlsxlib.py +3131 -3131
- pyxllib/file/xlsyncfile.py +341 -341
- pyxllib/prog/__init__.py +5 -5
- pyxllib/prog/cachetools.py +64 -64
- pyxllib/prog/deprecatedlib.py +233 -233
- pyxllib/prog/filelock.py +42 -42
- pyxllib/prog/ipyexec.py +253 -253
- pyxllib/prog/multiprogs.py +940 -940
- pyxllib/prog/newbie.py +451 -451
- pyxllib/prog/pupil.py +1197 -1197
- pyxllib/prog/sitepackages.py +33 -33
- pyxllib/prog/specialist/__init__.py +391 -391
- pyxllib/prog/specialist/bc.py +203 -203
- pyxllib/prog/specialist/browser.py +497 -497
- pyxllib/prog/specialist/common.py +347 -347
- pyxllib/prog/specialist/datetime.py +198 -198
- pyxllib/prog/specialist/tictoc.py +240 -240
- pyxllib/prog/specialist/xllog.py +180 -180
- pyxllib/prog/xlosenv.py +108 -108
- pyxllib/stdlib/__init__.py +17 -17
- pyxllib/stdlib/tablepyxl/__init__.py +10 -10
- pyxllib/stdlib/tablepyxl/style.py +303 -303
- pyxllib/stdlib/tablepyxl/tablepyxl.py +130 -130
- pyxllib/text/__init__.py +8 -8
- pyxllib/text/ahocorasick.py +39 -39
- pyxllib/text/airscript.js +744 -744
- pyxllib/text/charclasslib.py +121 -121
- pyxllib/text/jiebalib.py +267 -267
- pyxllib/text/jinjalib.py +32 -32
- pyxllib/text/jsa_ai_prompt.md +271 -271
- pyxllib/text/jscode.py +922 -922
- pyxllib/text/latex/__init__.py +158 -158
- pyxllib/text/levenshtein.py +303 -303
- pyxllib/text/nestenv.py +1215 -1215
- pyxllib/text/newbie.py +300 -300
- pyxllib/text/pupil/__init__.py +8 -8
- pyxllib/text/pupil/common.py +1121 -1121
- pyxllib/text/pupil/xlalign.py +326 -326
- pyxllib/text/pycode.py +47 -47
- pyxllib/text/specialist/__init__.py +8 -8
- pyxllib/text/specialist/common.py +112 -112
- pyxllib/text/specialist/ptag.py +186 -186
- pyxllib/text/spellchecker.py +172 -172
- pyxllib/text/templates/echart_base.html +10 -10
- pyxllib/text/templates/highlight_code.html +16 -16
- pyxllib/text/templates/latex_editor.html +102 -102
- pyxllib/text/vbacode.py +17 -17
- pyxllib/text/xmllib.py +747 -747
- pyxllib/xl.py +42 -39
- pyxllib/xlcv.py +17 -17
- {pyxllib-0.3.197.dist-info → pyxllib-0.3.200.dist-info}/METADATA +1 -1
- pyxllib-0.3.200.dist-info/RECORD +126 -0
- {pyxllib-0.3.197.dist-info → pyxllib-0.3.200.dist-info}/licenses/LICENSE +190 -190
- pyxllib-0.3.197.dist-info/RECORD +0 -126
- {pyxllib-0.3.197.dist-info → pyxllib-0.3.200.dist-info}/WHEEL +0 -0
@@ -1,240 +1,240 @@
|
|
1
|
-
#!/usr/bin/env python3
|
2
|
-
# -*- coding: utf-8 -*-
|
3
|
-
# @Author : 陈坤泽
|
4
|
-
# @Email : 877362867@qq.com
|
5
|
-
# @Date : 2020/09/20
|
6
|
-
|
7
|
-
|
8
|
-
import time
|
9
|
-
import timeit
|
10
|
-
|
11
|
-
from loguru import logger
|
12
|
-
from humanfriendly import format_timespan
|
13
|
-
|
14
|
-
from pyxllib.algo.pupil import natural_sort, ValuesStat
|
15
|
-
from pyxllib.text.pupil import shorten, listalign
|
16
|
-
from pyxllib.prog.newbie import human_readable_number
|
17
|
-
|
18
|
-
__tictoc = """
|
19
|
-
基于 pytictoc 代码,做了些自定义扩展
|
20
|
-
|
21
|
-
原版备注:
|
22
|
-
Module with class TicToc to replicate the functionality of MATLAB's tic and toc.
|
23
|
-
Documentation: https://pypi.python.org/pypi/pytictoc
|
24
|
-
__author__ = 'Eric Fields'
|
25
|
-
__version__ = '1.4.0'
|
26
|
-
__version_date__ = '29 April 2017'
|
27
|
-
"""
|
28
|
-
|
29
|
-
|
30
|
-
class TicToc:
|
31
|
-
"""
|
32
|
-
Replicate the functionality of MATLAB's tic and toc.
|
33
|
-
|
34
|
-
#Methods
|
35
|
-
TicToc.tic() #start or re-start the timer
|
36
|
-
TicToc.toc() #print elapsed time since timer start
|
37
|
-
TicToc.tocvalue() #return floating point value of elapsed time since timer start
|
38
|
-
|
39
|
-
#Attributes
|
40
|
-
TicToc.start #Time from timeit.default_timer() when t.tic() was last called
|
41
|
-
TicToc.end #Time from timeit.default_timer() when t.toc() or t.tocvalue() was last called
|
42
|
-
TicToc.elapsed #t.end - t.start; i.e., time elapsed from t.start when t.toc() or t.tocvalue() was last called
|
43
|
-
"""
|
44
|
-
|
45
|
-
def __init__(self, title='', *, disable=False, min_display_seconds=None):
|
46
|
-
"""Create instance of TicToc class."""
|
47
|
-
self.start = timeit.default_timer()
|
48
|
-
self.end = float('nan')
|
49
|
-
self.elapsed = float('nan')
|
50
|
-
self.title = title
|
51
|
-
self.disable = disable
|
52
|
-
# 只有达到这个时间,才会显示耗时
|
53
|
-
self.min_display_seconds = min_display_seconds
|
54
|
-
|
55
|
-
def tic(self):
|
56
|
-
"""Start the timer."""
|
57
|
-
self.start = timeit.default_timer()
|
58
|
-
|
59
|
-
def toc(self, msg='', restart=False):
|
60
|
-
"""
|
61
|
-
Report time elapsed since last call to tic().
|
62
|
-
|
63
|
-
Optional arguments:
|
64
|
-
msg - String to replace default message of 'Elapsed time is'
|
65
|
-
restart - Boolean specifying whether to restart the timer
|
66
|
-
"""
|
67
|
-
self.end = timeit.default_timer()
|
68
|
-
self.elapsed = self.end - self.start
|
69
|
-
if not self.disable:
|
70
|
-
# print(f'{self.title} {msg} {self.elapsed:.3f} 秒.')
|
71
|
-
print(f'{self.title} {msg} elapsed {human_readable_number(self.elapsed, "秒")}.')
|
72
|
-
if restart:
|
73
|
-
self.start = timeit.default_timer()
|
74
|
-
|
75
|
-
def tocvalue(self, restart=False):
|
76
|
-
"""
|
77
|
-
Return time elapsed seconds since last call to tic().
|
78
|
-
|
79
|
-
Optional argument:
|
80
|
-
restart - Boolean specifying whether to restart the timer
|
81
|
-
"""
|
82
|
-
self.end = timeit.default_timer()
|
83
|
-
self.elapsed = self.end - self.start
|
84
|
-
if restart:
|
85
|
-
self.start = timeit.default_timer()
|
86
|
-
return self.elapsed
|
87
|
-
|
88
|
-
@staticmethod
|
89
|
-
def process_time(msg='time.process_time():'):
|
90
|
-
"""计算从python程序启动到目前为止总用时"""
|
91
|
-
print(f'{msg} {human_readable_number(time.process_time(), "秒")}.')
|
92
|
-
|
93
|
-
def __enter__(self):
|
94
|
-
"""Start the timer when using TicToc in a context manager."""
|
95
|
-
if self.title == '__main__' and not self.disable:
|
96
|
-
logger.info(f'time.process_time(): {human_readable_number(time.process_time(), "秒")}.')
|
97
|
-
self.start = timeit.default_timer()
|
98
|
-
|
99
|
-
def __exit__(self, exc_type, exc_val, exc_tb):
|
100
|
-
"""On exit, print time elapsed since entering context manager."""
|
101
|
-
elapsed = self.tocvalue()
|
102
|
-
|
103
|
-
if exc_tb is None:
|
104
|
-
if not self.disable and (self.min_display_seconds is None or elapsed >= self.min_display_seconds):
|
105
|
-
logger.info(f'{self.title} finished in {human_readable_number(elapsed, "秒")}.')
|
106
|
-
else:
|
107
|
-
logger.info(f'{self.title} interrupt in {human_readable_number(elapsed, "秒")},')
|
108
|
-
|
109
|
-
|
110
|
-
__timer = """
|
111
|
-
|
112
|
-
"""
|
113
|
-
|
114
|
-
|
115
|
-
class Timer:
|
116
|
-
"""分析性能用的计时器类,支持with语法调用
|
117
|
-
必须显示地指明每一轮的start()和end(),否则会报错
|
118
|
-
"""
|
119
|
-
|
120
|
-
def __init__(self, title=''):
|
121
|
-
"""
|
122
|
-
:param title: 计时器名称
|
123
|
-
"""
|
124
|
-
# 不同的平台应该使用的计时器不同,这个直接用timeit中的配置最好
|
125
|
-
self.default_timer = timeit.default_timer
|
126
|
-
# 标题
|
127
|
-
self.title = title
|
128
|
-
self.data = []
|
129
|
-
self.start_clock = float('nan')
|
130
|
-
|
131
|
-
def start(self):
|
132
|
-
self.start_clock = self.default_timer()
|
133
|
-
|
134
|
-
def stop(self):
|
135
|
-
self.data.append(self.default_timer() - self.start_clock)
|
136
|
-
|
137
|
-
def report(self, msg=''):
|
138
|
-
""" 报告目前性能统计情况
|
139
|
-
"""
|
140
|
-
msg = f'{self.title} {msg}'
|
141
|
-
n = len(self.data)
|
142
|
-
|
143
|
-
if n >= 1:
|
144
|
-
print(msg, '用时(秒) ' + ValuesStat(self.data).summary(valfmt='.3f'))
|
145
|
-
elif n == 1:
|
146
|
-
sum_ = sum(self.data)
|
147
|
-
print(f'{msg} 用时: {sum_:.3f}s')
|
148
|
-
else: # 没有统计数据,则补充执行一次stop后汇报
|
149
|
-
print(f'{msg} 暂无计时信息')
|
150
|
-
|
151
|
-
def __enter__(self):
|
152
|
-
return self
|
153
|
-
|
154
|
-
def __exit__(self, *args):
|
155
|
-
self.report()
|
156
|
-
|
157
|
-
|
158
|
-
def performit(title, stmt="pass", setup="pass", repeat=1, number=1, globals=None):
|
159
|
-
""" 在timeit.repeat的基础上,做了层封装
|
160
|
-
|
161
|
-
200920周日15:33,简化函数,该函数不再获得执行结果,避免重复运行
|
162
|
-
|
163
|
-
:param title: 测试标题、名称功能
|
164
|
-
:return: 返回原函数单次执行结果
|
165
|
-
"""
|
166
|
-
data = timeit.repeat(stmt=stmt, setup=setup, repeat=repeat, number=number, globals=globals)
|
167
|
-
print(title, '用时(秒) ' + ValuesStat(data).summary(valfmt='.3f'))
|
168
|
-
return data
|
169
|
-
|
170
|
-
|
171
|
-
def perftest(title, stmt="pass", repeat=1, number=1, globals=None, res_width=None, print_=True):
|
172
|
-
""" 与performit的区别是,自己手动循环,记录程序运行结果
|
173
|
-
|
174
|
-
:param title: 测试标题、名称功能
|
175
|
-
:param res_width: 运行结果内容展示的字符上限数
|
176
|
-
:param print_: 输出报告
|
177
|
-
:return: 返回原函数单次执行结果
|
178
|
-
|
179
|
-
这里为了同时获得表达式返回值,就没有用标注你的timeit.repeat实现了
|
180
|
-
"""
|
181
|
-
# 1 确保stmt是可调用对象
|
182
|
-
if callable(stmt):
|
183
|
-
func = stmt
|
184
|
-
else:
|
185
|
-
code = compile(stmt, '', 'eval')
|
186
|
-
|
187
|
-
def func():
|
188
|
-
return eval(code, globals)
|
189
|
-
|
190
|
-
# 2 原函数运行结果(这里要先重载stdout)
|
191
|
-
data = []
|
192
|
-
res = ''
|
193
|
-
for i in range(repeat):
|
194
|
-
start = time.time()
|
195
|
-
for j in range(number):
|
196
|
-
res = func()
|
197
|
-
data.append(time.time() - start)
|
198
|
-
|
199
|
-
# 3 报告格式
|
200
|
-
if res_width is None:
|
201
|
-
# 如果性能报告比较短,只有一次测试,那res_width默认长度可以高一点
|
202
|
-
res_width = 50 if len(data) > 1 else 200
|
203
|
-
if res is None:
|
204
|
-
res = ''
|
205
|
-
else:
|
206
|
-
res = '运行结果:' + shorten(str(res), res_width)
|
207
|
-
if print_:
|
208
|
-
print(title, '用时(秒) ' + ValuesStat(data).summary(valfmt='.3f'), res)
|
209
|
-
|
210
|
-
return data
|
211
|
-
|
212
|
-
|
213
|
-
class PerfTest:
|
214
|
-
""" 这里模仿了unittest的机制
|
215
|
-
|
216
|
-
v0.0.38 重要改动,将number等参数移到perf操作,而不是类初始化中操作,继承使用上会更简单
|
217
|
-
"""
|
218
|
-
|
219
|
-
def perf(self, number=1, repeat=1, globals=None):
|
220
|
-
"""
|
221
|
-
|
222
|
-
:param number: 有些代码运算过快,可以多次运行迭代为一个单元
|
223
|
-
:param repeat: 对单元重复执行次数,最后会计算平均值、标准差
|
224
|
-
关于number和repeat的区别:
|
225
|
-
number张纸量repeat次
|
226
|
-
如果是纸箱这么厚的纸,number可以不设,默认是1
|
227
|
-
"""
|
228
|
-
# 1 找到所有perf_为前缀,且callable的函数方法
|
229
|
-
funcnames = []
|
230
|
-
for k in dir(self):
|
231
|
-
if k.startswith('perf_'):
|
232
|
-
if callable(getattr(self, k)):
|
233
|
-
funcnames.append(k)
|
234
|
-
|
235
|
-
# 2 自然排序
|
236
|
-
funcnames = natural_sort(funcnames)
|
237
|
-
funcnames2 = listalign([fn[5:] for fn in funcnames], 'r')
|
238
|
-
for i, funcname in enumerate(funcnames):
|
239
|
-
perftest(funcnames2[i], getattr(self, funcname),
|
240
|
-
number=int(number), repeat=int(repeat), globals=globals)
|
1
|
+
#!/usr/bin/env python3
|
2
|
+
# -*- coding: utf-8 -*-
|
3
|
+
# @Author : 陈坤泽
|
4
|
+
# @Email : 877362867@qq.com
|
5
|
+
# @Date : 2020/09/20
|
6
|
+
|
7
|
+
|
8
|
+
import time
|
9
|
+
import timeit
|
10
|
+
|
11
|
+
from loguru import logger
|
12
|
+
from humanfriendly import format_timespan
|
13
|
+
|
14
|
+
from pyxllib.algo.pupil import natural_sort, ValuesStat
|
15
|
+
from pyxllib.text.pupil import shorten, listalign
|
16
|
+
from pyxllib.prog.newbie import human_readable_number
|
17
|
+
|
18
|
+
__tictoc = """
|
19
|
+
基于 pytictoc 代码,做了些自定义扩展
|
20
|
+
|
21
|
+
原版备注:
|
22
|
+
Module with class TicToc to replicate the functionality of MATLAB's tic and toc.
|
23
|
+
Documentation: https://pypi.python.org/pypi/pytictoc
|
24
|
+
__author__ = 'Eric Fields'
|
25
|
+
__version__ = '1.4.0'
|
26
|
+
__version_date__ = '29 April 2017'
|
27
|
+
"""
|
28
|
+
|
29
|
+
|
30
|
+
class TicToc:
|
31
|
+
"""
|
32
|
+
Replicate the functionality of MATLAB's tic and toc.
|
33
|
+
|
34
|
+
#Methods
|
35
|
+
TicToc.tic() #start or re-start the timer
|
36
|
+
TicToc.toc() #print elapsed time since timer start
|
37
|
+
TicToc.tocvalue() #return floating point value of elapsed time since timer start
|
38
|
+
|
39
|
+
#Attributes
|
40
|
+
TicToc.start #Time from timeit.default_timer() when t.tic() was last called
|
41
|
+
TicToc.end #Time from timeit.default_timer() when t.toc() or t.tocvalue() was last called
|
42
|
+
TicToc.elapsed #t.end - t.start; i.e., time elapsed from t.start when t.toc() or t.tocvalue() was last called
|
43
|
+
"""
|
44
|
+
|
45
|
+
def __init__(self, title='', *, disable=False, min_display_seconds=None):
|
46
|
+
"""Create instance of TicToc class."""
|
47
|
+
self.start = timeit.default_timer()
|
48
|
+
self.end = float('nan')
|
49
|
+
self.elapsed = float('nan')
|
50
|
+
self.title = title
|
51
|
+
self.disable = disable
|
52
|
+
# 只有达到这个时间,才会显示耗时
|
53
|
+
self.min_display_seconds = min_display_seconds
|
54
|
+
|
55
|
+
def tic(self):
|
56
|
+
"""Start the timer."""
|
57
|
+
self.start = timeit.default_timer()
|
58
|
+
|
59
|
+
def toc(self, msg='', restart=False):
|
60
|
+
"""
|
61
|
+
Report time elapsed since last call to tic().
|
62
|
+
|
63
|
+
Optional arguments:
|
64
|
+
msg - String to replace default message of 'Elapsed time is'
|
65
|
+
restart - Boolean specifying whether to restart the timer
|
66
|
+
"""
|
67
|
+
self.end = timeit.default_timer()
|
68
|
+
self.elapsed = self.end - self.start
|
69
|
+
if not self.disable:
|
70
|
+
# print(f'{self.title} {msg} {self.elapsed:.3f} 秒.')
|
71
|
+
print(f'{self.title} {msg} elapsed {human_readable_number(self.elapsed, "秒")}.')
|
72
|
+
if restart:
|
73
|
+
self.start = timeit.default_timer()
|
74
|
+
|
75
|
+
def tocvalue(self, restart=False):
|
76
|
+
"""
|
77
|
+
Return time elapsed seconds since last call to tic().
|
78
|
+
|
79
|
+
Optional argument:
|
80
|
+
restart - Boolean specifying whether to restart the timer
|
81
|
+
"""
|
82
|
+
self.end = timeit.default_timer()
|
83
|
+
self.elapsed = self.end - self.start
|
84
|
+
if restart:
|
85
|
+
self.start = timeit.default_timer()
|
86
|
+
return self.elapsed
|
87
|
+
|
88
|
+
@staticmethod
|
89
|
+
def process_time(msg='time.process_time():'):
|
90
|
+
"""计算从python程序启动到目前为止总用时"""
|
91
|
+
print(f'{msg} {human_readable_number(time.process_time(), "秒")}.')
|
92
|
+
|
93
|
+
def __enter__(self):
|
94
|
+
"""Start the timer when using TicToc in a context manager."""
|
95
|
+
if self.title == '__main__' and not self.disable:
|
96
|
+
logger.info(f'time.process_time(): {human_readable_number(time.process_time(), "秒")}.')
|
97
|
+
self.start = timeit.default_timer()
|
98
|
+
|
99
|
+
def __exit__(self, exc_type, exc_val, exc_tb):
|
100
|
+
"""On exit, print time elapsed since entering context manager."""
|
101
|
+
elapsed = self.tocvalue()
|
102
|
+
|
103
|
+
if exc_tb is None:
|
104
|
+
if not self.disable and (self.min_display_seconds is None or elapsed >= self.min_display_seconds):
|
105
|
+
logger.info(f'{self.title} finished in {human_readable_number(elapsed, "秒")}.')
|
106
|
+
else:
|
107
|
+
logger.info(f'{self.title} interrupt in {human_readable_number(elapsed, "秒")},')
|
108
|
+
|
109
|
+
|
110
|
+
__timer = """
|
111
|
+
|
112
|
+
"""
|
113
|
+
|
114
|
+
|
115
|
+
class Timer:
|
116
|
+
"""分析性能用的计时器类,支持with语法调用
|
117
|
+
必须显示地指明每一轮的start()和end(),否则会报错
|
118
|
+
"""
|
119
|
+
|
120
|
+
def __init__(self, title=''):
|
121
|
+
"""
|
122
|
+
:param title: 计时器名称
|
123
|
+
"""
|
124
|
+
# 不同的平台应该使用的计时器不同,这个直接用timeit中的配置最好
|
125
|
+
self.default_timer = timeit.default_timer
|
126
|
+
# 标题
|
127
|
+
self.title = title
|
128
|
+
self.data = []
|
129
|
+
self.start_clock = float('nan')
|
130
|
+
|
131
|
+
def start(self):
|
132
|
+
self.start_clock = self.default_timer()
|
133
|
+
|
134
|
+
def stop(self):
|
135
|
+
self.data.append(self.default_timer() - self.start_clock)
|
136
|
+
|
137
|
+
def report(self, msg=''):
|
138
|
+
""" 报告目前性能统计情况
|
139
|
+
"""
|
140
|
+
msg = f'{self.title} {msg}'
|
141
|
+
n = len(self.data)
|
142
|
+
|
143
|
+
if n >= 1:
|
144
|
+
print(msg, '用时(秒) ' + ValuesStat(self.data).summary(valfmt='.3f'))
|
145
|
+
elif n == 1:
|
146
|
+
sum_ = sum(self.data)
|
147
|
+
print(f'{msg} 用时: {sum_:.3f}s')
|
148
|
+
else: # 没有统计数据,则补充执行一次stop后汇报
|
149
|
+
print(f'{msg} 暂无计时信息')
|
150
|
+
|
151
|
+
def __enter__(self):
|
152
|
+
return self
|
153
|
+
|
154
|
+
def __exit__(self, *args):
|
155
|
+
self.report()
|
156
|
+
|
157
|
+
|
158
|
+
def performit(title, stmt="pass", setup="pass", repeat=1, number=1, globals=None):
|
159
|
+
""" 在timeit.repeat的基础上,做了层封装
|
160
|
+
|
161
|
+
200920周日15:33,简化函数,该函数不再获得执行结果,避免重复运行
|
162
|
+
|
163
|
+
:param title: 测试标题、名称功能
|
164
|
+
:return: 返回原函数单次执行结果
|
165
|
+
"""
|
166
|
+
data = timeit.repeat(stmt=stmt, setup=setup, repeat=repeat, number=number, globals=globals)
|
167
|
+
print(title, '用时(秒) ' + ValuesStat(data).summary(valfmt='.3f'))
|
168
|
+
return data
|
169
|
+
|
170
|
+
|
171
|
+
def perftest(title, stmt="pass", repeat=1, number=1, globals=None, res_width=None, print_=True):
|
172
|
+
""" 与performit的区别是,自己手动循环,记录程序运行结果
|
173
|
+
|
174
|
+
:param title: 测试标题、名称功能
|
175
|
+
:param res_width: 运行结果内容展示的字符上限数
|
176
|
+
:param print_: 输出报告
|
177
|
+
:return: 返回原函数单次执行结果
|
178
|
+
|
179
|
+
这里为了同时获得表达式返回值,就没有用标注你的timeit.repeat实现了
|
180
|
+
"""
|
181
|
+
# 1 确保stmt是可调用对象
|
182
|
+
if callable(stmt):
|
183
|
+
func = stmt
|
184
|
+
else:
|
185
|
+
code = compile(stmt, '', 'eval')
|
186
|
+
|
187
|
+
def func():
|
188
|
+
return eval(code, globals)
|
189
|
+
|
190
|
+
# 2 原函数运行结果(这里要先重载stdout)
|
191
|
+
data = []
|
192
|
+
res = ''
|
193
|
+
for i in range(repeat):
|
194
|
+
start = time.time()
|
195
|
+
for j in range(number):
|
196
|
+
res = func()
|
197
|
+
data.append(time.time() - start)
|
198
|
+
|
199
|
+
# 3 报告格式
|
200
|
+
if res_width is None:
|
201
|
+
# 如果性能报告比较短,只有一次测试,那res_width默认长度可以高一点
|
202
|
+
res_width = 50 if len(data) > 1 else 200
|
203
|
+
if res is None:
|
204
|
+
res = ''
|
205
|
+
else:
|
206
|
+
res = '运行结果:' + shorten(str(res), res_width)
|
207
|
+
if print_:
|
208
|
+
print(title, '用时(秒) ' + ValuesStat(data).summary(valfmt='.3f'), res)
|
209
|
+
|
210
|
+
return data
|
211
|
+
|
212
|
+
|
213
|
+
class PerfTest:
|
214
|
+
""" 这里模仿了unittest的机制
|
215
|
+
|
216
|
+
v0.0.38 重要改动,将number等参数移到perf操作,而不是类初始化中操作,继承使用上会更简单
|
217
|
+
"""
|
218
|
+
|
219
|
+
def perf(self, number=1, repeat=1, globals=None):
|
220
|
+
"""
|
221
|
+
|
222
|
+
:param number: 有些代码运算过快,可以多次运行迭代为一个单元
|
223
|
+
:param repeat: 对单元重复执行次数,最后会计算平均值、标准差
|
224
|
+
关于number和repeat的区别:
|
225
|
+
number张纸量repeat次
|
226
|
+
如果是纸箱这么厚的纸,number可以不设,默认是1
|
227
|
+
"""
|
228
|
+
# 1 找到所有perf_为前缀,且callable的函数方法
|
229
|
+
funcnames = []
|
230
|
+
for k in dir(self):
|
231
|
+
if k.startswith('perf_'):
|
232
|
+
if callable(getattr(self, k)):
|
233
|
+
funcnames.append(k)
|
234
|
+
|
235
|
+
# 2 自然排序
|
236
|
+
funcnames = natural_sort(funcnames)
|
237
|
+
funcnames2 = listalign([fn[5:] for fn in funcnames], 'r')
|
238
|
+
for i, funcname in enumerate(funcnames):
|
239
|
+
perftest(funcnames2[i], getattr(self, funcname),
|
240
|
+
number=int(number), repeat=int(repeat), globals=globals)
|