Ursprüngliche Diskussion: https://users.rust-lang.org/t/unexpected-behaviors-of-trait-bounds/12286/10
Sieht so aus, als ob es mit diesem alten Problem zusammenhängt: https://github.com/rust-lang/rust/issues/29859
Grundsätzlich erfordert trait Complete
trait Partial
, und trait Partial
erfordert trait PartialEq
(oder ein anderes Merkmal), einen Typ ( struct TypeB
) mit impl Complete
muss nicht impl PartialEq
erfüllen, was nur wie ein Fehler aussieht.
Interessanterweise wird immer dann, wenn <TypeB as Partial>
angezeigt wird, die Prüfung durchgeführt und ein entsprechender Fehler ausgegeben.
Hier ist ein Repro:
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`
*/
}
Hier ist ein minimierter Repro in ICE-Form:
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
Anscheinend hat @ arielb1 auch seine eigene Ausgabe (# 43784) dafür geöffnet. Wie auch immer, dies wird durch # 43786 behoben (sollte es auch sein).
Ich glaube, dies ist ein Dup von # 43784, der schließt.
Hilfreichster Kommentar
Hier ist ein minimierter Repro in ICE-Form: