Why does the compiler assume that the value of if

2019-02-28 20:19发布

I have the following code:

use std::collections::HashSet;

fn translate() -> Option<String> {
    None
}

fn main() {
    let mut found = HashSet::new();

    if let Some(tr) = translate() {
        found.insert(tr);
    }
}

It works properly, but when I remove the semicolon after found.insert(tr), I get a compiler error:

error[E0308]: mismatched types
  --> src/main.rs:11:9
   |
7  | fn main() {
   |           - expected `()` because of default return type
...
11 |         found.insert(tr)
   |         ^^^^^^^^^^^^^^^^- help: try adding a semicolon: `;`
   |         |
   |         expected (), found bool
   |
   = note: expected type `()`
              found type `bool`

It doesn't matter where this code is located or whether it is the last expression of the function.

Why does the compiler assume that the expression inside the curly braces should be ()?

标签: rust
2条回答
劳资没心,怎么记你
2楼-- · 2019-02-28 20:47

When you omit a function's return type, the function actually returns (). That is,

fn foo() {}

is equivalent to:

fn foo() -> () {}

If I add return (); at the end of the function I still get the same error. So I'm not even sure if that has something to do with a function return value.

An if let expression that is used as a statement must return (), unless it is the last expression in the function's body, in which case its type must match the function's return type. Since your if let doesn't have an else clause, its type must be ().

查看更多
迷人小祖宗
3楼-- · 2019-02-28 20:52

According to the Rust Book (emphasis mine):

The value of the expression is the value of the last expression in whichever branch was chosen. An if without an else always results in () as the value.

This gives a constraint on the expression value inside the curly braces.

This is correct since the expression type matches ():

if let Some(_) = some() {
    ()
};

This is correct since there's an else statement (and the types between the branches match):

if let Some(_) = some() {
    true
} else {
    false
};

But this is wrong:

if let Some(_) = some() {
    true
};

This answer was inspired by this comment.

查看更多
登录 后发表回答