pyxllib 0.3.197__py3-none-any.whl → 3.201.1__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 (127) hide show
  1. pyxllib/__init__.py +14 -21
  2. pyxllib/algo/__init__.py +8 -8
  3. pyxllib/algo/disjoint.py +54 -54
  4. pyxllib/algo/geo.py +537 -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 +145 -149
  13. pyxllib/algo/unitlib.py +62 -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 +846 -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 +236 -240
  34. pyxllib/data/jsonlib.py +85 -89
  35. pyxllib/data/oss.py +72 -72
  36. pyxllib/data/pglib.py +1111 -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 +251 -246
  42. pyxllib/ext/drissionlib.py +277 -277
  43. pyxllib/ext/kq5034lib.py +12 -12
  44. pyxllib/ext/qt.py +449 -449
  45. pyxllib/ext/robustprocfile.py +493 -497
  46. pyxllib/ext/seleniumlib.py +76 -76
  47. pyxllib/ext/tk.py +173 -173
  48. pyxllib/ext/unixlib.py +821 -827
  49. pyxllib/ext/utools.py +345 -351
  50. pyxllib/ext/webhook.py +124 -119
  51. pyxllib/ext/win32lib.py +40 -40
  52. pyxllib/ext/wjxlib.py +91 -88
  53. pyxllib/ext/wpsapi.py +124 -124
  54. pyxllib/ext/xlwork.py +9 -9
  55. pyxllib/ext/yuquelib.py +1110 -1105
  56. pyxllib/file/__init__.py +17 -17
  57. pyxllib/file/docxlib.py +757 -761
  58. pyxllib/file/gitlib.py +309 -309
  59. pyxllib/file/libreoffice.py +165 -165
  60. pyxllib/file/movielib.py +144 -148
  61. pyxllib/file/newbie.py +10 -10
  62. pyxllib/file/onenotelib.py +1469 -1469
  63. pyxllib/file/packlib/__init__.py +330 -330
  64. pyxllib/file/packlib/zipfile.py +2441 -2441
  65. pyxllib/file/pdflib.py +422 -426
  66. pyxllib/file/pupil.py +185 -185
  67. pyxllib/file/specialist/__init__.py +681 -685
  68. pyxllib/file/specialist/dirlib.py +799 -799
  69. pyxllib/file/specialist/download.py +193 -193
  70. pyxllib/file/specialist/filelib.py +2825 -2829
  71. pyxllib/file/xlsxlib.py +3122 -3131
  72. pyxllib/file/xlsyncfile.py +341 -341
  73. pyxllib/prog/__init__.py +5 -5
  74. pyxllib/prog/cachetools.py +58 -64
  75. pyxllib/prog/deprecatedlib.py +233 -233
  76. pyxllib/prog/filelock.py +42 -42
  77. pyxllib/prog/ipyexec.py +253 -253
  78. pyxllib/prog/multiprogs.py +940 -940
  79. pyxllib/prog/newbie.py +451 -451
  80. pyxllib/prog/pupil.py +1208 -1197
  81. pyxllib/prog/sitepackages.py +33 -33
  82. pyxllib/prog/specialist/__init__.py +348 -391
  83. pyxllib/prog/specialist/bc.py +203 -203
  84. pyxllib/prog/specialist/browser.py +497 -497
  85. pyxllib/prog/specialist/common.py +347 -347
  86. pyxllib/prog/specialist/datetime.py +198 -198
  87. pyxllib/prog/specialist/tictoc.py +240 -240
  88. pyxllib/prog/specialist/xllog.py +180 -180
  89. pyxllib/prog/xlosenv.py +110 -108
  90. pyxllib/stdlib/__init__.py +17 -17
  91. pyxllib/stdlib/tablepyxl/__init__.py +10 -10
  92. pyxllib/stdlib/tablepyxl/style.py +303 -303
  93. pyxllib/stdlib/tablepyxl/tablepyxl.py +130 -130
  94. pyxllib/text/__init__.py +8 -8
  95. pyxllib/text/ahocorasick.py +36 -39
  96. pyxllib/text/airscript.js +754 -744
  97. pyxllib/text/charclasslib.py +121 -121
  98. pyxllib/text/jiebalib.py +267 -267
  99. pyxllib/text/jinjalib.py +27 -32
  100. pyxllib/text/jsa_ai_prompt.md +271 -271
  101. pyxllib/text/jscode.py +922 -922
  102. pyxllib/text/latex/__init__.py +158 -158
  103. pyxllib/text/levenshtein.py +303 -303
  104. pyxllib/text/nestenv.py +1215 -1215
  105. pyxllib/text/newbie.py +300 -300
  106. pyxllib/text/pupil/__init__.py +8 -8
  107. pyxllib/text/pupil/common.py +1121 -1121
  108. pyxllib/text/pupil/xlalign.py +326 -326
  109. pyxllib/text/pycode.py +47 -47
  110. pyxllib/text/specialist/__init__.py +8 -8
  111. pyxllib/text/specialist/common.py +112 -112
  112. pyxllib/text/specialist/ptag.py +186 -186
  113. pyxllib/text/spellchecker.py +172 -172
  114. pyxllib/text/templates/echart_base.html +10 -10
  115. pyxllib/text/templates/highlight_code.html +16 -16
  116. pyxllib/text/templates/latex_editor.html +102 -102
  117. pyxllib/text/vbacode.py +17 -17
  118. pyxllib/text/xmllib.py +741 -747
  119. pyxllib/xl.py +42 -39
  120. pyxllib/xlcv.py +17 -17
  121. pyxllib-3.201.1.dist-info/METADATA +296 -0
  122. pyxllib-3.201.1.dist-info/RECORD +125 -0
  123. {pyxllib-0.3.197.dist-info → pyxllib-3.201.1.dist-info}/licenses/LICENSE +190 -190
  124. pyxllib/ext/old.py +0 -663
  125. pyxllib-0.3.197.dist-info/METADATA +0 -48
  126. pyxllib-0.3.197.dist-info/RECORD +0 -126
  127. {pyxllib-0.3.197.dist-info → pyxllib-3.201.1.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)