Trait core::cmp::PartialEq 1.0.0[−][src]
pub trait PartialEq<Rhs: ?Sized = Self> { #[must_use] fn eq(&self, other: &Rhs) -> bool; #[must_use] fn ne(&self, other: &Rhs) -> bool { ... } }
Expand description
等值比较器的 Trait 为 部分等价关系。
x.eq(y)
也可以写成 x == y
,x.ne(y)
可以写成 x != y
。
我们在本文档的其余部分使用更易于阅读的中缀符号。
trait 允许部分相等,用于不具有完全等价关系的类型。
例如,在浮点数 NaN != NaN
中,因此浮点类型实现 PartialEq
,但不实现 Eq
。
实现必须确保 eq
和 ne
彼此一致:
a != b
当且仅当!(a == b)
(由默认实现确保)。
如果 Self
和 Rhs
也实现了 PartialOrd
或 Ord
,则它们的方法也必须与 PartialEq
一致 (具体要求请参见那些 traits 的文档)。
通过派生一些 traits 并手动实现其他一些行为,很容易使它们不以为然。
等式关系 ==
必须满足以下条件 (对于所有类型为 A
、B
、C
的 a
、b
、c
) :
-
对称: 如果
A: PartialEq<B>
和B: PartialEq<A>
,则 a == b意味着 'b == a
; 和 -
可传递: 如果
A: PartialEq<B>
和B: PartialEq<C>
以及A: 偏等式 <C>
,然后a == b
,而b == c
表示a == c
。
请注意,B: PartialEq<A>
(symmetric) 和 A: PartialEq<C>
(transitive) 强制不是强制存在的,但是这些要求只要存在就适用。
Derivable
该 trait 可以与 #[derive]
一起使用。在结构体上 derive
d 时,如果所有字段都相等,则两个实例相等; 如果任何字段不相等,则两个实例不相等。对枚举进行派生时,每个成员都等于自己,而不等于另一个成员。
如何实现 PartialEq
?
一个域的示例实现,在该域中,即使两本书的 ISBN 匹配,即使格式不同,也将其视为同一本书:
enum BookFormat { Paperback, Hardback, Ebook, } struct Book { isbn: i32, format: BookFormat, } impl PartialEq for Book { fn eq(&self, other: &Self) -> bool { self.isbn == other.isbn } } let b1 = Book { isbn: 3, format: BookFormat::Paperback }; let b2 = Book { isbn: 3, format: BookFormat::Ebook }; let b3 = Book { isbn: 10, format: BookFormat::Paperback }; assert!(b1 == b2); assert!(b1 != b3);Run
如何比较两种不同的类型?
您可以比较的类型由 PartialEq
的类型参数控制。
例如,让我们对之前的代码进行一些调整:
// 衍生工具 <BookFormat> == <BookFormat> 比较 #[derive(PartialEq)] enum BookFormat { Paperback, Hardback, Ebook, } struct Book { isbn: i32, format: BookFormat, } // 实现 <Book> == <BookFormat> 比较 impl PartialEq<BookFormat> for Book { fn eq(&self, other: &BookFormat) -> bool { self.format == *other } } // 实现 <BookFormat> == <Book> 比较 impl PartialEq<Book> for BookFormat { fn eq(&self, other: &Book) -> bool { *self == other.format } } let b1 = Book { isbn: 3, format: BookFormat::Paperback }; assert!(b1 == BookFormat::Paperback); assert!(BookFormat::Ebook != b1);Run
通过将 impl PartialEq for Book
更改为 impl PartialEq<BookFormat> for Book
,我们可以将 BookFormat 和 Book 进行比较。
像上面这样的比较 (它忽略了结构体的某些字段) 可能很危险。这很容易导致意外违反部分对等关系的要求。
例如,如果我们保留了以上针对 BookFormat
的 PartialEq<Book>
的实现,并为 Book
添加了 PartialEq<Book>
的实现 (通过 #[derive]
或第一个示例中的手动实现),则结果将违反传递性:
#[derive(PartialEq)] enum BookFormat { Paperback, Hardback, Ebook, } #[derive(PartialEq)] struct Book { isbn: i32, format: BookFormat, } impl PartialEq<BookFormat> for Book { fn eq(&self, other: &BookFormat) -> bool { self.format == *other } } impl PartialEq<Book> for BookFormat { fn eq(&self, other: &Book) -> bool { *self == other.format } } fn main() { let b1 = Book { isbn: 1, format: BookFormat::Paperback }; let b2 = Book { isbn: 2, format: BookFormat::Paperback }; assert!(b1 == BookFormat::Paperback); assert!(BookFormat::Paperback == b2); // 以下应该通过传递性来保持,但不是。 assert!(b1 == b2); // <-- PANICS }Run
Examples
let x: u32 = 0; let y: u32 = 1; assert_eq!(x == y, false); assert_eq!(x.eq(&y), false);Run