# 原创,转载请先留言

1.集合的由来

数组的长度是固定的,当需要增加或减少元素时需要对数组重新定义,太麻烦了。java内部给我们提供了集合类,能存储任意对象,长度可以改变的,随着元素的增加而增加,随着元素的减少而减少。

2.数组和集合的区别

(1)数组既可以存储基本数据类型,又可以存储引用数据类型(存储基本数据类型时存储的是值,存储引用数据类型的是地址值)。集合只能存储引用数据类型,但是实际用的时候可以存储基本数据类型,为什么呢?(是因为在存储的时候进行了自动装箱,例如 int 自动装箱为integer)

(2)数组长度是固定的,不能自动增长。集合的长度是可变的,根据元素的增加而增长。

3.数组和集合什么时候用

(1)如果元素个数是固定的推荐用数组(为什么不一直用集合呢?举个例子,你要存储固定的50个元素,用数组直接划分50个空间就可以存储了。但是用集合的话,例如,某些集合的底层原理是使用数组做的[ArrayList],他会先创建10个空间,不够再增长到1.5倍,到15,再不够再增长,效率低。)

(2)如果元素个数不是固定的推荐用集合

4.集合继承体系

看API文档可以知道Collection是接口。

而且还有两大子类接口List和Set。

List和Set下会有各种实现类。如下图:

5.各个接口或类常用方法

  • Collection接口

Collection是一个接口,所以不能实例化,即不能创建对象。

Collection有几个常见的接口方法,可以体验一下:

boolean add(E e)  添加元素

boolean remove(Object o)   删除元素

void clear()  清除集合

boolean contains(Object o)  是否包含

boolean isEmpty()  是否为空集合

int size()  集合元素的个数

object[] toArray  把集合转换为对象数组

boolean addAll(Collection<? extends E> c)    把另一个集合的元素添加到该集合

boolean removeAll(Collection<?> c)    删除指定集合中包含的所有此集合的元素

boolean containsAll(Collection<?> c)    如果此集合包含指定 集合中的所有元素,则返回true

boolean retainAll(Collection<?> c)   仅保留此集合中包含在指定集合中的元素(注意这个返回值比较反人类,后面示例2验证)

Iterator<E> iterator()    返回集合的迭代器

示例1:(toArray方法)

public static void main(String[] args) {
Collection c = new ArrayList();
c.add(new Students("张三",13)); // 这里其实实现了自动装箱,Object o = new Students("张三",13),因为add的参数是Object对象
c.add(new Students("李四",14));
c.add(new Students("王五",15)); //遍历集合,先把集合转化为数组,然后遍历
Object[] arr= c.toArray(); // 注意返回值是Object对象,不是Students对象
for (int i = 0;i < arr.length;i++){
// 如果要用到子类的特有的属性或者方法,比较向下转型为子类对象
Students s = (Students) arr[i];
System.out.println(s.getName() + "---" + s.getAge()); // 当然,你也可以重写toString方法直接输出,就不用向下转型了
}
}

示例2:(retainAll方法)

 public static void main(String[] args) {
Collection c = new ArrayList();
c.add(100);
c.add(200);
c.add(300);
Collection c2 = new ArrayList();
c2.add(100);
c2.add(200);
c2.add(300);
// c2.add(400);
Boolean flag = c.retainAll(c2); // 返回值有点古怪!c如果变化了,就是true。如果没变化就是false
System.out.println(c);
System.out.println(flag); }
  • List接口

List是一个接口,不能实例化。

List有几个Collection没有的接口方法。

void add(int index,E element)    在指定索引添加元素

E remove(int index)    删除指定索引的元素

E get(int index)    获取指定索引的元素

E set(int index,E element)    修改指定索引的值

易错点:(remove方法)

public static void main(String[] args) {
List l = new ArrayList();
l.add(100);
l.add(200);
l.add(300);
/*IndexOutOfBoundsException.注意,虽然Colletion有这个remove(Object o)接口方法,
但是remove([如果是int类型])不会自动装包,所以输入数字的时候默认是index。但是如果是其他类型是可以的。*/
l.remove(100);
System.out.println(l); }
  • ArrayList、LinkedList、Vetor类

ArrayList类底层是数组实现的。线程不安全,效率高。查询修改快,增删慢。

Vetor类底层是数组实现的。线程安全,效率低。查询修改快,增删慢。(基本已经被ArrayList取代了)

LinkList类底层是链表实现的。线程不安全,效率高。增删快,查询修改慢。

* 如果增删多用LinkedList,如果查询修改多用ArrayList

  • Set接口

打开API文档看了下,set接口的接口方法基本上和Collection的接口一样,所以可以set接口方法的使用可以参考上面的Collection接口方法。

  • HashSet类

HashSet类是Set接口的实现类。HashSet只能存储不同的元素、

boolean add(E e)    将指定的元素添加到此集合(如果尚未存在)

但是add对象的时候出现了问题:

public static void main(String[] args) {
HashSet<Students> hs = new HashSet<>();
hs.add(new Students("张三",13));
hs.add(new Students("张三",13));
hs.add(new Students("李四",14));
hs.add(new Students("李四",14));
hs.add(new Students("李四",14));
for (Students s:hs) {
System.out.println(s);
}
} // 输出结果:
Students{name='张三', age=13}
Students{name='李四', age=14}
Students{name='李四', age=14}
Students{name='张三', age=13}
Students{name='李四', age=14}

如果我们把同名与同年龄的看做同一个对象,希望输出结果只有张三,李四呢?

根据ArrayList遇到这种情况的时候,第一时间想到了equals方法,重写equals方法试试:

@Override
public boolean equals(Object obj) {
Students s = (Students)obj;
return this.name.equals(s.name) && this.age == s.age;
}

运行,发现并不可行,输出结果还是那个。问题出在哪里?

其实HashSet的判断方式是这样的,先判断hashCode,如果hashCode一样,才会判断equals方法。如果hashCode不一样,那么直接认为他们是不一样的。

我们先让Students对象的hashCode都是一个数,尝试一下:

@Override
public int hashCode() {
return 10;
}

发现结果符合我们的预期了,返回的是张三、李四两个对象。是因为大家的hashCode都一样,然后就运行equals方法进行比较。

但是这样的话,hashCode的方法就完全没有意义了。而且每次都会运行equals方法,效率很低。设想这么一个场景,每个对象都是不一样的,然而返回的哈希值一样,然后每一个都运行equals方法进行比较,效率十分低。

最理想的状态就是运行hashCode方法时相同的对象就是相同的值,不同的对象就是不同的值。这样才是最高效率的。而equals方法只是一个后备,万一不同的对象算出来的哈希值一样,equals方法才用的上。所以必须hashCode方法与equals方法都必须重写!

比较重要的是要想办法重写一个漂亮的hashCode方法。但是IDE已经帮我们完成了这个- -

IDEA下Alt+Insert选择equals and hashCode就行了。

  • LinkedHashSet类(上面的图没有画出来)

LinkedHashSet是HashSet的子类,用法基本和HashSet一样。LinkedHashSet底层是用链表与哈希表实现。LinkedHashSet最大的特点就是它是Set族中唯一一个能怎么存怎么取的。

  • TreeSet集合

TreeSet集合可以对元素进行排序,当然也可以保证元素的唯一。

关于TreeSet要注意的内容比较多,可以见我的另一篇博文:

java的collection集合的更多相关文章

  1. java 15-1 Collection集合的概述以及小功能介绍

     集合的由来:  我们学习的是面向对象语言,而面向对象语言对事物的描述是通过对象体现的,为了方便对多个对象进行操作,我们就必须把这多个对象进行存储.  而要想存储多个对象,就不能是一个基本的变量,而应 ...

  2. Java 基础 - Collection集合通用方法及操作/ArrayList和LinkedList的差别优势 /弃用的Vector

    Collection的笔记: /**存储对象考虑使用: * 1.数组, ①一旦创建,其长度不可变!② 长度难于应对实际情况 * 2.Java集合, ①Collection集合: 1.set: 元素无序 ...

  3. Java的Collection集合的常用方法

    boolean add(E e) 添加元素到Collection集合中. boolean addAll(Collection<? extends E> c) 将指定c中的所有元素都添加到此 ...

  4. Java基础Collection集合

    1.Collection是所有集合的父类,在JDK1.5之后又加入了Iterable超级类(可以不用了解) 2.学习集合从Collection开始,所有集合都继承了他的方法 集合结构如图:

  5. Java基础知识强化之集合框架笔记05:Collection集合的遍历

    1.Collection集合的遍历 Collection集合直接是不能遍历的,所以我们要间接方式才能遍历,我们知道数组Array方便实现变量,我们可以这样: 使用Object[]  toArray() ...

  6. Java基础知识强化之集合框架笔记04:Collection集合的基本功能测试

    1. Collection集合的基本功能测试: package cn.itcast_01; import java.util.ArrayList; import java.util.Collectio ...

  7. Java Collection集合方法

    一.简单方法 package cn.itcast.day15; import java.util.ArrayList; import java.util.Arrays; import java.uti ...

  8. 万字长文深入理解java中的集合-附PDF下载

    目录 1. 前言 2. List 2.1 fail-safe fail-fast知多少 2.1.1 Fail-fast Iterator 2.1.2 Fail-fast 的原理 2.1.3 Fail- ...

  9. JAVA collection集合之 扑克牌游戏

    主要内容:这里使用collection集合,模拟香港电影中大佬们玩的扑克牌游戏. 1.游戏规则:两个玩家每人手中发两张牌,进行比较.比较每个玩家手中牌最大的点数,大小由A-2,点数大者获胜.如果点数相 ...

随机推荐

  1. BZOJ2453:维护队列——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=2453 Description 你小时候玩过弹珠吗? 小朋友A有一些弹珠,A喜欢把它们排成队列,从左到 ...

  2. UVA.129 Krypton Factor (搜索+暴力)

    UVA.129 Krypton Factor (搜索+暴力) 题意分析 搜索的策略是:优先找长串,若长串不合法,则回溯,继续找到合法串,直到找到所求合法串的编号,输出即可. 注意的地方就是合法串的判断 ...

  3. redux的一些插件总结(redux-actions,reselect)

    redux本身还是过于简单,实际使用的时候需要配合许多插件. 下面是一些插件与vuex的功能对比 redux-actions <=> vuex的mutation的写法 reselect & ...

  4. JavaScript数据类型转换方法汇总

    转换为布尔型 用两次非运算(!): 1 !!5 ==> true 用布尔型的构造函数: 1 new Boolean(5) == > true 值转换为布尔类型为false:0,+0,-0, ...

  5. Linux环境下用Weblogic发布项目【二】 -- 配置Domain域

    配置注意事项: 修改密码时密码长度最少8位:在"<下一步>"后面为空即表示敲回车: 具体配置步骤如下: [root@GPS-App ~]# [root@GPS-App ...

  6. SQL Server 2008设置主键为自增

    环境:SQL Server 2008 问题:设置主键,将主键设为自增. 解决:点击table->选中表->design->选中需要设置主键的字段,单击右键"设置主键&quo ...

  7. ubuntu14.04安装GTX 1080 ti遇到黑屏问题

    实验室给我配置了一个1080ti的卡,那个激动,windows下1000+的FPS,跑分40W,无敌,言归正传,ubuntu14.04下配nvidia 1080的驱动还是出现了很多问题,差点就要重装系 ...

  8. Moodle简介

    Moodle简介 一.概述 Moodle是Modular Object-Oriented Dynamic Learning Environment(模块化面向对象动态学习环境)的简称,中文译名为魔灯, ...

  9. scrapy架构设计分析

    scrapy是一个Python爬虫框架.我们自己用requests也能写爬虫(GET某个URL,然后Parse网页的内容),那么,问题来了,scrapy高明在哪些地方呢?下面就来讨论下这个话题,看看业 ...

  10. 基于html5的动画库,非svg和canvas

    基于html5的动画库,非svg和canvas https://greensock.com/docs/#/HTML5/GSAP/TweenLite/