Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Is it possible to pass patterns as function arguments in Rust? [duplicate]

Tags:

rust

I'm implementing a parser in Rust and whitespace is a common pattern that I want to reuse in match patterns.

This code works:

let ch = ' ';

match ch {
    ' ' | '\n' | '\t' | '\r' => println!("whitespace"),
     _ => println!("token"),
}

This would get really repetitive if I need to keep on specifying the whitespace pattern each time. I would like to define that once and reuse it. I want to do something like:

let whitespace = ' ' | '\n' | '\t' | '\r';

let ch = ' ';

match ch {
    whitespace => println!("whitespace"),
    _          => println!("token"),
}

The compiler does not like the ws assignment. It interprets the | as a binary operation instead of alternation.

Can patterns be stored in variables somehow? Is there a better or more idiomatic way to do this?

like image 760
Brennan Cheung Avatar asked Nov 22 '25 06:11

Brennan Cheung


1 Answers

Can patterns be stored in variables somehow?

No. Patterns are a compile-time construct, and variables hold run-time concepts.

Is there a better or more idiomatic way to do this?

Creating a function or method is always a good solution to avoid repeating code. You can then use this as a guard clause:

fn is_whitespace(c: char) -> bool {
    match c {
        ' ' | '\n' | '\t' | '\r' => true,
        _ => false,
    }
}

fn main() {
    let ch = ' ';

    match ch {
        x if is_whitespace(x) => println!("whitespace"),
        _ => println!("token"),
    }
}

I'd also strongly recommend using an existing parser, of which there are a multitude, but everyone wants their Rust "hello world" to be parsing, for whatever reason.

A parsing library I use allows writing code akin to this, where whitespace is a function that knows how to parse the valid types of whitespace:

sequence!(pm, pt, {
    _          = literal("if");
    ws         = whitespace;
    _          = literal("let");
    ws         = append_whitespace(ws);
    pattern    = pattern;
    ws         = optional_whitespace(ws);
    _          = literal("=");
    ws         = optional_whitespace(ws);
    expression = expression;
}, |_, _| /* do something with pieces */);

Each of the things on the right-hand side are still individual functions that know how to parse something specific.

like image 53
Shepmaster Avatar answered Nov 24 '25 15:11

Shepmaster



Donate For Us

If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!