Исходное обсуждение: https://users.rust-lang.org/t/unexpected-behaviors-of-trait-bounds/12286/10
Похоже, это связано с этой старой проблемой: https://github.com/rust-lang/rust/issues/29859
По сути, для trait Complete
требуется trait Partial
, а для trait Partial
требуется trait PartialEq
(или любой другой признак), тип ( struct TypeB
) с impl Complete
не требует удовлетворения impl PartialEq
, что только выглядит как ошибка.
Интересно, что всякий раз, когда отображается <TypeB as Partial>
, выполняется проверка и выдается соответствующая ошибка.
Вот репродукция:
pub trait Partial: PartialEq {
fn foo() -> Option<bool>;
}
pub trait Complete: Partial {
fn foo() -> bool;
}
impl<T> Partial for T
where
T: Complete,
{
fn foo() -> Option<bool> {
Some(<Self as Complete>::foo())
}
}
// ----
#[derive(PartialEq)]
pub struct TypeA {}
impl Partial for TypeA {
fn foo() -> Option<bool> {
None
}
}
// ----
// BUG: No compile warning about `PartialEq` not being implemented
// #[derive(PartialEq)]
pub struct TypeB {}
impl Complete for TypeB {
fn foo() -> bool {
true
}
}
// ----
pub fn main() {
println!("{:?}", TypeA::foo());
// This works, but shouldn't!
println!("{:?}", <TypeB as Complete>::foo());
// This would trigger the issue, though.
//println!("{:?}", <TypeB as Partial>::foo());
/* Result in:
error[E0277]: the trait bound `TypeB: std::cmp::PartialEq` is not satisfied
--> src/bin/cyclic_traits.rs:48:22
|
48 | println!("{:?}", <TypeB as Partial>::foo());
| ^^^^^^^^^^^^^^^^^^^^^^^ can't compare `TypeB` with `TypeB`
|
= help: the trait `std::cmp::PartialEq` is not implemented for `TypeB`
= note: required by `Partial::foo`
*/
}
Вот уменьшенное изображение в форме ICE:
trait Base {
fn base() {}
}
trait Partial: Base {}
trait Complete: Partial {}
impl<T: Complete> Partial for T {}
struct TypeB;
impl Complete for TypeB {}
fn main() {
ice::<TypeB>();
}
fn ice<P: Partial>() {
P::base();
}
error: internal compiler error: /checkout/src/librustc/traits/trans/mod.rs:75: Encountered error `Unimplemented` selecting `Binder(<TypeB as Base>)` during trans
note: the compiler unexpectedly panicked. this is a bug.
note: we would appreciate a bug report: https://github.com/rust-lang/rust/blob/master/CONTRIBUTING.md#bug-reports
note: rustc 1.21.0-nightly (cbbe17aa7 2017-08-07) running on x86_64-unknown-linux-gnu
Похоже, @ arielb1 также открыл для этого свой собственный выпуск (# 43784). Во всяком случае, это исправлено (должно быть) # 43786.
Я считаю, что это дубликат № 43784, закрытие.
Самый полезный комментарий
Вот уменьшенное изображение в форме ICE: