Rust_2025:阶段1:day7.1 类型转换

as

rust 复制代码
fn average(values: &[f64]) -> f64 {
    let total = values.iter().sum::<f64>();
    total / values.len() as f64
}
  1. 只有一些标准库的特定类型可以用as
  2. 可能会丢失精度,因为会截断

From

rust 复制代码
impl From<&str> for Person {
    fn from(s: &str) -> Person {
        let parts = s.split(',').collect::<Vec <&str>>();
        if parts.len() != 2 {
            return Self::default();
        }
        if parts[0] == "" {
            return Self::default();
        }
        if let Err(_) = parts[1].parse::<usize>(){
            return Self::default();
        }
        Person{ name : parts[0].to_string() , age : parts[1].parse::<usize>().unwrap()}
    }
}
  1. 只能返回Self,无法进行错误处理
  2. 上文如果发现非法字符只能返回默认选项

FromStr

rust 复制代码
impl FromStr for Person {
    type Err = ParsePersonError;
    fn from_str(s: &str) -> Result<Person, Self::Err> {
        if s.len() == 0 {
            return Err(ParsePersonError::Empty);
        }
        let parts = s.split(',').collect::<Vec <&str>>();
        if parts.len() != 2 {
            return Err(ParsePersonError::BadLen);
        }
        if parts[0] == "" {
            return Err(ParsePersonError::NoName);
        }
        if let Err(x) = parts[1].parse::<usize>(){
            return Err(ParsePersonError::ParseInt(x));
        }
        Ok(Person{ name : parts[0].to_string() , age : parts[1].parse::<usize>().unwrap()})
    }
}
  1. 特指字符串转换为其它类型
  2. 实现后可以调用parse< T >()方法,用于将字符串转换为指定类型

TryFrom

rust 复制代码
impl TryFrom<(i16, i16, i16)> for Color {
    type Error = IntoColorError;
    fn try_from(tuple: (i16, i16, i16)) -> Result<Self, Self::Error> {
        if tuple.0 > 255 || tuple.0 < 0 || tuple.1 > 255 || tuple.1 < 1 || tuple.2 > 255 || tuple.2 < 1{
            return Err(Self::Error::IntConversion);
        }
        Ok(Self{red : tuple.0 as u8, green : tuple.1 as u8, blue : tuple.2 as u8})
    }
}

// Array implementation
impl TryFrom<[i16; 3]> for Color {
    type Error = IntoColorError;
    fn try_from(arr: [i16; 3]) -> Result<Self, Self::Error> {
        let mut p = 0;
        for x in arr.iter() {
            if *x > 255 || *x < 1{
                return Err(Self::Error::IntConversion);
            }
            p += 1;
        }
        if p != 3 {
            return Err(Self::Error::BadLen);
        }
        Ok(Self{red : arr[0] as u8, green : arr[1] as u8, blue : arr[2] as u8})
    }
}

// Slice implementation
impl TryFrom<&[i16]> for Color {
    type Error = IntoColorError;
    fn try_from(slice: &[i16]) -> Result<Self, Self::Error> {
        if slice.len() != 3 {
            return Err(Self::Error::BadLen);
        }
        for x in slice.iter() {
            if *x > 255 || *x < 1{
                return Err(Self::Error::IntConversion);
            }
        }
        
        Ok(Self{red : slice[0] as u8, green : slice[1] as u8, blue : slice[2] as u8})
    }
}
  1. 相较于from,可以进行错误处理
  2. 相较于FromStr,可以用于其他类型
  3. 注:类型实现中的Error类型是指:在这类方法中,临时增加一个字段用于存储一个枚举值(该枚举值常用于错误处理,即经常被定义为错误枚举型)

AsRef和AsMut

rust 复制代码
fn byte_counter<T : AsRef<str>>(arg: T) -> usize {
    arg.as_ref().as_bytes().len()
}

// Obtain the number of characters (not bytes) in the given argument.
// TODO: Add the AsRef trait appropriately as a trait bound.
fn char_counter<T : AsRef<str>>(arg: T) -> usize {
    arg.as_ref().chars().count()
}

// Squares a number using as_mut().
// TODO: Add the appropriate trait bound.
fn num_sq<T : AsMut<u32>>(arg: &mut T) {
    // TODO: Implement the function body.
    let mut x = arg.as_mut();
    *x *= *x;
}
  1. 两者分别对应两个方法:
  • as_ref,根据特性约束中的泛型,返回对应泛型的不可变引用(第一个函数返回了一个&str)
  • as_mut,返回可变引用。
rust 复制代码
fn foo<T: AsRef<str> + AsRef<[u8]>>(arg: T) {
   let s: &str = arg.as_ref();
   let bytes: &[u8] = arg.as_ref();
   // ...可以同时使用
}
  1. 可以通过明确类型来同时使用多个AsRef
  2. 但特性约束必须要写。
相关推荐
花褪残红青杏小6 小时前
Rust图像处理第8节-暗角 & 复古胶片特效:四周衰减中心高亮
rust·webassembly·图形学
独孤留白21 小时前
从C到Rust:Rust 的 Trait 不是Interface,那是什么?
rust
花褪残红青杏小1 天前
Rust图像处理第7节-马赛克像素化:分块取平均色实现打码风格
rust·webassembly·图形学
doiito2 天前
【Agent Harness】Gliding Horse 设计细节 -- 不跟风开发自己的AI Agent
架构·rust·agent
doiito2 天前
【Agent Harness】Gliding Horse 核心设计理念,不跟风开发自己的AI Agent
ai·rust·架构设计·系统设计·ai agent
花褪残红青杏小2 天前
Rust图像处理第6节- 均值模糊 & 中值模糊:3×3 邻域的两种经典玩法
rust·webassembly·图形学
子兮曰3 天前
前端工具链的「Rust 化」:一场没有赢家的军备竞赛?
前端·后端·rust
星栈3 天前
写 Dioxus Demo 不难,难的是把它写成项目
前端·rust·前端框架
mCell3 天前
【锐评】桌面端技术营销:别拿跑分当工程判断
前端·rust·electron
武子康3 天前
调查研究-201 Rust 里的 dev build 和 release build:为什么同一份代码性能差这么多?
后端·架构·rust