Rust常用特型之ToOwned特型示例详解

 更新时间:2024年04月22日 09:32:23   作者:AiMateZero  
在Rust中,假定某类型实现了Clone特型,如果给你一个对它引用,那我们得到它指向内容的备份的最常见方式是调用其clone()函数,这篇文章主要介绍了Rust常用特型之ToOwned特型,需要的朋友可以参考下

在Rust标准库中,存在很多常用的工具类特型,它们能帮助我们写出更具有Rust风格的代码。

ToOwned

这次我们来学一个和Borrow特型相关的特型,叫ToOwned类型。看字面意思Borrow是代表借出,而ToOwned代表去拥有它。

在Rust中,假定某类型实现了Clone特型,如果给你一个对它引用,那我们得到它指向内容的备份的最常见方式是调用其clone()函数。但是如果你想克隆&str或者&[i32]时会发生什么呢?你的目的可能是想得到一个String或者Vec<i32>。但是根据Clone特型的定义,你无法得到它们。根据定义,对一个&T调用clone() 会返回一个T的值,也就是说会返回str或者[i32]。而我们前面学习Sized特型的时候提到过,str或者[i32] 是切片类型,无固定大小,是不能保存在变量中或者作为函数结果返回的。

std::borrow::ToOwned 特型提供了一个稍微宽松的方法将一引用转换为可拥有的值。

trait ToOwned {
  type Owned: Borrow<Self>;
  fn to_owned(&self) -> Self::Owned;
}

上面的定义中,to_owned函数返回一个类型为Self::Owned的新鲜值,但是这个Owned可不是任意类型,它有个类型约束,也就是实现了Borrow<Self>. 也就是说A能借出B/&B(实现了Borrow<B>),B才能拥有A.

例如,你可以从Vec<T>中借出一个&[T](这里的泛型U 为 [T] ),因此[T]可以实现ToOwned<Owned=Vec<T>>,只要T实现了Clone特型。这里为什么要对T限制呢?毕竟你要得到一个备份,如果一个T不能克隆,那么这个备份是无法实现的,因为需要把切片的元素复制到新的向量中去。相似的,str实现了ToOwned<Owned=String>,因此我们可以调用&strto_owned函数得到一个全新的字符串。Path也实现了ToOwned<Owned=PathBuf>,我们也可以从Path引用中得到一个全新的PathBuf值。

Humble Cow

BorrowToOwned联动可以实现一个很有意思的类型,Cow,注意它不是奶牛的意思,而是指 clone on write 我们趁热来学习它。

充分利用 Rust 需要深思熟虑所有权问题,例如某个函数是否应该通过引用或值接收参数。通常你能确定使用其中的一种或者另一种(使用引用还是值),函数的参数类型代表了你的决定。但是存在这样一些场景,你只有在运行时才知道到底是需要借用还是引用,这时,std::borrow::Cow类型就派上用场了,它的定义如下:

enum Cow<'a, B: ?Sized>
  where B: ToOwned
  {
  Borrowed(&'a B),
  Owned(<B as ToOwned>::Owned),
}

这里可以看到,Cow是一个枚举,有两个变量,分别代表借用和拥有。其Borrow变量绑定了一个&B(这里先忽视生命周期标记),这个B是个泛型,它的约束为B: ToOwned。它的目标类型我们先假定为U,那么U 必定实现了Borrow<B>

它的第二个枚举变量为Owned,绑定了一个<B as ToOwned>::Owned的值,也就是U的值,所以Owned变量可以写成Owned<U>。其中可以从U借出B,当然,也可以从B拥有U的新值。

第一个枚举变量,是绑定了&B,因此我们可以很方便的得到&B,第二个变量,是绑定了U,然而U又可以借出B,因此我们仍然可以很容易的得到&B( 通过U的borrow() 函数)。两个变量都可以方便的得到&B,因此它也实现了Deref特型,这样你可以直接在Cow上调用B的相关函数,而不管Cow是借用了B还是拥有了U。

你还可以在Cow类型的值上调用to_mut函数得到一个&mut B。 如果Cow变量刚好好Borrowed,则to_mut函数会先调用&Bto_owned方法得到它自己拥有的一个U的Copy,并对原来的变量进行重新赋值,这样就从Borrowed变量转换成了Owned变量,然后再从新拥有的U的值中借出一个mut 引用。这里正是clone on write的含义所在(写时clone).

我们来看一下这个Deref的实现代码:

#[stable(feature = "rust1", since = "1.0.0")]
impl<B: ?Sized + ToOwned> Deref for Cow<'_, B>
where
    B::Owned: Borrow<B>,
{
    type Target = B;
    fn deref(&self) -> &B {
        match *self {
            Borrowed(borrowed) => borrowed,
            Owned(ref owned) => owned.borrow(),
        }
    }
}

你代码中我们可以看到,如果Cow是引用 ,直接将这个引用返回,如果是拥有的U值,则从U值借出,这里有一个细节:

Owned(ref owned) => owned.borrow(), 因为我们的deref函数接收参数为&self,因此我们无法在函数内部消耗掉Cow本身,而match直接匹配时便会消耗这个值,因此为了阻止这种行为,添加了ref owned,代表这个owned只是获取一个引用 ,因此这里的owned的类型其实为&U,所以直接调用其borrow()函数也就得到了一个&B. 注意borrow函数也是接收一个引用而非值作为参数。

to_mut函数的解释为:

Acquires a mutable reference to the owned form of the data.

Clones the data if it is not already owned.

使用示例为:

use std::borrow::Cow;
let mut cow = Cow::Borrowed("foo");
cow.to_mut().make_ascii_uppercase();
assert_eq!(
  cow,
  Cow::Owned(String::from("FOO")) as Cow<'_, str>
);

通过上面的示例我们可以看到,就算我们的cow是Borrowed变量,拥有一个共享的引用,到最后也变成了一个Owned变量。

我们来看一下实现过程:

#[stable(feature = "rust1", since = "1.0.0")]
pub fn to_mut(&mut self) -> &mut <B as ToOwned>::Owned {
    match *self {
        Borrowed(borrowed) => {
            *self = Owned(borrowed.to_owned());
            match *self {
                Borrowed(..) => unreachable!(),
                Owned(ref mut owned) => owned,
            }
        }
        Owned(ref mut owned) => owned,
    }
}

这里 如果是Borrowed,则首先会to_owned得到U的一个新值,然后再将self重新赋值为Owned,然后再重新对self进行match操作,

此时已经是一个Owned,所以直接借出了ref mut,注意因为match操作会消耗值,所以这里的Owned(ref mut owned) => owned, 中加了ref 代表是一个引用,结合上例,我们就得到了一个&mut String。

相似的,Cow也实现了into_owned方法将引用转换为一个拥有的值。如果必须,则可以将值的所有权转移给调用者,在这个过程中Cow本身的值会被消耗掉。

Cow一个常见的用法是返回一个静态的字符串文字值常量或者一个动态的字符串。例如,假定你需要将一个枚举类型转换成一个消息,枚举的大多数变量都可用于固定的字符串,但是有一些变量或者一些额外的信息,因此你可以返回一个Cow<'static str>

use std::path::PathBuf;
use std::borrow::Cow;
fn describe(error: &Error) -> Cow<'static, str> {
  match *error {
      Error::OutOfMemory => "out of memory".into(),
      Error::StackOverflow => "stack overflow".into(),
      Error::MachineOnFire => "machine on fire".into(),
      Error::Unfathomable => "machine bewildered".into(),
      Error::FileNotFound(ref path) => {
      format!("file not found: {}", path.display()).into()
    }
  }
}

上面的代码使用了CowInto特型实现来构造值。这里其实是CowFrom实现,然后相对应的&str就有了Into实现。这个Match的绝大多数分支都返回一个静态分配的文字串文本用于Cow::Borrowed绑定,只有最后一个分支返回一个String用于Owned变量绑定。

describe函数的调用者不用管返回的到底Cow的哪个变量,它只用简单的将返回值看成是&str就行了,例如:

println!("Disaster has struck: {}", describe(&error));

如果你需要一个拥有的值,调用into_owned函数就可,例如 (describe(&error).into_owned() 就返回一个String

使用Cow可以让describle函数和它的调用者直到在需要时才会分配内存来保存新生成的字符串。

到此这篇关于Rust常用特型之ToOwned特型的文章就介绍到这了,更多相关Rust ToOwned特型内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:

相关文章

  • 一文弄懂rust声明宏

    一文弄懂rust声明宏

    Rust支持两种宏,一种是声明宏,一种是过程宏,本文主要介绍了一文弄懂rust声明宏,通过声明宏可以减少一些样板代码,具有一定的参考价值,感兴趣的可以了解一下
    2024-03-03
  • Rust中FFI编程知识点整理总结(推荐)

    Rust中FFI编程知识点整理总结(推荐)

    这篇文章主要介绍了Rust中FFI编程知识点整理总结,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-09-09
  • Rust中使用Serde对json数据进行反序列化

    Rust中使用Serde对json数据进行反序列化

    JSON作为目前流行的数据格式之一,被大家广泛使用,在日常的开发实践中,将JSON数据反序列化为对应的类型具有重要的意义,在Rust中,Serde几乎成了JSON数据解析的事实标准,本文将给大家介绍Rust中使用Serde对json数据进行反序列化,需要的朋友可以参考下
    2024-01-01
  • 深入探究在Rust中函数、方法和关联函数有什么区别

    深入探究在Rust中函数、方法和关联函数有什么区别

    在 Rust 中,函数、方法和关联函数都是用来封装行为的,它们之间的区别主要在于它们的定义和调用方式,本文将通过一个简单的rust代码示例来给大家讲讲Rust中函数、方法和关联函数区别,需要的朋友可以参考下
    2023-08-08
  • 利用rust编一个静态博客工具

    利用rust编一个静态博客工具

    这篇文章主要为大家详细介绍了如何利用rust编一个静态博客工具,这个静态博客的工具主要是把md文档转为html静态网站/博客,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-12-12
  • 深入了解Rust的生命周期

    深入了解Rust的生命周期

    生命周期指的是引用保持有效的作用域,Rust 的每个引用都有自己的生命周期。本文将通过示例和大家详细说说Rust的生命周期,需要的可以参考一下
    2022-11-11
  • rust zip异步压缩与解压的代码详解

    rust zip异步压缩与解压的代码详解

    在使用actix-web框架的时候,如果使用zip解压任务将会占用一个工作线程,因为zip库是同步阻塞的,想用异步非阻塞需要用另一个库,下面介绍下rust zip异步压缩与解压的示例,感兴趣的朋友一起看看吧
    2024-04-04
  • Rust如何使用Sauron实现Web界面交互

    Rust如何使用Sauron实现Web界面交互

    Sauron 是一个多功能的 Web 框架和库,用于构建客户端和/或服务器端 Web 应用程序,重点关注人体工程学、简单性和优雅性,这篇文章主要介绍了Rust使用Sauron实现Web界面交互,需要的朋友可以参考下
    2024-03-03
  • 一文掌握Rust编程中的生命周期

    一文掌握Rust编程中的生命周期

    在Rust语言中, 每一个引用都有其生命周期, 通俗讲就是每个引用在程序执行的过程中都有其自身的作用域, 一旦离开其作用域, 其生命周期也宣告结束, 值不再有效,这篇文章主要介绍了Rust编程中的生命周期,需要的朋友可以参考下
    2023-11-11
  • Rust中向量的学习笔记

    Rust中向量的学习笔记

    在Rust语言中,向量是一种动态数组类型,可以存储相同类型的元素,并且可以在运行时改变大小,本文就来介绍一下Rust中向量,感兴趣的可以了解一下
    2024-03-03

最新评论