我对正则表达式的经验非常有限,所以我希望有人可以帮忙。
我正在制作一个Python 3游戏,它的棋盘有一个矩形网格。我正在尝试为用户提供一种方法,可以按以下形式一次将多个板坐标输入到名为coords
的字符串中:
(x1, y1), (x2, y2), (x3, y3), ... (xn, yn)
我希望输出为类似形式的称为cells
的元组列表:
[(x1, y1), (x2, y2), (x3, y3), ... (xn, yn)]
所以本质上我想模仿如何用Python代码编写元组。
现在我正在使用:
cells = [tuple(coords[i.find('(') + 1: i.rfind(')')].split(',')) for i in coords.split()]
可以为(1,2) (3,4) (5,6)
形式的输入产生所需的结果,输入的元组内部没有空格,并且元组之间没有空格。但是,这会破坏不完全遵循该格式的输入,并且它不会检查有效输入。对于cells
中元组中的每个x和y值,我需要验证:
理想情况下,如果用户输入多个有效坐标和一些无效坐标,则将有效的元组添加到cells
,并且会向用户显示类似"The following coordinates were invalid: (x1, y1), ..."
的消息。
我知道我可以通过一堆循环和流控制来完成所有这些工作,但是有没有更多的Python方式可以使用正则表达式完成此操作?
编辑:拼写
答案 0 :(得分:2)
正则表达式用于测试整体结构-其余部分不带正则表达式:
inp = "(3, 4), (a, 7), (-3, 3), (3, 3)"
def MaybeInt(n):
"""Returns an int if possible, else the original value."""
try:
return int(n)
except:
return n
def inX(n):
"""True if inside X of board."""
return 0<=n<5
def inY(n):
"""True if inside Y of board."""
return 0<=n<5
def structOk(t):
import re
return re.match(r'^\s*([^,]+,[^,]+\)\s*(?:,\s*\([^,]+,[^,]+\))*)\s*$',t)
def validate(t):
t = t.replace(" ","")
if not structOk(t):
raise ValueError("Structually unsound - try again: "+t)
k = [ x.lstrip("(") for x in t.replace(" ","").rstrip(")").split("),")]
tups = [ tuple(map(MaybeInt,tu.split(","))) for tu in k]
# our "decider" for whats valid/invalid which is used to divide the tuples
# into our valid/invalid return list's
test = lambda q: all(isinstance(num,int) for num in q) and inX(q[0]) and inY(q[1])
rv = [[],[]] # prepare and append by our decider test
for k in tups:
# True == 1, False == 0
rv[1-test(k)].append(k)
return rv
valid, invalid = validate(inp)
print(valid)
print(invalid)
输出:
[(3, 4), (3, 3)] # valid
[('a', 7), (-3, 3)] # invalid
有关正则表达式和详细说明,请参见https://regex101.com/r/OrHGaR/1。
简短说明:它寻找(...,...)时...不是...-您可以使用f.e. [1234567890a-zA-Z]
而不是[^,]
,但是它早晚会进入ValueError。