有没有办法使用 python 的“match”语句来测试 Iterable 是否包含模式?

tjf*_*801 7 python iterable iterable-unpacking python-3.10 structural-pattern-matching

这与新的 Python 3.10 beta 和新语法有关match。有没有什么方法可以检查模式是否简单地包含在可迭代中?最明显的解决方案是简单地在两侧放置两个通配符,但这会SyntaxError由于来自可迭代解包的解包语法而引发。

有没有可能的方法来做到这一点?注意:在示例中使用包装类之类的东西numbers 就可以了,只要它可以使用匹配块并且至少具有一定的可读性,但我已经尝试过这一点,但没有取得太大成功

例子:

numbers = [1, 2, 3, 5, 7, 8, 9] #does not have to be a list, could be a class if needed

match numbers:
    # this just raises a SyntaxError, but I need a way to do something equivalent to this
    case [*_, (5 | 6), *_]:
        print("match!")
Run Code Online (Sandbox Code Playgroud)

Bra*_*her 6

不,我们目前没有任何计划支持可迭代包含检查作为结构模式匹配语法的一部分。

用合法的 Python 重写示例的最佳方法是使用正常if测试:

if any(i in (5, 6) for i in numbers):
    print("match!")
Run Code Online (Sandbox Code Playgroud)

如果包含检查只是更复杂模式的一部分,您可以将其编写为防护:

match something:
    case [pattern, capturing, numbers] if any(i in (5, 6) for i in numbers):
        print("match!")
Run Code Online (Sandbox Code Playgroud)

当然,如果你有一个序列,也可以在已知索引处找到一个项目:

match numbers:
    case [_, _, _, 5 | 6, *_]:
        print("match at position 3!")
    case [*_, 5 | 6, _, _, _]:
        print("match at position -4!")
Run Code Online (Sandbox Code Playgroud)

照这样说...

在示例中使用诸如包装类之类的东西numbers就可以了,只要它可以使用匹配块并且至少具有一定的可读性

...我认为映射模式可以被黑客攻击来完成这项工作(前提是您的所有项目都是可散列的并且没有不寻常的平等规则):

match dict.fromkeys(numbers):
    case {5: _} | {6: _}:
        print("match!")
Run Code Online (Sandbox Code Playgroud)

不过,我强烈推荐if该表格。