详解Rust泛型用法

news2024/11/27 5:31:24

文章目录

    • 基础语法
    • 泛型与结构体
    • 泛型约束
    • 泛型与生命周期
    • 泛型与枚举
    • 泛型和Vec
    • 静态泛型(const 泛型)
    • 类型别名
    • 默认类型参数
    • Sized Trait与泛型
    • 常量函数与泛型
    • 泛型的性能

Rust是一种系统编程语言,它拥有强大的泛型支持,泛型是Rust中用于实现代码复用和类型安全的重要特性。通过泛型程序员可以编写能够操作不同类型数据的函数、结构体、枚举和方法,同时又能确保类型安全,避免类型错误。在Rust中泛型的使用不仅能够提升代码的复用性,还能使得代码更加灵活,尤其是在实现与数据类型无关的算法时。

泛型的关键特点:
1.通过类型占位符(如 T)使得代码能够与多种类型一起工作。
2.Rust会在编译时检查类型,保证类型一致性。
3.通过泛型,我们可以在不牺牲类型安全的前提下编写通用代码。

基础语法

Rust泛型的基础语法是使用尖括号<>来指定类型的占位符。

fn print_value<T>(value: T) {
    println!("{:?}", value);
}

fn main() {
    print_value(42);           // T 由 i32 类型替代
    print_value("Hello, Rust!"); // T 由 &str 类型替代
}

在上面的代码中print_value函数接受一个类型为T的参数value并打印它的值。这里T是一个泛型类型,在main函数中分别传入了i32和&str类型。

泛型与结构体

Rust中的结构体也可以使用泛型,这使得我们能够定义更加通用的容器结构体。泛型结构体允许我们在实例化时为结构体的字段指定具体的类型。

struct Point<T> {
    x: T,
    y: T,
}

impl<T> Point<T> {
    fn new(x: T, y: T) -> Self {
        Point { x, y }
    }

    fn get_x(&self) -> &T {
        &self.x
    }

    fn get_y(&self) -> &T {
        &self.y
    }
}

fn main() {
    let p1 = Point::new(1, 2);     //T由i32类型替代
    let p2 = Point::new(1.1, 2.2); //T由f64类型替代

    println!("p1: ({}, {})", p1.get_x(), p1.get_y());
    println!("p2: ({}, {})", p2.get_x(), p2.get_y());
}

//两个参数可以类型不同  
struct Point<T,U> {
    x: T,
    y: U,
}
fn main() {
    let p = Point{x: 1, y :1.1};
}

泛型约束

泛型本身并不限制类型的行为,但在某些情况下我们希望限制泛型类型的行为或特性。为此Rust引入了泛型约束(Traits)。通过where关键字或者impl块中的trait约束,可以确保泛型类型实现了某些特定的行为。

use std::fmt::Debug;

fn print_debug<T: Debug>(value: T) {
    println!("{:?}", value);
}

fn main() {
    print_debug(42);             //适用于实现了Debug的i32类型
    print_debug("Hello, Rust!"); //适用于实现了Debug的&str类型
}

除了在函数签名中使用T: Trait语法外,Rust还允许通过where语法进行更复杂的泛型约束。

fn print_debug<T>(value: T)
where
    T: Debug,
{
    println!("{:?}", value);
}

泛型与生命周期

Rust的生命周期(lifetimes)和泛型是密切相关的。为了保证内存安全,Rust强制要求你显式标注引用类型的生命周期。这使得在处理泛型类型时,生命周期标注变得尤为重要。生命周期的内容在后面的文章中会详细介绍,这里就不细说了。

//生命周期 'a 确保了返回的引用在两个输入字符串的生命周期内有效
fn longest<'a, T>(s1: &'a str, s2: &'a str) -> &'a str {
    if s1.len() > s2.len() {
        s1
    } else {
        s2
    }
}

fn main() {
    let string1 = String::from("long string");
    let string2 = String::from("short");
    let result = longest(&string1, &string2);
    println!("The longest string is: {}", result);
}

泛型与枚举

Rust的枚举也可以使用泛型,这使得我们可以定义更灵活和强大的枚举类型。比如,Option 和 Result<T, E> 就是标准库中的泛型枚举类型。

//Option 用于值的存在与否不同,Result 关注的主要是值的正确性   
enum Option<T> {
    Some(T),
    None,
}
enum Result<T, E> {
    Ok(T),
    Err(E),
}

fn main() {
    let some_value = Option::Some(42);
    let none_value: Option<i32> = Option::None;
    
    match some_value {
        Option::Some(value) => println!("Some value: {}", value),
        Option::None => println!("No value"),
    }
}

泛型和Vec

Rust标准库中的Vec就是一个泛型集合类型,允许我们存储任意类型的元素。

fn main() {
    let mut numbers: Vec<i32> = Vec::new();
    numbers.push(1);
    numbers.push(2);
    numbers.push(3);

    for number in numbers {
        println!("{}", number);
    }
}

静态泛型(const 泛型)

Rust1.51版本引入了const泛型,使得你可以在编译时为泛型类型提供常量值。这通常用于数组大小、结构体字段或其他与常量相关的场景。通过const泛型,类型不仅限于具体的类型,也可以是编译时常量。

//在ArrayWrapper结构体中N代表数组的大小 而T则代表元素的类型。
struct ArrayWrapper<T, const N: usize> {
    data: [T; N],
}

impl<T, const N: usize> ArrayWrapper<T, N> {
    fn new(data: [T; N]) -> Self {
        ArrayWrapper { data }
    }

    fn print(&self) {
        for item in &self.data {
            println!("{:?}", item);
        }
    }
}

fn main() {
    let arr = ArrayWrapper::<i32, 5>::new([1, 2, 3, 4, 5]);
    arr.print();
}

//泛型T必须支持Debug特性  
//N这个泛型参数,它是一个基于值的泛型参数  任何长度的数组都可以传入  
fn display_array<T: std::fmt::Debug, const N: usize>(arr: [T; N]) {
    println!("{:?}", arr);
}
fn main() {
    let arr: [i32; 3] = [1, 2, 3];
    display_array(arr);

    let arr: [i32; 2] = [1, 2];
    display_array(arr);
}

这种方式使得Rust在编译时就能够推断出类型和常量大小,从而实现编译时的类型安全和高效性。

类型别名

Rust允许使用type关键字来创建类型别名,特别是当泛型类型变得过于复杂或冗长时。

type StringResult = Result<String, std::io::Error>;

fn get_file_content() -> StringResult {
    // 假设这是一个读取文件的函数,返回的是一个包含内容或错误的结果
    Ok("Hello, file!".to_string())
}

fn main() {
    match get_file_content() {
        Ok(content) => println!("{}", content),
        Err(e) => println!("Error: {}", e),
    }
}

默认类型参数

Rust中的泛型不仅可以是任意类型,也可以为泛型参数提供默认值。使用默认值可以减少函数或结构体定义中的样板代码,并使得用户在调用时不必每次都显式提供类型。

struct Wrapper<T = i32> {
    value: T,
}

impl<T> Wrapper<T> {
    fn new(value: T) -> Self {
        Wrapper { value }
    }
}

fn main() {
    let default_wrapper = Wrapper::new(42);  //T默认为 i32
    let string_wrapper = Wrapper::new(String::from("Hello")); //使用 String 类型
    println!("{}", default_wrapper.value);
    println!("{}", string_wrapper.value);
}

Sized Trait与泛型

Rust中的类型有一个特殊的trait叫 Sized,它表示一个类型的大小在编译时是已知的。绝大多数类型都实现了Sized trait,但也有一些例外(例如动态大小类型DST,如 str、[T]等)。在泛型中Sized trait很常见,通常它会隐式地应用于泛型参数。

//T: Sized限制了T必须是一个已知大小的类型  
fn print_size<T: Sized>(value: T) {
    println!("Size of value: {}", std::mem::size_of::<T>());
}

fn main() {
    let x = 42;
    print_size(x);
}

//例如str就是一个DST,如果想要在泛型函数中接受动态大小类型可以通过 ?Sized 来消除 Sized 限制
fn print_size<T: ?Sized>(value: &T) {
    // 允许接受动态大小类型
    println!("Size of value: {}", std::mem::size_of_val(value));
}

fn main() {
    let s: &str = "hello";
    print_size(s);
}

常量函数与泛型

const fn即常量函数。const fn允许我们在编译期对函数进行求值。在编译期就计算出一些值,以提高运行时的性能或满足某些编译期的约束条件提高运行时的性能,还使代码更加简洁和安全。

const fn add(a: usize, b: usize) -> usize {
    a + b
}
const RESULT: usize = add(5, 10);
fn main() {
    println!("The result is: {}", RESULT);
}


//const fn 与 const 泛型相结合  
struct Buffer<const N: usize> {
    data: [u8; N],
}

const fn compute_buffer_size(factor: usize) -> usize {
    factor * 1024
}

fn main() {
    const SIZE: usize = compute_buffer_size(4);
    let buffer = Buffer::<SIZE> {
        data: [0; SIZE],
    };
    println!("Buffer size: {} bytes", buffer.data.len());
}

泛型的性能

Rust中的泛型在编译时通过 monomorphization(单态化)机制进行优化。即每当一个泛型函数或结构体被实例化时,Rust会根据具体的类型生成专门的代码,从而避免了运行时的性能开销。因此Rust的泛型在运行时与手写的非泛型代码几乎没有区别,提供了与静态类型语言一样的性能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2248199.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Vue3+Pinia】Vue新一代状态管理器Pinia

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

【消息序列】详解(7):剖析回环模式--设备测试的核心利器

目录 一、概述 1.1. 本地回环模式 1.2. 远程环回模式 二、本地回环模式&#xff08;Local Loopback mode&#xff09; 2.1. 步骤 1&#xff1a;主机进入本地环回模式 2.2. 本地回环测试 2.2.1. 步骤 2a&#xff1a;主机发送HCI数据包并接收环回数据 2.2.2. 步骤 2b&…

大厂也在用的分布式链路追踪:TraceIdFilter + MDC + Skywalking

痛点 查线上日志时&#xff0c;同一个 Pod 内多线程日志交错&#xff0c;很难追踪每个请求对应的日志信息。 日志收集工具将多个 Pod 的日志收集到同一个数据库中后&#xff0c;情况就更加混乱不堪了。 解决 TraceId MDC 前端每次请求时&#xff0c;添加 X-App-Trace-Id 请…

leetcode - 2116. Check if a Parentheses String Can Be Valid

Description A parentheses string is a non-empty string consisting only of ‘(’ and ‘)’. It is valid if any of the following conditions is true: It is ().It can be written as AB (A concatenated with B), where A and B are valid parentheses strings.It ca…

如何启动 Docker 服务:全面指南

如何启动 Docker 服务:全面指南 一、Linux 系统(以 Ubuntu 为例)二、Windows 系统(以 Docker Desktop 为例)三、macOS 系统(以 Docker Desktop for Mac 为例)四、故障排查五、总结Docker,作为一种轻量级的虚拟化技术,已经成为开发者和运维人员不可或缺的工具。它允许用…

安装MySQL服务

安装版本MySQL8的安装包 安装界面 在这里选择MySQL中的Server only 只安装服务器端 如果选择custom需要如下图 进入配置导向&#xff0c;点击ready to configure&#xff0c;点击next即可 采用默认形式 执行成功后&#xff0c;会出现自动选择项 点击next然后再点击Finish 启动…

第六届国际科技创新学术交流大会暨新能源科学与电力工程国际(NESEE 2024)

重要信息 会议官网&#xff1a;nesee.iaecst.org 会议时间&#xff1a;2024年12月6-8日 会议地点&#xff1a; 中国-广州&#xff08;越秀国际会议中心) 大会简介 新能源科学与电力工程国际学术会议&#xff08;NESEE 2024&#xff09;作为第六届国际科技创新学术交流大会分…

Windows安装nacos

目录 一、下载 二、运行 三、运行失败 四、运行成功 一、下载 下载链接: Nacos Server 下载 | Nacos 官网 解压: 二、运行 进入nacos/bin, 运行startup.cmd 三、运行失败 如果出现黑窗口一闪而过, 说明失败了, 可能原因如下: ① 环境变量: 需要 JAVA_HOME 为 JDK8 ②…

Rust Newtype模式(通过结构体封装现有类型来创建新的类型)(单字段结构体,通过.0访问)模式匹配、解构、DerefMut

文章目录 深入理解Rust中的Newtype模式什么是Newtype模式&#xff1f;Newtype模式的基本形式Newtype的访问访问 Newtype 的值1. 通过 .0 访问字段2. 通过方法访问3. 通过模式匹配&#xff08;解构&#xff09;访问 总结 Newtype模式的应用场景1. 类型安全2. 增强可读性3. 定制化…

网络层协议IP

对于网络层我们直接通过IP协议来了解其内容 一.IP协议 首先我们先来了解几个概念&#xff1a; 主机&#xff1a;配有IP地址&#xff0c;但是不进行路由控制的设备 路由器&#xff1a;配有IP地址&#xff0c;同时进行路由控制的设备 节点&#xff1a;主机和路由器的统称 所以现在…

Qt界面篇:QMessageBox高级用法

1、演示效果 2、用法注意 2.1 设置图标 用于显示实际图标的pixmap取决于当前的GUI样式。也可以通过设置icon pixmap属性为图标设置自定义pixmap。 QMessageBox::Icon icon(

【强化学习的数学原理】第02课-贝尔曼公式-笔记

学习资料&#xff1a;bilibili 西湖大学赵世钰老师的【强化学习的数学原理】课程。链接&#xff1a;强化学习的数学原理 西湖大学 赵世钰 文章目录 一、为什么return重要&#xff1f;如何计算return&#xff1f;二、state value的定义三、Bellman公式的详细推导四、公式向量形式…

【数据结构实战篇】用C语言实现你的私有队列

&#x1f3dd;️专栏&#xff1a;【数据结构实战篇】 &#x1f305;主页&#xff1a;f狐o狸x 在前面的文章中我们用C语言实现了栈的数据结构&#xff0c;本期内容我们将实现队列的数据结构 一、队列的概念 队列&#xff1a;只允许在一端进行插入数据操作&#xff0c;在另一端…

实战项目负载均衡式在线 OJ

> 作者&#xff1a;დ旧言~ > 座右铭&#xff1a;松树千年终是朽&#xff0c;槿花一日自为荣。 > 目标&#xff1a;能自己实现负载均衡式在线 OJ。 > 毒鸡汤&#xff1a;有些事情&#xff0c;总是不明白&#xff0c;所以我不会坚持。早安! > 专栏选自&#xff1…

银河麒麟桌面系统——桌面鼠标变成x,窗口无关闭按钮的解决办法

银河麒麟桌面系统——桌面鼠标变成x&#xff0c;窗口无关闭按钮的解决办法 1、支持环境2、详细操作说明步骤1&#xff1a;用root账户登录电脑步骤2&#xff1a;导航到kylin-wm-chooser目录步骤3&#xff1a;编辑default.conf文件步骤4&#xff1a;重启电脑 3、结语 &#x1f49…

数据结构--AVL树(平衡二叉树)

✅博客主页:爆打维c-CSDN博客​​​​​​ &#x1f43e; &#x1f539;分享c、c知识及代码 &#x1f43e; &#x1f539;Gitee代码仓库 五彩斑斓黑1 (colorful-black-1) - Gitee.com 一、AVL树是什么&#xff1f;&#xff08;含义、性质&#xff09; 1.AVL树的概念 AVL树是最…

【Unity How】如何让物体跟随平台移动或旋转?

先看下最终要实现的效果&#xff1a; 当查找这个问题的资料时&#xff0c;发现多数的方案都是将物体设置为平台的子对象。 但是如果平台是非均匀缩放时&#xff0c;物体在移动或旋转时就会发生变形。 参考&#xff1a;Unity中父对象是非均匀缩放时出现倾斜或剪切现象 那有没有…

C语言函数递归经典题型——汉诺塔问题

一.汉诺塔问题介绍 Hanoi&#xff08;汉诺&#xff09;塔问题。古代有一个梵塔&#xff0c;塔内有3个座A、B、C&#xff0c;开始时&#xff21;座上有64个盘子&#xff0c;盘子大小不等&#xff0c;大的在下&#xff0c;小的在上。有一个老和尚想把这64个盘子从&#xff21;座移…

Unity中动态生成贴图并保存成png图片实现

实现原理&#xff1a; 要生成长x宽y的贴图&#xff0c;就是生成x*y个像素填充到贴图中&#xff0c;如下图&#xff1a; 如果要改变局部颜色&#xff0c;就是从x1到x2(x1<x2),y1到y2(y1<y2)这个范围做处理&#xff0c; 或者要想做圆形就是计算距某个点&#xff08;x1,y1&…

vue3封装Element Plus table表格组件

支持绝大部分Element Plus原有设置属性&#xff0c;支持分页&#xff0c;支持动态适配高度 效果展示 组件代码&#xff1a; <template><div class"table-wrap" ref"tableWrap"><el-tableclass"w100 h100":data"tableInfo.…