- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
在this MSDN页面上写着:
Warning:
If you override the GetHashCode method, you should also override Equals, and vice versa. If your overridden Equals method returns true when two objects are tested for equality, your overridden GetHashCode method must return the same value for the two objects.
我也看到过许多类似的建议,我可以理解在重写 Equals 方法时我也想重写 GetHashCode。据我所知,GetHashCode 与哈希表查找一起使用,这与相等性检查不同。
下面是一个例子来帮助解释我想问的问题:
public class Temperature /* Immutable */
{
public Temperature(double value, TemperatureUnit unit) { ... }
private double Value { get; set; }
private TemperatureUnit Unit { get; set; }
private double GetValue(TemperatureUnit unit)
{
/* return value converted into the specified unit */
}
...
public override bool Equals(object obj)
{
Temperature other = obj as Temperature;
if (other == null) { return false; }
return (Value == other.GetValue(Unit));
}
public override int GetHashCode()
{
return Value.GetHashCode() + Unit.GetHashCode();
}
}
在这个例子中,两个温度对象被认为是相等的,即使它们在内部存储的不是相同的东西(例如 295.15 K == 22 摄氏度)。目前,GetHashCode 方法将为每个方法返回不同的值。这两个温度对象是相等的,但它们也不相同,所以它们具有不同的哈希码是不是不正确?
最佳答案
在哈希表中存储值时,例如 Dictionary<>
, 框架会先调用 GetHashCode()
并检查哈希表中是否已经有一个bucket 用于该哈希码。如果有,它将调用 .Equals()
查看新值是否确实等于现有值。如果不是(意味着两个对象不同,但产生相同的散列码),您就会遇到所谓的冲突。在这种情况下,这个桶中的项目被存储为一个链表并且检索某个值变成了 O(n)。
如果你实现了 GetHashCode()
但没有实现Equals()
,该框架将求助于使用引用相等性来检查相等性,这将导致每个实例产生冲突。
如果你实现了 Equals()
但没有实现GetHashCode()
,您可能会遇到这样的情况,您有两个相等的对象,但产生不同的哈希码,这意味着它们将在您的哈希表中维护自己独立的值。这可能会使使用您的类(class)的任何人感到困惑。
至于哪些对象被视为相等,这取决于您。如果我创建一个基于温度的哈希表,我是否可以使用它的摄氏度或华氏度值来引用同一项目?如果是这样,它们需要产生相同的散列值和 Equals()
需要返回 true。
更新:
让我们退后一步,首先看一下哈希码的用途。在此上下文中,哈希码用作识别两个对象是否最有可能相等的快速方法。如果我们有两个具有不同哈希码的对象,我们就可以确定它们不是相等的。如果我们有两个具有相同哈希码的对象,我们就知道它们很可能是相等的。我说最有可能是因为 int 只能用来表示几十亿个可能的值,而字符串当然可以包含 Charles Dickens 的全集,或者任意数量的可能值。 .NET 框架中的很多内容都基于这些事实,使用您的代码的开发人员将假定一切以与框架的其余部分一致的方式工作。
如果您有两个具有不同哈希码的实例,但具有 Equals()
的实现返回真,你打破了这个约定。比较两个对象的开发人员可能会使用其中一个对象来引用哈希表中的键,并期望得到一个现有值。如果哈希码突然不同,则此代码可能会导致运行时异常。或者可能返回对完全不同对象的引用。
295.15k 和 22C 在您的程序域内是否相等是您的选择(在我看来,它们不是)。但是,无论您决定如何,相等的对象必须返回相同的代码。
关于c# - 为什么 GetHashCode 应该实现与 Equals 相同的逻辑?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34818069/
当我尝试加载库 Raster 时,我收到如下错误: 错误:inDL(x, as.logic(local), as.logic(now), ...) 中的“raster”的包或命名空间加载失败:无法加载
当我尝试加载库 Raster 时,我收到如下错误: 错误:inDL(x, as.logic(local), as.logic(now), ...) 中的“raster”的包或命名空间加载失败:无法加载
望着help section about_Comparison_Operators of PowerShell我是这样理解的: PS C:\> $false,$false -eq $true PS C
我刚刚修改了旧代码,现在似乎没有任何效果。请您指导我哪里出错了。 一些不起作用的事情是: 以前,焦点始终停留在屏幕上唯一的输入字段上。 (现在不行了),代码中的 if else 条件也不起作用。 On
请帮我找到一个使用普通 'ol javascript 的解决方案(我无法使用外部框架)。此外,CSS :hover 选择器不适用于现实世界的实现。 注册事件发生的事情设置所有调用最后注册事件数组项。
我想创建一个软件来为残障 child 交通规划公交路线(及其最佳载客量)。 这些总线具有以下规范: m 个座位(最多 7 个 - 因为有司机和助理) o 轮椅“座位”(最多 4 个) 固定的最大负载量
有人能帮我吗?似乎我的 for 逻辑根本不起作用,因为它一直在上午 12:00 返回我的开始时间 这是我的代码 Sub forlogic() Dim i As Single Dim t
我正在尝试设置 OR两个切片器过滤器之间的逻辑。两个切片器来自相同的数据集。以下是更多详细信息: 我的源表: 带切片器的视觉效果: 我的目标是,如果我从切片器 1 和切片器 2 中选择任何值,我的视觉
我有以下 C 语句: int res = x & (x ^ y); 有没有办法做同样的事情,但每次只使用一次x和y? 例如: x | (~x & y) == x | y 最佳答案 是的,通过扩展 xo
我正在创建 Azure 逻辑应用程序以将新的 Sharepoint 文件添加到 Azure Blob。 Sharepoint 由我的公司运行,我使用我的凭据登录来为逻辑应用程序创建 Sharepoin
我有一个问题要求为给定函数合成最简单的乘积表达式总和。基本上,如果 AB == CD,则函数为 1,否则为 0,结果如下: (!A && !B && !C && !D) || (!A && B &&
我正在尝试确定是否可以在不溢出的情况下计算两个 32 位整数的总和,同时仅使用某些按位运算符和其他运算符。因此,如果整数 x 和 y 可以相加而不会溢出,则以下代码应返回 1,否则返回 0。 ((((
处理乍一看需要许多嵌套 if 语句的复杂业务逻辑的好方法是什么? 例子: 折扣券。可能: 1a) 超值折扣 1b) 百分比折扣 2a) 正常折扣 2b) 累进折扣 3a) 需要访问优惠券 3b) 不需
假设我有一个“numbers”对象数组,其中包含“startNo”整数和“endNo”整数。 数组中可以有多个“数字”,我想获取一个包含修改对象的新数组,该数组仅具有不重叠的范围。 例如:如果数组有:
我在这个问题上遇到了困难。我正在使用 JavaScript。 我有一个文本区域,用于检测 @ 输入并将其位置存储在数组中。 var input = "@a @b @c" //textarea var
默认 IN 使用 OR 基本逻辑。有没有办法在范围内使用 AND 基本逻辑。 例如下面的查询 SELECT ItemId,CategoryID FROM ItemCategories WHERE Ca
我想在您将鼠标悬停在网站图像上时添加叠加层。我在这里实现了这个,它工作正常http://jsfiddle.net/stujLbjh/ 这是js代码: var divs = document.query
这个问题在这里已经有了答案: Which is faster: x>2 是否比 x>>31 快?换句话说,sar x, 2 是否比 sar x, 31 快?我做了一些简单的测试,他们似乎有相同的速度
我有grails criteriaQuery,我在这里再次检查OR逻辑,就像这样一个状态变量: or { eq("status", Status.ONE) eq("status",
我有grails criteriaQuery,我在这里再次检查OR逻辑,就像这样一个状态变量: or { eq("status", Status.ONE) eq("status",
我是一名优秀的程序员,十分优秀!