在JAVA中更改HashSet中对象的属性值的问题[重复]

slsn1g29  于 2024-01-05  发布在  Java
关注(0)|答案(3)|浏览(206)

此问题在此处已有答案

Java HashSet contains duplicates if contained element is modified(7个答案)
6天前关闭
我正在学习java,遇到了一个非常奇怪的问题,我想用我的代码来解释这个沿着会更容易一些
这是我的班级:

  1. class Node
  2. {
  3. private int val;
  4. public Node(int val)
  5. {
  6. this.val = val;
  7. }
  8. public int getVal()
  9. {
  10. return this.val;
  11. }
  12. public void setVal(int newVal)
  13. {
  14. this.val = newVal;
  15. }
  16. @Override
  17. public int hashCode() {
  18. return this.val;
  19. }
  20. @Override
  21. public boolean equals(Object obj) {
  22. System.out.println("Inside equal of " + this.val);
  23. //i expect it will print out when i use set.contains()
  24. if(this.hashCode() == ((Node)obj).hashCode())
  25. {
  26. return true;
  27. }
  28. return false;
  29. }
  30. }

字符串
这是我使用HashSet的主要块,它与Node类一起工作

  1. public class Main
  2. {
  3. public static void main(String[] args)
  4. {
  5. Set<Node> s = new HashSet<>();
  6. Node n1 = new Node(1);
  7. s.add(n1);
  8. /*as i know, when we add a object to the set, we actually add a refference to the origin memory area (the one that n1 points to) to the set*/
  9. if(s.contains(n1))
  10. {
  11. System.out.println("YES");
  12. }
  13. for(Node i : s)
  14. {
  15. i.setVal(5); //i expect n1 to change too and it does
  16. }
  17. for(Node i : s)
  18. {
  19. System.out.println(i.getVal());
  20. }
  21. System.out.println(n1.getVal()); //n1 changes
  22. if(s.contains(n1))
  23. {
  24. System.out.println("Still here");
  25. //This is where i don't understand why this does not print ?
  26. }
  27. System.out.println();
  28. }
  29. }


这是我的命令

  1. D:\Desktop\draft\JavaDraft>java Main
  2. YES
  3. 5
  4. 5


我不明白为什么set没有意识到n1仍然在其中,为什么Node类的函数“equals”中的代码没有被触发,因为据我所知,HashSet使用“equals”来获得所有元素的唯一性,对吗?提前谢谢你。

dsekswqp

dsekswqp1#

HashSet使用“equals”来获得所有元素的唯一性,对吗?
不是
HashSet是HashMap的一个facade,其中的项存储在键集中。HashMap创建并比较来自其内部哈希节点的键hashCode。
一旦项目存储在其hashCode数组元素下,它就在那里。更改hashCode不会替换它。你写的是一个不好的例子如何在HashMap中使用hashCode。
contains方法将检查底层hashmap键集的键(哈希数组),而不是调用所有存储对象的equals方法。

bfhwhh0e

bfhwhh0e2#

这是因为通过修改Node的值,你也改变了hashcode(由于你的hashcode覆盖)。这通常不是这种情况,也不应该是这种情况,因为你不希望hashcode是可变的。它应该是常量和不变的。
HashSet的底层代码使用HashMap,contains()使用的内部方法将使用对象的hashcode来确定存储节点的bucket。因此,当您的hashcode设置为值1时,您调用add(),节点n1存储在bucket 1中。当您将值更改为5时,hashcode现在是5,调用s.contains(n1)将尝试检查bucket 5中的节点是否为空(null)。
正确的方法是删除你实现的hashcode覆盖,让默认实现完成它的工作。另请参阅:

  1. import java.util.*;
  2. public class Main {
  3. public static void main(String[] args)
  4. {
  5. Set<Node> s = new HashSet<>();
  6. Node n1 = new Node(1);
  7. System.out.println(n1);
  8. s.add(n1);
  9. /*as i know, when we add a object to the set, we actually add a refference to the origin memory area (the one that n1 points to) to the set*/
  10. if(s.contains(n1))
  11. {
  12. System.out.println("YES");
  13. }
  14. for(Node i : s)
  15. {
  16. i.setVal(5); //i expect n1 to change too and it does
  17. }
  18. for(Node i : s)
  19. {
  20. System.out.println(i.getVal());
  21. }
  22. System.out.println(n1.getVal()); //n1 changes
  23. System.out.println(n1);
  24. if(s.contains(n1))
  25. {
  26. System.out.println("Still here");
  27. //This is where i don't understand why this does not print ?
  28. }
  29. System.out.println();
  30. }
  31. }
  32. class Node
  33. {
  34. private int val;
  35. public Node(int val)
  36. {
  37. this.val = val;
  38. }
  39. public int getVal()
  40. {
  41. return this.val;
  42. }
  43. public void setVal(int newVal)
  44. {
  45. this.val = newVal;
  46. }
  47. @Override
  48. public boolean equals(Object obj) {
  49. System.out.println("Inside equal of " + this.val);
  50. //i expect it will print out when i use set.contains()
  51. if(this.hashCode() == ((Node)obj).hashCode())
  52. {
  53. return true;
  54. }
  55. return false;
  56. }
  57. }

字符串
输出量:
没有哈希码覆盖

  1. Node@4783da3f
  2. YES
  3. 5
  4. 5
  5. Node@4783da3f
  6. Still here


使用Hashcode override

  1. Node@1
  2. YES
  3. 5
  4. 5
  5. Node@5

展开查看全部
fxnxkyjh

fxnxkyjh3#

你需要理解按引用传递和按值传递的区别。你对hashcode的重写实际上是有效的。但是,你没有创建一个新对象,n1和s中的对象实际上是同一个对象。

相关问题