- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
Rust 中可用的少数隐式转换之一是 pointer weakening , 它可以变成 &mut T
进入&T
:
fn just_foo<T>(_: &T) {}
just_foo(&mut vec![1, 2, 3]);
但是,匹配特征时不会发生这种情况。例如,虽然带有引用作为右侧值的 + 运算符是为数字类型实现的,但它们不会接受对同一类型的可变引用:
5 + &mut 5;
(&5) + &mut 5;
错误信息:
error[E0277]: the trait bound `{integer}: std::ops::Add<&mut {integer}>` is not satisfied
--> src/main.rs:38:7
|
38 | 5 + &mut 5;
| ^ no implementation for `{integer} + &mut {integer}`
|
= help: the trait `std::ops::Add<&mut {integer}>` is not implemented for `{integer}`
error[E0277]: the trait bound `&{integer}: std::ops::Add<&mut {integer}>` is not satisfied
--> src/main.rs:43:10
|
43 | (&5) + &mut 5;
| ^ no implementation for `&{integer} + &mut {integer}`
|
= help: the trait `std::ops::Add<&mut {integer}>` is not implemented for `&{integer}`
另一个更有趣的例子是,我添加了 Add
的各种实现。对于单位类型 Foo
:
use std::ops::Add;
#[derive(Debug, Default)]
struct Foo;
impl Add<Foo> for Foo {
type Output = Foo;
fn add(self, _: Foo) -> Foo {
Foo
}
}
impl<'a> Add<&'a Foo> for Foo {
type Output = Foo;
fn add(self, _: &'a Foo) -> Foo {
Foo
}
}
impl<'a, 'b> Add<&'a Foo> for &'b Foo {
type Output = Foo;
fn add(self, _: &'a Foo) -> Foo {
Foo
}
}
才发现我可以执行&Foo + &mut Foo
, 但不是 Foo + &mut Foo
:
&Foo + &mut Foo; // ok
Foo + &mut Foo; // not ok
第二种情况符合上面前面的例子,但第一种情况则不然。似乎 RHS &mut Foo
被迫&Foo
匹配 &Foo + &Foo
的实现.看起来也没有发生其他强制转换,因为 &Foo as Add<&Foo>
的接收类型已经是&Foo
.我也可以扔掉语法糖并获得相同的结果:
(&Foo).add(&mut Foo); // ok
Foo.add(&mut Foo); // not ok
鉴于强制转换,根据 Nomicon,在进行特征匹配时不应该发生,为什么会这样 &Foo + &mut Foo
工作时&i32 + &mut i32
不是吗?是因为 Add
的实现只有一个吗?对于 &Foo
?如果是这样,为什么它会使编译器的行为有所不同?
最佳答案
Is it because there is a single implementation of
Add
for&Foo
?
让我们看看添加此实现后会发生什么:
impl<'b> Add<Foo> for &'b Foo {
type Output = Foo;
fn add(self, _: Foo) -> Foo {
Foo
}
}
现在&Foo + &mut Foo
和 &Foo + &mut &mut Foo
编译失败:
error[E0277]: the trait bound `&Foo: std::ops::Add<&mut Foo>` is not satisfied
--> src/main.rs:39:10
|
39 | &Foo + &mut Foo;
| ^ no implementation for `&Foo + &mut Foo`
|
= help: the trait `std::ops::Add<&mut Foo>` is not implemented for `&Foo`
error[E0277]: the trait bound `&Foo: std::ops::Add<&mut &mut Foo>` is not satisfied
--> src/main.rs:40:10
|
40 | &Foo + &mut &mut Foo;
| ^ no implementation for `&Foo + &mut &mut Foo`
|
= help: the trait `std::ops::Add<&mut &mut Foo>` is not implemented for `&Foo`
所以答案是是。
If so, why does it make the compiler behave differently?
当 Add<T>
有一个适用的实现时(或任何其他通用特征),编译器不需要推断 T
从论点;它已经解决了T
基于那个单一的实现。基本上,就好像这个特征根本不是通用的。因此,也可以应用适用于非泛型参数的强制转换。
关于reference - 为什么 &mut 的引用弱化发生在某些特征方法调用中?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49454023/
Rust 中 *、&、mut、&mut、ref、ref mut 的用法和区别 在 Rust 中,*、ref、mut、& 和 ref mut 是用于处理引用、解引用和可变性的关
这个问题在这里已经有了答案: How to avoid writing duplicate accessor functions for mutable and immutable referenc
我正在尝试使用指向我的结构的指针调用 pthread_join,以便 C 线程可以将结构填充到我指向它的内存中。 (是的,我知道这是非常不安全的..) pthread_join的函数签名: pub u
这个问题在这里已经有了答案: is it possible to filter on a vector in-place? (4 个答案) Is there a way to drain parts
这个问题在这里已经有了答案: is it possible to filter on a vector in-place? (4 个答案) Is there a way to drain parts
以下代码编译失败,因为 MutRef 不是Copy。无法复制,因为&'a mut i32 不是Copy。有什么方法可以让 MutRef 具有与 &'a mut i32 相似的语义吗? 这样做的动机是能
我想转换数组。 例子: func()-> *mut *mut f32; ... let buffer = func(); for n in 0..48000 { buffer[0][n] =
我想对 &mut [u8] 进行一些操作。 在我的测试代码中我有: #[test] fn test_swap_bytes() { let input: &[u8] = b"abcdef";
首先,我不是在问 &mut 之间有什么区别?和 ref mut本身。 我问是因为我想: let ref mut a = MyStruct 与相同 let a = &mut MyStruct 考虑从函数
这个问题在这里已经有了答案: What's the difference between placing "mut" before a variable name and after the ":"
我正在从书中学习 Rust,并且我正在处理第 8 章末尾的练习,但是关于将单词转换为 Pig Latin 的练习我遇到了困难。我想具体看看我是否可以将 &mut String 传递给一个接受 &mut
这段代码: let mut a2 = 99; let b: *mut i32 = &mut a2; *b = 11; // does not compile , even after unsafe {
我正在尝试了解借用检查器。我有一个带有签名的函数 fn SerializeChar(&mut self, value: &mut u8) 我想从 u8 和 i8 获取数据,因为我不关心符号: let
在下面的示例中,t1 编译但 t2 不编译。 &mut &stream有什么特别之处吗?我认为 Deref 不会起作用。 use std::net::TcpStream; fn t1() {
我正在围绕 C 库编写一个 Rust 包装器,同时我正在尝试利用 The Book 中提到的“可空指针优化” , 但我找不到转换 Option 的好方法至 *const T和 Option至 *mut
我试图在 Rust 中实现一个链表,但我在理解这两个函数之间的区别时遇到了一些困难: enum List { Nil, Cons(T, Box>) } fn foo(list: &mu
根据文档,ManuallyDrop是一个零成本包装器。这是否意味着我可以取消引用指向 ManuallyDrop 的原始指针从原始指针转换为 T ? 最佳答案 ManuallyDrop 是 declar
我一直在尝试用 Rust 编写 Redis 模块。这是我第一次尝试使用 Rust FFI 和绑定(bind)。如何在不破坏 Redis 指针的情况下调用此方法并在 Rust 中得到一个数据值? ext
我很难理解以下代码为何具有2种不同的行为: pub fn get(&self, idx: usize) -> &T { let arr = unsafe { core::slice::from
在 documentation为 Unwindsafe我们有: Types such as &mut T and &RefCell are examples which are not unwind
我是一名优秀的程序员,十分优秀!