详解Rust 结构体

Rust 语言是一种高效、可靠的通用高级语言。其高效不仅限于开发效率,它的执行效率也是令人称赞的,是一种少有的兼顾开发效率和执行效率的语言。

Rust 中的结构体(Struct)与元组(Tuple)都可以将若干个类型不一定相同的数据捆绑在一起形成整体,但结构体的每个成员和其本身都有一个名字,这样访问它成员的时候就不用记住下标了。元组常用于非定义的多值传递,而结构体用于规范常用的数据结构。结构体的每个成员叫做”字段”。

结构体定义

这是一个结构体定义:

struct Site {
   domain: String,
   name: String,
   nation: String,
   found: u32
}

注意:如果你常用 C/C++,请记住在 Rust 里 struct 语句仅用来定义,不能声明实例,结尾不需要 ; 符号,而且每个字段定义之后用 , 分隔。

结构体实例

Rust 很多地方受 JavaScript 影响,在实例化结构体的时候用 JSON 对象的 key: value 语法来实现定义:

实例

let runoob = Site {
   domain: String::from("www.runoob.com"),
   name: String::from("RUNOOB"),
   nation: String::from("China"),
   found: 2013
};

如果你不了解 JSON 对象,你可以不用管它,记住格式就可以了:

结构体类名 {
   字段名 : 字段值,
   ...
}

这样的好处是不仅使程序更加直观,还不需要按照定义的顺序来输入成员的值。

如果正在实例化的结构体有字段名称和现存变量名称一样的,可以简化书写:

实例

let domain = String::from("www.runoob.com");
let name = String::from("RUNOOB");
let runoob = Site {
   domain,  // 等同于 domain : domain,
   name,    // 等同于 name : name,
   nation: String::from("China"),
   traffic: 2013
};

有这样一种情况:你想要新建一个结构体的实例,其中大部分属性需要被设置成与现存的一个结构体属性一样,仅需更改其中的一两个字段的值,可以使用结构体更新语法:

let site = Site {
   domain: String::from("www.runoob.com"),
   name: String::from("RUNOOB"),
   ..runoob
};

注意:..runoob 后面不可以有逗号。这种语法不允许一成不变的复制另一个结构体实例,意思就是说至少重新设定一个字段的值才能引用其他实例的值。

元组结构体

有一种更简单的定义和使用结构体的方式:元组结构体。

元组结构体是一种形式是元组的结构体。

与元组的区别是它有名字和固定的类型格式。它存在的意义是为了处理那些需要定义类型(经常使用)又不想太复杂的简单数据:

struct Color(u8, u8, u8);
struct Point(f64, f64);

let black = Color(0, 0, 0);
let origin = Point(0.0, 0.0);

“颜色”和”点坐标”是常用的两种数据类型,但如果实例化时写个大括号再写上两个名字就为了可读性牺牲了便捷性,Rust 不会遗留这个问题。元组结构体对象的使用方式和元组一样,通过 . 和下标来进行访问:

实例

fn main() {
   struct Color(u8, u8, u8);
   struct Point(f64, f64);

   let black = Color(0, 0, 0);
   let origin = Point(0.0, 0.0);

   println!("black = ({}, {}, {})", black.0, black.1, black.2);
   println!("origin = ({}, {})", origin.0, origin.1);
}

运行结果:

black = (0, 0, 0)
origin = (0, 0)

结构体所有权

结构体必须掌握字段值所有权,因为结构体失效的时候会释放所有字段。

这就是为什么本章的案例中使用了 String 类型而不使用 &str 的原因。

但这不意味着结构体中不定义引用型字段,这需要通过”生命周期”机制来实现。

但现在还难以说明”生命周期”概念,所以只能在后面章节说明。

输出结构体 调试中,完整地显示出一个结构体实例是非常有用的。但如果我们手动的书写一个格式会非常的不方便。所以 Rust 提供了一个方便地输出一整个结构体的方法:

实例

#[derive(Debug)]

struct Rectangle {
   width: u32,
   height: u32,
}

fn main() {
   let rect1 = Rectangle { width: 30, height: 50 };

   println!("rect1 is {:?}", rect1);
}

如第一行所示:一定要导入调试库 #[derive(Debug)] ,之后在 println 和 print 宏中就可以用 {:?} 占位符输出一整个结构体:

rect1 is Rectangle { width: 30, height: 50 }

如果属性较多的话可以使用另一个占位符 {:#?} 。

输出结果:

rect1 is Rectangle {
   width: 30,
   height: 50
}

结构体方法

方法(Method)和函数(Function)类似,只不过它是用来操作结构体实例的。

如果你学习过一些面向对象的语言,那你一定很清楚函数一般放在类定义里并在函数中用 this 表示所操作的实例。

Rust 语言不是面向对象的,从它所有权机制的创新可以看出这一点。但是面向对象的珍贵思想可以在 Rust 实现。

结构体方法的第一个参数必须是 &self,不需声明类型,因为 self 不是一种风格而是关键字。

计算一个矩形的面积:

实例

struct Rectangle {
   width: u32,
   height: u32,
}
 
impl Rectangle {
   fn area(&self) -> u32 {
       self.width * self.height
   }
}

fn main() {
   let rect1 = Rectangle { width: 30, height: 50 };
   println!("rect1's area is {}", rect1.area());
}

输出结果:

rect1's area is 1500 

请注意,在调用结构体方法的时候不需要填写 self ,这是出于对使用方便性的考虑。

一个多参数的例子:

实例

struct Rectangle {
   width: u32,
   height: u32,
}

impl Rectangle {
   fn area(&self) -> u32 {
       self.width * self.height
   }

   fn wider(&self, rect: &Rectangle) -> bool {
       self.width > rect.width
   }
}

fn main() {
   let rect1 = Rectangle { width: 30, height: 50 };
   let rect2 = Rectangle { width: 40, height: 20 };

   println!("{}", rect1.wider(&rect2));
}

运行结果:

false

这个程序计算 rect1 是否比 rect2 更宽。

结构体关联函数

之所以”结构体方法”不叫”结构体函数”是因为”函数”这个名字留给了这种函数:它在 impl 块中却没有 &self 参数。

这种函数不依赖实例,但是使用它需要声明是在哪个 impl 块中的。

一直使用的 String::from 函数就是一个”关联函数”。

实例

#[derive(Debug)]
struct Rectangle {
   width: u32,
   height: u32,
}

impl Rectangle {
   fn create(width: u32, height: u32) -> Rectangle {
       Rectangle { width, height }
   }
}

fn main() {
   let rect = Rectangle::create(30, 50);
   println!("{:?}", rect);
}

运行结果:

Rectangle { width: 30, height: 50 }

贴士:结构体 impl 块可以写几次,效果相当于它们内容的拼接!

单元结构体

结构体可以值作为一种象征而无需任何成员:

struct UnitStruct;

我们称这种没有身体的结构体为单元结构体(Unit Struct)。

文章来源网络,作者:运维,如若转载,请注明出处:https://shuyeidc.com/wp/199919.html<

(0)
运维的头像运维
上一篇2025-04-05 01:37
下一篇 2025-04-05 01:38

相关推荐

  • 个人主题怎么制作?

    制作个人主题是一个将个人风格、兴趣或专业领域转化为视觉化或结构化内容的过程,无论是用于个人博客、作品集、社交媒体账号还是品牌形象,核心都是围绕“个人特色”展开,以下从定位、内容规划、视觉设计、技术实现四个维度,详细拆解制作个人主题的完整流程,明确主题定位:找到个人特色的核心主题定位是所有工作的起点,需要先回答……

    2025-11-20
    0
  • 社群营销管理关键是什么?

    社群营销的核心在于通过建立有温度、有价值、有归属感的社群,实现用户留存、转化和品牌传播,其管理需贯穿“目标定位-内容运营-用户互动-数据驱动-风险控制”全流程,以下从五个维度展开详细说明:明确社群定位与目标社群管理的首要任务是精准定位,需明确社群的核心价值(如行业交流、产品使用指导、兴趣分享等)、目标用户画像……

    2025-11-20
    0
  • 香港公司网站备案需要什么材料?

    香港公司进行网站备案是一个涉及多部门协调、流程相对严谨的过程,尤其需兼顾中国内地与香港两地的监管要求,由于香港公司注册地与中国内地不同,其网站若主要服务内地用户或使用内地服务器,需根据服务器位置、网站内容性质等,选择对应的备案路径(如工信部ICP备案或公安备案),以下从备案主体资格、流程步骤、材料准备、注意事项……

    2025-11-20
    0
  • 如何企业上云推广

    企业上云已成为数字化转型的核心战略,但推广过程中需结合行业特性、企业痛点与市场需求,构建系统性、多维度的推广体系,以下从市场定位、策略设计、执行落地及效果优化四个维度,详细拆解企业上云推广的实践路径,精准定位:明确目标企业与核心价值企业上云并非“一刀切”的方案,需先锁定目标客户群体,提炼差异化价值主张,客户分层……

    2025-11-20
    0
  • PS设计搜索框的实用技巧有哪些?

    在PS中设计一个美观且功能性的搜索框需要结合创意构思、视觉设计和用户体验考量,以下从设计思路、制作步骤、细节优化及交互预览等方面详细说明,帮助打造符合需求的搜索框,设计前的规划明确使用场景:根据网站或APP的整体风格确定搜索框的调性,例如极简风适合细线条和纯色,科技感适合渐变和发光效果,电商类则可能需要突出搜索……

    2025-11-20
    0

发表回复

您的邮箱地址不会被公开。必填项已用 * 标注