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.
Files changed (126) hide show
  1. pyxllib/__init__.py +21 -21
  2. pyxllib/algo/__init__.py +8 -8
  3. pyxllib/algo/disjoint.py +54 -54
  4. pyxllib/algo/geo.py +541 -541
  5. pyxllib/algo/intervals.py +964 -964
  6. pyxllib/algo/matcher.py +389 -389
  7. pyxllib/algo/newbie.py +166 -166
  8. pyxllib/algo/pupil.py +629 -629
  9. pyxllib/algo/shapelylib.py +67 -67
  10. pyxllib/algo/specialist.py +241 -241
  11. pyxllib/algo/stat.py +494 -494
  12. pyxllib/algo/treelib.py +149 -149
  13. pyxllib/algo/unitlib.py +66 -66
  14. pyxllib/autogui/__init__.py +5 -5
  15. pyxllib/autogui/activewin.py +246 -246
  16. pyxllib/autogui/all.py +9 -9
  17. pyxllib/autogui/autogui.py +852 -852
  18. pyxllib/autogui/uiautolib.py +362 -362
  19. pyxllib/autogui/virtualkey.py +102 -102
  20. pyxllib/autogui/wechat.py +827 -827
  21. pyxllib/autogui/wechat_msg.py +421 -421
  22. pyxllib/autogui/wxautolib.py +84 -84
  23. pyxllib/cv/__init__.py +5 -5
  24. pyxllib/cv/expert.py +267 -267
  25. pyxllib/cv/imfile.py +159 -159
  26. pyxllib/cv/imhash.py +39 -39
  27. pyxllib/cv/pupil.py +9 -9
  28. pyxllib/cv/rgbfmt.py +1525 -1525
  29. pyxllib/cv/slidercaptcha.py +137 -137
  30. pyxllib/cv/trackbartools.py +251 -251
  31. pyxllib/cv/xlcvlib.py +1040 -1040
  32. pyxllib/cv/xlpillib.py +423 -423
  33. pyxllib/data/echarts.py +240 -240
  34. pyxllib/data/jsonlib.py +89 -89
  35. pyxllib/data/oss.py +72 -72
  36. pyxllib/data/pglib.py +1127 -1127
  37. pyxllib/data/sqlite.py +568 -568
  38. pyxllib/data/sqllib.py +297 -297
  39. pyxllib/ext/JLineViewer.py +505 -505
  40. pyxllib/ext/__init__.py +6 -6
  41. pyxllib/ext/demolib.py +246 -246
  42. pyxllib/ext/drissionlib.py +277 -277
  43. pyxllib/ext/kq5034lib.py +12 -12
  44. pyxllib/ext/old.py +663 -663
  45. pyxllib/ext/qt.py +449 -449
  46. pyxllib/ext/robustprocfile.py +497 -497
  47. pyxllib/ext/seleniumlib.py +76 -76
  48. pyxllib/ext/tk.py +173 -173
  49. pyxllib/ext/unixlib.py +827 -827
  50. pyxllib/ext/utools.py +351 -351
  51. pyxllib/ext/webhook.py +124 -119
  52. pyxllib/ext/win32lib.py +40 -40
  53. pyxllib/ext/wjxlib.py +88 -88
  54. pyxllib/ext/wpsapi.py +124 -124
  55. pyxllib/ext/xlwork.py +9 -9
  56. pyxllib/ext/yuquelib.py +1105 -1105
  57. pyxllib/file/__init__.py +17 -17
  58. pyxllib/file/docxlib.py +761 -761
  59. pyxllib/file/gitlib.py +309 -309
  60. pyxllib/file/libreoffice.py +165 -165
  61. pyxllib/file/movielib.py +148 -148
  62. pyxllib/file/newbie.py +10 -10
  63. pyxllib/file/onenotelib.py +1469 -1469
  64. pyxllib/file/packlib/__init__.py +330 -330
  65. pyxllib/file/packlib/zipfile.py +2441 -2441
  66. pyxllib/file/pdflib.py +426 -426
  67. pyxllib/file/pupil.py +185 -185
  68. pyxllib/file/specialist/__init__.py +685 -685
  69. pyxllib/file/specialist/dirlib.py +799 -799
  70. pyxllib/file/specialist/download.py +193 -193
  71. pyxllib/file/specialist/filelib.py +2829 -2829
  72. pyxllib/file/xlsxlib.py +3131 -3131
  73. pyxllib/file/xlsyncfile.py +341 -341
  74. pyxllib/prog/__init__.py +5 -5
  75. pyxllib/prog/cachetools.py +64 -64
  76. pyxllib/prog/deprecatedlib.py +233 -233
  77. pyxllib/prog/filelock.py +42 -42
  78. pyxllib/prog/ipyexec.py +253 -253
  79. pyxllib/prog/multiprogs.py +940 -940
  80. pyxllib/prog/newbie.py +451 -451
  81. pyxllib/prog/pupil.py +1197 -1197
  82. pyxllib/prog/sitepackages.py +33 -33
  83. pyxllib/prog/specialist/__init__.py +391 -391
  84. pyxllib/prog/specialist/bc.py +203 -203
  85. pyxllib/prog/specialist/browser.py +497 -497
  86. pyxllib/prog/specialist/common.py +347 -347
  87. pyxllib/prog/specialist/datetime.py +198 -198
  88. pyxllib/prog/specialist/tictoc.py +240 -240
  89. pyxllib/prog/specialist/xllog.py +180 -180
  90. pyxllib/prog/xlosenv.py +108 -108
  91. pyxllib/stdlib/__init__.py +17 -17
  92. pyxllib/stdlib/tablepyxl/__init__.py +10 -10
  93. pyxllib/stdlib/tablepyxl/style.py +303 -303
  94. pyxllib/stdlib/tablepyxl/tablepyxl.py +130 -130
  95. pyxllib/text/__init__.py +8 -8
  96. pyxllib/text/ahocorasick.py +39 -39
  97. pyxllib/text/airscript.js +744 -744
  98. pyxllib/text/charclasslib.py +121 -121
  99. pyxllib/text/jiebalib.py +267 -267
  100. pyxllib/text/jinjalib.py +32 -32
  101. pyxllib/text/jsa_ai_prompt.md +271 -271
  102. pyxllib/text/jscode.py +922 -922
  103. pyxllib/text/latex/__init__.py +158 -158
  104. pyxllib/text/levenshtein.py +303 -303
  105. pyxllib/text/nestenv.py +1215 -1215
  106. pyxllib/text/newbie.py +300 -300
  107. pyxllib/text/pupil/__init__.py +8 -8
  108. pyxllib/text/pupil/common.py +1121 -1121
  109. pyxllib/text/pupil/xlalign.py +326 -326
  110. pyxllib/text/pycode.py +47 -47
  111. pyxllib/text/specialist/__init__.py +8 -8
  112. pyxllib/text/specialist/common.py +112 -112
  113. pyxllib/text/specialist/ptag.py +186 -186
  114. pyxllib/text/spellchecker.py +172 -172
  115. pyxllib/text/templates/echart_base.html +10 -10
  116. pyxllib/text/templates/highlight_code.html +16 -16
  117. pyxllib/text/templates/latex_editor.html +102 -102
  118. pyxllib/text/vbacode.py +17 -17
  119. pyxllib/text/xmllib.py +747 -747
  120. pyxllib/xl.py +42 -39
  121. pyxllib/xlcv.py +17 -17
  122. {pyxllib-0.3.197.dist-info → pyxllib-0.3.200.dist-info}/METADATA +1 -1
  123. pyxllib-0.3.200.dist-info/RECORD +126 -0
  124. {pyxllib-0.3.197.dist-info → pyxllib-0.3.200.dist-info}/licenses/LICENSE +190 -190
  125. pyxllib-0.3.197.dist-info/RECORD +0 -126
  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)