-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathlib_strings.py
More file actions
317 lines (272 loc) · 11.2 KB
/
lib_strings.py
File metadata and controls
317 lines (272 loc) · 11.2 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
# -*- coding: utf-8 -*-
from __future__ import division
TYPE_CHECKING = False
if TYPE_CHECKING:
from typing import Any, Callable
from .lib_object import BaseManager
try:
chr = unichr # type: ignore
except Exception:
pass
class Strings:
"""
Strings 提供了对字符串的扩展操作
"""
_manager = BaseManager()
def __init__(self, manager): # type: (BaseManager) -> None
"""初始化并返回一个新的 Strings
Args:
manager (BaseManager):
用于管理引用对象的对象管理器
"""
self._manager = manager
def _force_cast(self, string): # type: (Any) -> str
"""
_force_cast 验证给定的对象是否为字符串,
并将该字符串强制转换为 Unicode 表示形式
Args:
string (Any):
欲验证并需要强制转换的字符串
Raises:
Exception:
如果给定的对象不是字符串,
则抛出相应的错误
Returns:
str: 返回给定字符串的 Unicode 表示
"""
if not isinstance(string, str):
raise Exception(
"_force_cast: Exist an argument that should be a string but is not a string"
)
try:
return string.decode(encoding="utf-8") # type: ignore
except Exception:
return string
def cast(self, ptr): # type: (int) -> str
"""
cast 将 ptr 指向的对象强制转换为字符串
Args:
ptr (int): 目标对象的指针
Returns:
str: 强制转换后所得的字符串
"""
return str(self._manager.deref(ptr))
def length(self, string): # type: (str) -> int
"""length 返回字符串的长度
Args:
string (str): 给定的字符串
Returns:
int: 字符串的长度
"""
return len(self._force_cast(string))
def format(self, string, *args): # type: (str, ...) -> str
"""format 格式化给定的字符串
Args:
string (str): 欲格式化的字符串
Returns:
str: 目标字符串的格式化结果
"""
return self._force_cast(string).format(*args)
def sub(self, string, start, end): # type: (str, int, int) -> str
"""sub 返回字符串的子字符串
Args:
string (str): 给定的字符串
start (int): 子字符串的起始索引
end (int): 子字符串的结束索引
Raises:
Exception:
如果给出的起始或结束索引超出字符串范围,
或结束索引小于起始索引,则抛出相应的错误
Returns:
str: 产生的子字符串
"""
string = self._force_cast(string)
if start < 0 or start > len(string):
raise Exception(
"strings.sub: Start index out of range [{}] with length {}".format(
start, len(string)
)
)
if end < 0 or end > len(string):
raise Exception(
"strings.sub: End index out of range [{}] with length {}".format(
end, len(string)
)
)
if end < start:
raise Exception(
"strings.sub: The end index can't be less than the start index (start={}, end={})".format(
start, end
)
)
return string[start:end]
def join(self, string, slice_ptr): # type: (str, int) -> str
"""
join 将切片中的元素以 string 作为分隔符连接形成一个新的字符串。
join 的调用者有义务确保切片中的所有元素都是字符串,否则会发生错误
Args:
string (str): 用作分隔符的字符串
slice_ptr (int): 目标切片的指针
Raises:
Exception:
如果 slice_ptr 指向的对象不是切片,
或切片中的某个元素不是字符串,
则抛出相应的错误
Returns:
str: 连接所得的字符串
"""
obj = self._manager.deref(slice_ptr)
if not isinstance(obj, list):
raise Exception("strings.join: Given ptr of slice is not a slice")
return self._force_cast(string).join(obj)
def split(
self, string, sep=None, maxsplit=-1
): # type: (str, str | None, int) -> int
"""
split 将字符串按指定分隔符拆分为多个子字符串,
并将这些子字符串按顺序放入一个新的切片中
Args:
string (str):
要被拆分的字符串
sep (str | None, optional):
拆分所使用的分隔符。
默认值为 None
maxsplit (int, optional):
最大拆分次数。
默认值为 -1
Returns:
int: 所产生的切片的指针
"""
return self._manager.ref(
self._force_cast(string).split(
self._force_cast(sep) if sep is not None else None,
maxsplit,
)
)
def rsplit(
self, string, sep=None, maxsplit=-1
): # type: (str, str | None, int) -> int
"""
rsplit 将字符串按指定分隔符从右侧开始拆分为多个子字符串,
并将这些子字符串按顺序放入一个新的切片中。
rsplit 与 split 的唯一区别在于,
rsplit 是从字符串的右侧开始拆分的,
而 split 则是从左侧开始拆分的
Args:
string (str):
给定的字符串
sep (str | None, optional):
分隔符。默认值为 None
maxsplit (int, optional):
最大拆分次数。默认值为 -1
Returns:
int: 所产生的切片的指针
"""
return self._manager.ref(
self._force_cast(string).rsplit(
self._force_cast(sep) if sep is not None else None,
maxsplit,
)
)
def equalfold(self, string_a, string_b): # type: (str, str) -> bool
"""
equalfold 比较两个字符串在忽略大小写的情况下是否相等
Args:
string_a (str): 被比较的第一个字符串
string_b (str): 被比较的第二个字符串
Returns:
bool: 两个字符串在忽略大小写的情况下是否相等
"""
return self._force_cast(string_a).lower() == self._force_cast(string_b).lower()
def build_func(
self,
origin, # type: dict[str, Callable[..., int | bool | float | str]]
): # type: (...) -> None
"""
build_func 构建 strings 模块的内置函数,
并将构建结果写入到传递的 origin 字典中
Args:
origin (dict[str, Callable[..., int | bool | float | str]]):
用于存放所有内置函数的字典
"""
funcs = {} # type: dict[str, Callable[..., int | bool | float | str]]
funcs["strings.cast"] = self.cast
funcs["strings.length"] = self.length
funcs["strings.format"] = self.format
funcs["strings.sub"] = self.sub
funcs["strings.ord"] = lambda string: ord(self._force_cast(string))
funcs["strings.chr"] = lambda i: chr(i)
funcs["strings.capitalize"] = lambda string: self._force_cast(
string
).capitalize()
funcs["strings.center"] = lambda string, width, fillchar=" ": self._force_cast(
string
).center(width, self._force_cast(fillchar))
funcs["strings.startswith"] = (
lambda string, prefix, start=None, end=None: self._force_cast(
string
).startswith(self._force_cast(prefix), start, end)
)
funcs["strings.endswith"] = (
lambda string, prefix, start=None, end=None: self._force_cast(
string
).endswith(self._force_cast(prefix), start, end)
)
funcs["strings.find"] = (
lambda string, prefix, start=None, end=None: self._force_cast(string).find(
self._force_cast(prefix), start, end
)
)
funcs["strings.rfind"] = (
lambda string, prefix, start=None, end=None: self._force_cast(string).rfind(
self._force_cast(prefix), start, end
)
)
funcs["strings.index"] = (
lambda string, prefix, start=None, end=None: self._force_cast(string).index(
self._force_cast(prefix), start, end
)
)
funcs["strings.rindex"] = (
lambda string, prefix, start=None, end=None: self._force_cast(
string
).rindex(self._force_cast(prefix), start, end)
)
funcs["strings.isalnum"] = lambda string: self._force_cast(string).isalnum()
funcs["strings.isalpha"] = lambda string: self._force_cast(string).isalpha()
funcs["strings.isdigit"] = lambda string: self._force_cast(string).isdigit()
funcs["strings.islower"] = lambda string: self._force_cast(string).islower()
funcs["strings.isspace"] = lambda string: self._force_cast(string).isspace()
funcs["strings.istitle"] = lambda string: self._force_cast(string).istitle()
funcs["strings.isupper"] = lambda string: self._force_cast(string).isupper()
funcs["strings.join"] = self.join
funcs["strings.ljust"] = lambda string, width, fillchar=" ": self._force_cast(
string
).ljust(width, self._force_cast(fillchar))
funcs["strings.rjust"] = lambda string, width, fillchar=" ": self._force_cast(
string
).rjust(width, self._force_cast(fillchar))
funcs["strings.lower"] = lambda string: self._force_cast(string).lower()
funcs["strings.upper"] = lambda string: self._force_cast(string).upper()
funcs["strings.lstrip"] = lambda string, chars=None: self._force_cast(
string
).lstrip(self._force_cast(chars) if chars is not None else None)
funcs["strings.rstrip"] = lambda string, chars=None: self._force_cast(
string
).rstrip(self._force_cast(chars) if chars is not None else None)
funcs["strings.strip"] = lambda string, chars=None: self._force_cast(
string
).strip(self._force_cast(chars) if chars is not None else None)
funcs["strings.replace"] = lambda string, old, new, count=-1: self._force_cast(
string
).replace(self._force_cast(old), self._force_cast(new), count)
funcs["strings.split"] = self.split
funcs["strings.rsplit"] = self.rsplit
funcs["strings.swapcase"] = lambda string: self._force_cast(string).swapcase()
funcs["strings.title"] = lambda string: self._force_cast(string).title()
funcs["strings.zfill"] = lambda string, width: self._force_cast(string).zfill(
width
)
funcs["strings.equalfold"] = self.equalfold
for key, value in funcs.items():
origin[key] = value