1.结构体特点

Rust的结构体跟元组类型比较类似,它们都包含多个相关的值。和元组一样,结构体的每一部分可以是不同类型。但不同于元组,结构体需要命名各部分数据以便能清楚的表明其值的意义。由于有了这些名字,结构体比元组更灵活:不需要依赖顺序来指定或访问实例中的值。

定义结构体,需要使用 struct 关键字并为整个结构体提供一个名字。结构体的名字需要描述它所组合的数据的意义。接着,在大括号中,定义每一部分数据的名字和类型,一般称为 字段field)。

2.结构体定义和实例化

下面是一个结构体定义的示例:

struct User {
    active: bool,
    username: String,
    email: String,
    sign_in_count: u64,
}

一旦定义了结构体后,为了使用它,通过为每个字段指定具体值来创建这个结构体的 实例。创建一个实例需要以结构体的名字开头,接着在大括号中使用 key: value 键 - 值对的形式提供字段,其中 key 是字段的名字,value 是需要存储在字段中的数据值。实例中字段的顺序不需要和它们在结构体中声明的顺序一致。换句话说,结构体的定义就像一个类型的通用模板,而实例则会在这个模板中放入特定数据来创建这个类型的值。

下面是一个结构体的应用示例:

fn main() {
    let mut user1 = User {
        active: true,
        username: String::from("suntiger"),
        email: String::from("suntiger@example.com"),
        sign_in_count: 1,
    };
​
    user1.email = String::from("anotheremail@example.com");
}

整体代码如下:

Rust结构体的定义和实例化-LMLPHP

将代码进行编译, 可以发现email的内容被替换了,如图:

Rust结构体的定义和实例化-LMLPHP

注意看接收结构体实例的user1变量, 前面有mut关键字, 这样方便我们修改结构体成员, 注意整个结构体实例必须是可变的, Rust不允许只将结构体的某个字段标记为可变。

3.在函数中使用结构体

先来看一段在函数中返回结构体实例的代码:

fn build_user(email: String, username: String) -> User {
    User {
        active: true,
        username: username,
        email: email,
        sign_in_count: 1,
    }
}

我们定义了一个build_user 函数,它返回一个带有给定的 email 和用户名的 User 结构体实例。active 字段的值为 true,并且 sign_in_count 的值为 1。需要注意同其他任何表达式一样,我们可以在函数体的最后一个表达式中构造一个结构体的新实例,来隐式地返回这个实例。

4.字段初始化简化方式

为函数参数起与结构体字段相同的名字是可以理解的,但是不得不重复 emailusername 字段名称与变量会让人感到厌烦。如果结构体有更多字段,重复每个名称就更使人抓狂。还好Rust准备了简化的方法。

先看下面的代码:

fn build_user(email: String, username: String) -> User {
    User {
        active: true,
        username,
        email,
        sign_in_count: 1,
    }
}

参数名与字段名都完全相同,我们可以使用 字段初始化简写语法来重写 build_user,这样其行为与之前完全相同,无需在字段后面跟上相同名称的字段内容。

5.结构体更新语法

使用旧实例的大部分值但改变其部分值来创建一个新的结构体实例通常是很有用的。这可以通过结构体更新语法实现。

看下面的应用代码:

fn main() {
    let user2 = User {
        active: user1.active,
        username: user1.username,
        email: String::from("another@example.com"),
        sign_in_count: user1.sign_in_count,
    };
}

使用结构体更新语法,我们可以通过更少的代码来达到相同的效果,看下面的代码:

fn main() {
​
    let user2 = User {
        email: String::from("another@example.com"),
        ..user1
    };
}

.. 语法指定了剩余未显式设置值的字段应有与给定实例对应字段相同的值。与此同时这段代码也在 user2 中创建了一个新实例,但该实例中 email 字段的值与 user1 不同,而 usernameactivesign_in_count 字段的值与 user1 相同。..user1 必须放在最后,以指定其余的字段应从 user1 的相应字段中获取其值,但我们可以选择以任何顺序为任意字段指定值,而不用考虑结构体定义中字段的顺序。

6.元组结构体

也可以定义与元组类似的结构体,称为 元组结构体。元组结构体有着结构体名称提供的含义,但没有具体的字段名,只有字段的类型。当你想给整个元组取一个名字,并使元组成为与其他元组不同的类型时,元组结构体是很有用的,这时像常规结构体那样为每个字段命名就显得多余和形式化了。

要定义元组结构体,以 struct 关键字和结构体名开头并后跟元组中的类型。看下面这段代码:

struct Color(i32, i32, i32);
struct Point(i32, i32, i32);
​
fn main() {
    let black = Color(0, 0, 0);
    let origin = Point(0, 0, 0);
}

注意 blackorigin 值的类型不同,因为它们是不同的元组结构体的实例。我们定义的每一个结构体有其自己的类型,即使结构体中的字段可能有着相同的类型。例如,一个获取 Color 类型参数的函数不能接受 Point 作为参数,即便这两个类型都由三个 i32 值组成。在其他方面,元组结构体实例类似于元组,你可以将它们解构为单独的部分,也可以使用 . 后跟索引来访问单独的值,等等。

7.类单元结构体

我们也可以定义一个没有任何字段的结构体!它们被称为 类单元结构体, 因为它们类似于unit 类型。类单元结构体常常在你想要在某个类型上实现 trait 但不需要在类型中存储数据的时候发挥作用。

定义代码如下:

struct AlwaysEqual;
​
fn main() {
    let subject = AlwaysEqual;
}

要定义 AlwaysEqual,同样也使用 struct 关键字,然后后面跟想要的名称,然后是一个分号。不需要花括号或圆括号!然后,我们可以以类似的方式在 subject 变量中获得 AlwaysEqual 的实例:使用我们定义的名称,不需要任何花括号或圆括号。想象一下,我们将实现这个类型的行为,即每个实例始终等于每一个其他类型的实例,也许是为了获得一个已知的结果以便进行测试。我们不需要任何数据来实现这种行为。

11-09 10:00