例如,存元素的顺序是11、22、33。那么集合中,元素的存储就是按照11、22、33的顺序完成的)
它是一个 带有索引 的集合,通过索引就可以精确的操作集合中的元素(与数组的索引是一个道理)
与 set 不同,List 中 可以有重复的元素,通过元素的 equals
方法,来比较是否为重复的元素
List 接口的常用实现类有:
ArrayList 集合
LinkedList 集合
add(int index, Object e)
: 向集合指定索引处,添加指定的元素,原元素依次 后移remove(int index)
:将指定索引处的元素,从集合中删除,返回值为被删除的元素set(int index,E element)
:将指定索引处的元素,替换成指定的元素,返回值为替换前的元素get(int index)
:获取指定索引处的元素,并返回该元素List<Person> a = new LinkedList<>();
a.add(new Person("zs",10));
a.add(new Person("lisi",20));
a.add(new Person("wangwu",30));
//利用Iterator来遍历List
Iterator<Person> iterator = a.iterator();
//利用迭代器对象遍历
while(listItr.hasNext()) {
System.out.println(listItr.next());
}
//List独有的遍历方式
for (int i = 0; i < a.size(); i++) {
System.out.println(a.get(i));
}
ListIterator
add(E, e)
:将指定元素插入列表boolean hasPrevious()
:逆向遍历列表,若列表迭代器有多个元素,则返回 true,也就是判断是否有前一个元素previous()
:返回列表的前一个元素Iterator<Person> iterator = a.iterator();
ListIterator<Person> listItr = a.listIterator(a.size());
//先顺序遍历,让 cursor 到最后
while(listItr.hasNext()) {
System.out.println(listItr.next());
}
//逆向遍历
//先previous向前移动一个位置,再访问cursor指向的元素
while(listItr.hasPrevious()) {
System.out.println(listItr.previous());
}
java.util.ConcurrentModificationException
首先,在遍历集合的过程中修改集合;其次,修改集合行为,不是迭代器对象来完成的,而是直接修改 Collection 对象
//场景实现
List<String> list = new ArrayList<String>();
list.add("abc1");
list.add("abc2");
list.add("abc3");
list.add("abc4");
//对集合使用迭代器进行获取,获取时候判断集合中是否存在 "abc3"对象
//如果有,添加一个元素 "ABC3"
Iterator<String> it = list.iterator();
while(it.hasNext()){
String s = it.next();
//对获取出的元素s,进行判断,是不是有"abc3"
if(s.equals("abc3")){
list.add("ABC3");
}
System.out.println(s);
}
在迭代过程中,使用了集合的方法对元素进行操作。导致迭代器并不知道集合中的变化,容易引发数据的不确定性。迭代器对象,是依赖与当前的数据集合产生的(换言之,迭代器依赖于数据集,它们必须对应)
public class ListDemo{
public static void main(String[] args) {
List<Person> a = new LinkedList<>();
//迭代器
ListIterator<Person> listItr = a.listIterator();
a.add(new Person("zs",10));
a.add(new Person("lisi",20));
a.add(new Person("wangwu",30));
while(listItr.hasNext()){
Person p = listItr.next();
//这种添加元素的方式,会产生异常
//a.add(new Person("zhaoliu", 40));
//解决: 利用ListIterator对象添加元素
listItr.add(new Person("zhaoliu", 40));
}
}
System.out.println(s);
//针对List还有另外一种,在遍历集合同时修改集合的解决方案
for (int i = 0; i < a.size(); i++){
if("lisi".equals(a.get(i).name)){
a.add(new Person("zhaoliu", 40));
}
}
System.out.println(a);
//如果使用 ListIterator 的add方法向集合中添加元素,这个元素的位置处在当前遍历到的元素之后的位置
//如果使用 集合对象的 add(e) 方法添加元素,插入的元素处在表尾位置
}
//创建了一个长度为0的Object类型数组
ArrayList al=new ArrayList();
al.add("abc");
//本质:
//底层会创建一个长度为10的Object数组 Object[] obj=new Object[10]
//obj[0]="abc"
//如果添加的元素的超过10个,底层会开辟一个1.5*10的长度的新数组
//把原数组中的元素拷贝到新数组,再把最后一个元素添加到新数组中
addFirst(E) 添加到链表的开头
addLast(E) 添加到链表的结尾
E getFirst() 获取链表的开头
E getLast() 获取链表的结尾
E removeFirst() 移除并返回链表的开头
E removeLast() 移除并返回链表的结尾
它是个 不包含重复元素 的集合,没索引
是一个不包含重复元素的 collection
无序集合,没有索引,不存储重复元素
Set无序:存储和取出的顺序不同,
Set集合取出元素的方式可以采用:迭代器、增强for
代码的编写上,和 ArrayList 完全一致
Set集合常用实现类:
HashSet 集合
LinkedHashSet 集合
存入元素数量 > 哈希表长度 * 加载因子
,就要扩容,因此加载因子决定扩容时机public int hashCode()
来计算,计算结果 int 整数hashCode()
方法,见源码存取原理:
每存入一个新的元素都要走以下三步:
1. 首先调用本类的hashCode()
方法算出哈希值
2. 在容器中找是否与新元素哈希值相同的老元素,如果没有直接存入,如果有转到第三步
3. 新元素会与该索引位置下的老元素利用equals 方法
一一对比,一旦新元素.equals(老元素)
,返回 true,停止对比,说明重复,不再存入,如果与该索引位置下的老元素都通过 equals 方法对比返回 false,说明没有重复,存入
//重写hachCode() 方法
public int hashCode(){
return name.hashCode() +age +id; //算法为name的hashCode值+age+id
}
//重写equals
public boolean equals(Object obj){
if(this == obj)
return true;
if(obj == null)
return false;
if(obj instanceof Student){
Student s = (Student)obj;
return name.equals(s.name) && age == s.age && id == s.id;
}
return false;
}
LinkedHashSet
基于链表的哈希表实现,继承自 HashSet
LinkedHashSet 自身特性:
具有顺序,存储和取出的顺序相同的,线程不安全,运行速度块
public class LinkedHashSetDemo {
public static void main(String[] args) {
LinkedHashSet<Integer> link = new LinkedHashSet<Integer>();
link.add(123);
link.add(44);
link.add(33);
link.add(33);
System.out.println(link);
}
}
contains()
原理:底层依赖于equals()
ArrayList的 contains方法 调用时,传入的元素的调用 equals方法 依次与集合中的旧元素所比较,从而根据返回的布尔值判断是否有重复元素。
此时,当ArrayList存放自定义类型时,由于自定义类型在未重写equals方法前, 判断是否重复的依据是地址值,所以如果想根据内容判断是否为重复元素,需要重写元素的 equals方法。
add()
和 contains()
底层都依赖hashCode()
与 equals()
Set集合不能存放重复元素,其添加方法在添加时会判断是否有重复元素,有重复不添加,没重复则添加。
HashSet集合由于是无序的,其判断唯一的依据是元素类型的hashCode与equals方法的返回结果。规则如下:1. 先判断新元素与集合内已经有的旧元素的HashCode值
2. 如果不同,说明是不同元素,添加到集合。
3. 如果相同,再判断equals比较结果。返回true则相同元素;返回false则不同元素,添加到集合。
所以,使用HashSet存储自定义类型,如果没有重写该类的 hashCode()与equals(),则判断重复时,使用的是地址值,如果想通过内容比较元素是否相同,需要重写该元素类的 hashcode()与equals()。
Set 的另外一种实现,底层由 红黑树 实现;也就是说TreeSet会根据元素的大小关系,将元素默认从小到大排列
特点:
元素无序(迭代或者存储顺序和插入顺序)
不能存储重复元素
没有位序
Comparator comparator();如果TreeSet采用了定制排序,则该方法返回定制排序所使用 Comparator;如果TreeSet采用了自然排序,则返回null;
java.lang.ClassCastException
现象:直接向一个 TreeSet 中放入自定义类型的对象,发现直接抛出异常
原因:TreeSet 不知道如何对自定义的类对象进行排序,不像字符串可以根据字典顺序
通过某种方式告诉 TreeSet 我们 自定义对象的比较规则
Comparable接口
,根据 CompareTo 方法
来指定比较规则:比较此对象与指定对象的顺序。如果该对象小于、等于或大于指定对象,则分别返回负整数、零或正整数。负整数 -> 小于,0 -> 等于,正整数 -> 大于//compareTo方法
@Override
public int compareTo(Student o) {
int result;
if(age == o.age) {
//当两个同学的年龄形同的时候,进一步按照名字排序
result = name.compareTo(o.name);
} else if(age > o.age) {
result = 100;
} else {
result = -34;
}
return result;
}
Comparator
//通过比较器,来制定比较的规则
TreeSet<Student> students = new TreeSet<>(new Comparator<Student>() {
@Override
public int compare(Student o1, Student o2) {
return o1.getAge() - o2.getAge();
}
});