Zachary Milner
Zachary Milner

Reputation: 333

How to use match case to check for a variable's type in python?

I have this code to check whether or not a variable is a number or a Vector2:

def __mul__(self, other):
    match type(other):
        case int | float:
            pass
        case Vector2:
            pass

If I run this, I get

SyntaxError: name capture 'int' makes remaining patterns unreachable

And when I hover in vscode, it gives me:

"int" is not accessed
Irrefutable pattern allowed only as the last subpattern in an "or" pattern
All subpatterns within an "or" pattern must target the same names
Missing names: "float"
Irrefutable pattern is allowed only for the last case statement

If I remove | float it still won't work, so I can't make them separate cases.

Upvotes: 15

Views: 11330

Answers (3)

Gabriel S.
Gabriel S.

Reputation: 19

This also works:

def print_type(v):
    match v:
        # None must come before int
        case None:
            print("NONE")
        # bool() must come before int
        case bool():
            print("BOOL")
        case int():
            print("INT")
        case float():
            print("FLOAT")
        case Vector2():
            print("VECTOR2")
        case str():
            print("STR")

Examples:

print_type(1)  # prints INT
a = 1
print_type(a)  # prints INT
print_type(None)  # prints NONE
print_type("Hello")  # prints STR
v = Vector2()
print_type(v)  # prints VECTOR2

Read more about this from the PEP 622 documentation.

Upvotes: 0

user3755605
user3755605

Reputation: 133

You can add a condition on the pattern:

class A:
    pass

class B:
    pass

class C:
    pass


def check_type(obj: Union[A, B, C]):
    match obj:
        case x if isinstance(x, A):
            return "A"
        case x if isinstance(x, B):
            return "B"
        case x if isinstance(x, C):
            return "C"
        case _:
            return "Unknown"

if __name__ == "__main__":
    a = A()
    b = B()
    c = C()
    print(check_type(a))
    print(check_type(b))
    print(check_type(c))

Output:

A
B
C

See Adding conditions to patterns section of PEP 636.

Upvotes: 3

Rob S.
Rob S.

Reputation: 2548

Case with a variable (ex: case _: or case other:) needs to be the last case in the list. It matches any value, where the value was not matched by a previous case, and captures that value in the variable.

A type can be used in a case, but implies isinstance(), testing to determine if the value being matched is an instance of that type. Therefore, the value used for the match should be the actual variable other rather than the type type(other), since type(other) is a type whose type would match type().

def __mul__(self, other):
    match other:
        case int() | float():
            pass
        case Vector2():
            pass

Upvotes: 23

Related Questions