Rust那些事之Borrow VS AsRef

最近看到两个trait长得挺像,带着疑惑前来学习一下。

Borrow VS AsRef

Borrow与AsRef是Rust中非常相似的两个trait,分别是:

go 复制代码
pub trait Borrow<Borrowed: ?Sized> {
    fn borrow(&self) -> &Borrowed;
}

pub trait AsRef<T: ?Sized> {
    fn as_ref(&self) -> &T;
}

可以看到这两个从接口层面是一样的。

使用方式如下:

go 复制代码
let s = String::from("Hello World");
let s_ref: &str = s.borrow(); // using Borrow
let s_ref: &str = s.as_ref(); // using AsRef

那么问题来的,什么时候使用Borrow,什么时候使用AsRef呢?

1.trait一致性

  • Borrow also requires that Hash, Eq and Ord for a borrowed value are equivalent to those of the owned value. For this reason, if you want to borrow only a single field of a struct you can implement AsRef, but not Borrow.

Borrow 还要求对于借用值的HashEqOrd 与拥有值相等。因此,如果你只想借用结构体的单个字段,你可以实现 AsRef,但不能实现 Borrow

如果类型T1实现了Borrow<T2>,在为T1实现一些特殊的trait(例如:Eq、Ord、Hash)时,其行为应该与T2有相同的行为。

例如:标准库 HashMap 对 get() 和 get_mut() 方法使用了Borrow 特征。K是HashMap的key,而get接收的是k类型是&Q

例如:HashMap<String, xx>,K就是String,Q就是str,这允许我们可以传递任何可以作为 K 借用的类型 Q。通过额外要求 Q: Hash + Eq,它表明要求 K 和 Q 具有产生相同结果的 Hash 和 Eq trait的实现。

go 复制代码
impl<K, V> HashMap<K, V> {
    pub fn get<Q>(&self, k: &Q) -> Option<&V>
    where
        K: Borrow<Q>,
        Q: Hash + Eq + ?Sized
    {
        // ...
    }
}

https://doc.rust-lang.org/std/borrow/trait.Borrow.html

2.默认实现

  • Unlike AsRef, Borrow has a blanket impl for any T, and can be used to accept either a reference or a value. (See also note on AsRef's reflexibility below.)

AsRef 不同,Borrow 为任何 T 都有一个默认实现,并且可以用来接受引用或值。

例如:使用Borrow下面代码可以正常工作。

go 复制代码
use std::borrow::Borrow;

fn print_value<T: Borrow<i32>>(value: T) {
    println!("Value is: {}", value.borrow());
}

fn main() {
    let number = 42;
    print_value(&number); 
    print_value(number); 
}

而如果使用AsRef,则不可以,报错:

go 复制代码
the trait `std::convert::AsRef<i32>` is not implemented for `{integer}`

示例:

go 复制代码
fn print_value<T: AsRef<i32>>(value: T) {
    println!("Value is: {}", value.as_ref());
}

fn main() {
    let number = 42;
    print_value(&number); 
    print_value(number); 
}

https://doc.rust-lang.org/std/convert/trait.AsRef.html#relation-to-borrow


往期回顾:

我的春招求职面经

热度更新,手把手实现工业级线程池

相关推荐
AdSet聚合广告2 小时前
APP广告变现,开发者如何判断对接的广告SDK安全合规?
大数据·后端·算法·安全·uni-app
不二狗2 小时前
每日算法 -【Swift 算法】实现回文数判断!
开发语言·算法·swift
csdn_aspnet3 小时前
Java 程序求圆弧段的面积(Program to find area of a Circular Segment)
java·开发语言
野犬寒鸦3 小时前
Redis核心数据结构操作指南:字符串、哈希、列表详解
数据结构·数据库·redis·后端·缓存·哈希算法
无敌小肥0074 小时前
Springboot 整合 WebSocket 实现聊天室功能
spring boot·后端·websocket
进击的_鹏4 小时前
【C++】红黑树的实现
开发语言·c++
hdsoft_huge4 小时前
VUE npm ERR! code ERESOLVE, npm ERR! ERESOLVE could not resolve, 错误有效解决
vue.js·rust·npm
无心水5 小时前
【后端高阶面经:MongoDB篇】41、MongoDB 是怎么做到高可用的?
java·开发语言·mongodb·java面试·高可用·后端高阶面经·后端工程师的高阶面经
无心水5 小时前
【后端高阶面经:MongoDB篇】40、怎么优化MongoDB的查询性能?
java·开发语言·mongodb·java面试·后端高阶面经·后端工程师的高阶面经·java高阶面经
信息化未来5 小时前
python 生成复杂表格,自动分页等功能
开发语言·数据结构·python