首先set也是Collection接口的子接口,即:
|---List
有序(存储顺序和获取顺序一致),可重复
|---Set
|---HashSet(通过HashCode和equlas保证唯一性)
|---TreeSet(通过comparable的compareTo或者comparator的compare)
无序(存储顺序和获取顺序不一致),不可重复
先从HashSet开始,它是不可重复的,如果存入String,或者Integer,那么java会对其默认不可重复,不需要我们再做研究,但是如果我们
要存入HashSet集合中的是自定义的类呢?
例如:
package collection;
import java.util.HashSet;
import java.util.Set;
public class SetDemo1 {
public static void main(String[] args) {
fun2();
}
// 对自定义类元素确保元素唯一性
private static void fun2() {
// 创建集合
Set s = new HashSet();
// 创建Student对象
Student s1 = new Student("a", 1);
Student s2 = new Student("b", 2);
Student s3 = new Student("c", 3);
Student s4 = new Student("a", 1);
//添加元素
s.add(s1);
s.add(s2);
s.add(s3);
s.add(s4);
// 遍历集合
for (Student st : s) {
System.out.println(st.getName() + "--" + st.getAge());
}
}
private static void fun1() {
// 创建set集合
Set s = new HashSet();
// 添加元素
s.add("a");
s.add("b");
s.add("c");
s.add("d");
s.add("a");
// 遍历
for (String str : s) {
System.out.print(str);
}
}
}
// 定义一个Student类,包含姓名年龄
class Student {
private String name;
private int age;
public Student(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
fun1():
结果:dbca
总结:如果是String、Integer之类的元素,java会自动把重复的替换而不是重复添加到集合中
fun2():
结果: c--3
a--1
b--2
a--1
总结:如果添加的是自定义类的元素,那么java则不会自动把重复的替换,即不能保证set集合的唯一性,这时就需要自己来通过
同过API,我们知道HashSet底层是由哈希表实现的,那么保证它唯一性的就是元素的HashCode和equals方法,即需要在Stduent类中重写
HashCode和equals方法。
例如:
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + age;
result = prime * result + ((name == null) ? 0 : name.hashCode());
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
Student other = (Student) obj;
if (age != other.age)
return false;
if (name == null) {
if (other.name != null)
return false;
} else if (!name.equals(other.name))
return false;
return true;
}
注意补充:
为什么重写了equals()还需要重写HashCode()?
原因是因为在java源码中,equals()是在Hash值相同的时候才能调用到equals(),所以一定要重写HashCode(),否则Hash值都不一样也永远不可能进入到equals()中去。