JAVA基础--JAVA API集合框架(其他集合类,集合原理)15
一、ArrayList介绍
1.ArrayList介绍
ArrayList它是List接口的真正的实现类。也是我们开发中真正需要使用集合容器对象。
ArrayList类,它是List接口的实现。肯定拥有角标。并且可以存放重复元素,也能够使用List接口中的所有特有方法。ArrayList集合容器,它的底层使用的可变的数组作为存储元素的容器。
上述分析ArrayList底层的可变数组。这个可变数组也称为ArrayList集合的底层数据存储的结构(数据结构)。
ArrayList它的底层使用的可变数组:
它中增删效率相对较低。查询的效率较快。ArrayList它是JDK1.2出现的集合容器。它不保证数据的安全。
2. ArrayList的构造方法和方法介绍
ArrayList中的方法全部来自于Collection和List接口。
二、LinkedList 介绍
List 接口的链接列表实现。实现所有可选的列表操作,并且允许所有元素(包括 null)。
除了实现 List 接口外,LinkedList 类还为在列表的开头及结尾 get、remove 和 insert 元素提供了统一的命名方法。这些操作允许将链接列表用作堆栈、队列或双端队列。
LinkedList集合它是List接口的直接实现类。它实现了Collection和List接口中的所有方法。
LiknedList集合它的底层使用的链接列表:
LinkedList:集合底层使用的链表结构,由于链表有头和尾,因此在LinkedList集合中重新定义了新的方法,这些方法都是围绕链表的头和尾而设计的方法:
addFirst
addLast
removeFirst
removeLast
getFirst
getLast
链表结构可以完成数据结构中的部分结构的实现:
队列:先进先出,后进后出。最先 进入到结构中的数据,被最先取出。例如:排队买票,火车过山洞。
堆栈:先进后出,后进先出。例如:Java内存结构中的栈内存。手枪弹夹。
//模拟队列或者堆栈数据结构
class Queue{ //定义一个LinkedList集合
private LinkedList list = new LinkedList(); public void myAdd( Object obj ){
list.addFirst(obj);
} public Object myGet(){
return list.removeLast();
} public boolean isEmpty(){
return list.isEmpty();
} }
//测试类
public class LinkedListDemo2 {
public static void main(String[] args) { Queue q = new Queue(); q.myAdd("aaa");
q.myAdd("bbb");
q.myAdd("ccc");
q.myAdd("ddd"); while( !q.isEmpty() ){
System.out.println(q.myGet());
}
}
}
三、 List接口下的集合总结:
List:它下面的集合容器都有脚标,都可以使用角标或者ListIterator进行遍历。
ArrayList:它的底层是可变数组,增删慢,查询快。不安全。
LinkedList:它的底层是链表结构,有头和尾,可以模拟队列或者堆栈数据结构。增删快,查询慢。不安全
四、Vetor集合
1.Vector介绍
Vector集合是它JDK1.0时期就存在的一个集合容器对象。在JDK1.2的时候出现了集合体系,Vector变成List接口下的一个实现类。
Vector的底层使用的也是可变数组,和ArrayList底层一致。Vector它可以保证安全,因为它的什么效率都慢。
后期开发中看到Vector就可以直接把它看成ArrayList使用即可。
Vector中的方法名只要是和element相关的都是Vector的原生方法。其他的都是从Collection或者List接口中实现的。
/*
* 介绍古老的Vector集合
*/
public class VectorDemo {
public static void main(String[] args) { //创建集合对象
Vector v = new Vector(); v.addElement("aaa");
v.addElement("aaa");
v.addElement("abc");
v.addElement("ccc"); //遍历
for( Enumeration en = v.elements(); en.hasMoreElements() ; ){
System.out.println(en.nextElement());
} for( Iterator it = v.iterator(); it.hasNext() ; ){
System.out.println(it.next());
}
}
}
Enumeration:它是Iterator的前身,也可以遍历集合容器。
但是由于它的功能和Iterator严重重复,并且Iterator中提供的方法名称比Enumeration中的方法名短。
开发中我们优先使用Iterator,而不使用Enumeration迭代集合。
五、SET集合
1. Set接口介绍
Set接口是Collection接口下的一个子接口。它和List接口相同,都是Collection的子接口。
区别:
List:有脚本、可以重复、拥有特有的ListIterator迭代器
Set:没有脚本、不能包含重复元素,没有特有的迭代器,只能使用Iterator进行遍历。
2.HashSet集合
Set接口没有自己的特有的方法,所有方法全部来自于Collection接口。
重点学习Set接口下的两个实现类:
HashSet:
TreeSet:
此类实现 Set 接口,由哈希表(实际上是一个 HashMap 实例)支持。它不保证 set 的迭代顺序;特别是它不保证该顺序恒久不变。此类允许使用 null 元素。
HashSet集合的底层使用的哈希表结构。
2.1 哈希表介绍
如果多个对象它们计算出的哈希值相同,这时在计算机中称为哈希冲突。如果哈希值相同了,这时必须调用对象的equals方法比较2个对象(多个对象)是否是同一个对象。
任何给哈希表中存储的对象都应该具备hashCode和equals方法。
/*
* 演示HashSet集合
*/
public class HashSetDemo {
public static void main(String[] args) { //创建集合对象
HashSet set = new HashSet(); //添加元素
set.add("bbb");
set.add("cccc");
set.add("cccc");
set.add("abc");
set.add("abc");
//遍历HashSet集合
for( Iterator it = set.iterator() ; it.hasNext(); ){
System.out.println(it.next());
}
}
}
2.2 HashSet 保存自定义对象
/*
* 演示给HashSet中保存自定义类的对象
*/
public class HashSetDemo2 {
public static void main(String[] args) { //创建集合对象
HashSet set = new HashSet(); //创建自定义对象
//Person p = new Person("华安",23);
//set.add(p);
set.add(new Person("华安",23));
set.add(new Person("华安",23));
set.add(new Person("秋香",18));
set.add(new Person("9527",23));
set.add(new Person("小书童",33)); //遍历
for( Iterator it = set.iterator() ; it.hasNext(); ){
System.out.println( it.next() );
}
}
}
分析:
我们自己定义了一个Person类,然后创建Person对象,将这些Person对象存储到HashSet集合中。
希望同姓名和年龄的Person对象,应该是同一个人,在集合中只能有一个。但是通过程序发现依然可以保存多个同姓名和年龄的Person对象。
HashSet集合底层使用的哈希表,给哈希表中存储对象的时候,需要调用当前对象自己的hashCode计算位置,equals判断对象是否相同。而我们的Person类继承了Object类,自然就会具备hashCode和equals方法。
但是Person类中并没有去复写hashCode和equals方法,说明在将Person对象给HashSet集合中保存的时候,调用的hashCode和equals方法依然使用的Object类中的方法。
Object类中的equals方法是在比较2个对象的地址值是否相同。
hashCode方法计算哈希值的是也是根据对象的地址值在进行计算。
如果创建了多个Person对象,它们的姓名和年龄相同,我们认为应该是同一个人,但是由于是new了多次,在堆中就会有多个Person对象,它们的地址值肯定是不相同的。
2.3 HashSet保证对象唯一的原因
HashSet保证对象的唯一依赖当前给HashSet集合中保存的对象的hashCode和equals方法。
使用hashCode计算位置,使用equals比较2个对象是否相同。
2.4 HashSet总结
要保证给HashSet集合中存储的自定义对象中的数据唯一,自定义对象所属的类需要复写Object类中的hashCode和equals方法 。
2.5 LinkedHashSet介绍
HashSet集合不保证存取的顺序,保证对象唯一。
LinkedHashSet它是HashSet集合的子类,它也属于Set接口下的集合容器,同样也能保证对象的唯一。
和HashSet的区别是:LinkedHashSet可以保证存放的元素的存取顺序一致。
LinkedHashSet它没有自己的特有方法,所有方法全部来自于父类。
/*
* 演示LinkedHashSet集合
*/
public class LinkedHashSetDemo {
public static void main(String[] args) { //创建集合对象
LinkedHashSet set = new LinkedHashSet(); set.add("ccc");
set.add("aaa");
set.add("abc");
set.add("ccc");
set.add("aaa"); //遍历
for (Iterator it = set.iterator(); it.hasNext();) {
System.out.println(it.next());
}
}
}
六、TreeSet介绍
1. TreeSet介绍
基于 TreeMap
的 NavigableSet
实现。使用元素的自然顺序对元素进行排序,或者根据创建 set 时提供的 Comparator
进行排序,具体取决于使用的构造方法。
TreeSet集合它是Set接口的间接实现类。TreeSet集合它也不能存放重复的元素。
TreeSet集合的底层使用的数据结构中的二叉树(红黑树)结构。这种数据结构可以对保存在其中的数据进行排序。
2. TreeSet演示
/*
* TreeSet集合的简单演示
*/
public class TreeSetDemo {
public static void main(String[] args) { //创建集合对象
TreeSet set = new TreeSet(); //添加元素
set.add("ddd");
set.add("aaa");
set.add("bbb");
set.add("bbb");
set.add("abc");
set.add("ABC");
set.add("ABC");
set.add("aBC");
set.add("eee");
set.add("def");
set.add("def");
set.add("ddf");
set.add("ddf"); //遍历
for( Iterator it = set.iterator() ; it.hasNext(); ){
System.out.println(it.next());
}
}
}
3. 树结构介绍
介绍数据结构中的树结构:
树结构:
二叉树:每个节点最多只能有2个子节点。
非规则二叉树:有点节点可能会有2个子节点,有的只有一个。甚至没有。
普通的树结构:非规则树结构。
二叉树:
4.TreeSet保存自定义对象
如果将自定义对象存储到TreeSet集合(二叉树结构)中,这时底层会将传递的这个对象强制转成Comparable类型。
发生上述的原因是:
给TreeSet中保存的对象需要进行比较大小,然后确定它们在二叉树中存储位置,存储的对象就必须具备比较大小的功能。如果要使一个对象具备比较大小的功能,这个对象所属的类需要实现Java中提供的Comparable接口。实现其中的compareTo方法。
/*
* 演示给TreeSet集合中保存自定义对象
*/
public class TreeSetDemo2 { public static void main(String[] args) { //创建TreeSet集合对象
TreeSet set = new TreeSet(); set.add(new Person("华安",23));
set.add(new Person("华安",23));
set.add(new Person("秋香",13));
set.add(new Person("秋香",23));
set.add(new Person("石榴",33)); //遍历
for( Iterator it = set.iterator() ; it.hasNext() ; ){
System.out.println(it.next());
}
}
}
将对象保存到TreeSet集合中过程中:
在将对象保存到集合中的时候,需要将当前这个对象和集合中已经存在的对象进行大小比较,也就是去调用对象的compareTo方法。如果这个方法返回的结果是零,就说明当前正要往集合中存储的对象和已经在集合中的对象大小一样。这时这个对象就不会被保存到集合中。
5. Comparable介绍使用
/*
* 让我们自己的类实现Comparable接口,让其对象具备比较大小的功能
*/ public class Person implements Comparable{
private String name;
private int age; public Person(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;
}
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + "]";
}
/*
* 实现Comparable接口中的compareTo方法
* 需要在compareTo方法中根据对象的属性数据进行大小的比较
*
* compareTo方法的调用时间:
* 当正要给TreeSet中保存对象的时候,由TreeSet集合的底层,会将
* 当前这个对象强转成Comparable类型,然后再用强转后的这个对象
* 调用compareTo方法,将已经在集合中的对象传递Object o ,然后
* 在compareTo方法中比较当前正要存放的对象和已经在集合中的对象的大小。
*/
public int compareTo(Object o) { //判断传递进来的o是否是Person类型
if( !(o instanceof Person) ){
//判断成立,说明当前传递进来的o不是Person类型
throw new ClassCastException("传递的对象不是Person类型");
}
Person p = (Person) o;
/*
//比较
if( this.age == p.age ){
//年龄相同,判断姓名
if( this.name.equals(p.name) ){
//判断成立说明 年龄和姓名都相同认为是同一个人
return 0;
}else{
return this.name.compareTo(p.name);
}
}else{
return this.age - p.age;
}
*/
int temp = this.age - p.age;
return temp == 0 ? this.name.compareTo(p.name) : temp;
}
}
七、比较器介绍
1. 比较器介绍
需求:将字符串数据保到TreeSet集合中,要求按照字符串的长度进行排序。
“AAAAA” “aaa”
分析:
本身TreeSet集合可以对存放在其中的任何对象进行排序,依赖的是对象自己的compareTo方法。String类本身已经具备compareTo方法,但是这个方法是按照字符串中每个字符的编码值大小进行比较。而需求要求按照字符串的长度进行比较。String类中的compareTo方法不能满足需求。
针对上述的问题,本身应该可以定义一个新的类,继承String类,复写compareTo方法,然后使用子类自己的比较方式。但是由于String类被final修饰,不能再有子类。
Java给出另外一种处理方案:如果对象自身具备了比较功能,或者对象自身根本就没有比较功能,这时依然需要对这些对象进行排序,我们可以自己创建一个专门负责比较对象大小的新的比较器对象。
2. 比较器演示
/*
* 演示使用比较器对TreeSet集合中的元素(对象)进行比较大小排序
*/
public class TreeSetDemo3 {
public static void main(String[] args) {
//没有传递比较器,会按照元素自身的compareTo进行大小比较
//TreeSet set = new TreeSet( );
//传递比较器
TreeSet set = new TreeSet( new MyComparator()); set.add("AAAAA");
set.add("aaa");
set.add("aa");
set.add("AAA");
set.add("AAAAA"); for (Iterator it = set.iterator(); it.hasNext();) {
System.out.println(it.next());
}
}
} //自定义比较器(自己定义比较规则)
public class MyComparator implements Comparator{ //实现按照字符串长度比较大小
public int compare(Object o1, Object o2) { //按照长度比较
if( !( o1 instanceof String && o2 instanceof String ) ){
throw new ClassCastException("数据类型不统一,无法比较");
} String s1 = (String) o1;
String s2 = (String) o2; int temp = s1.length() - s2.length(); return temp == 0 ? s1.compareTo(s2) : temp ;
}
}
3. Comparator和Comparable总结
相同点:Comparator和Comparable:它们都可以提供比较大小的方法。
不同点:
Comparable:它是让某个类具备比较功能,当创建出这个类的对象,就可以直接使用这个对象中的compareTo方法进行比较大小。Comparable是让对象自己具备了比较功能
Comparator:它是提供比较两个对象大小,但是这个功能和对象自身并没有任何的关系。只要将需要比较的两个对象传递给compare方法即可。提供单独的比较器对象,可以比较任意的两个对象
Collection下的集合总结
JAVA基础--JAVA API集合框架(其他集合类,集合原理)15的更多相关文章
- Java基础学习(四)-- 接口、集合框架、Collection、泛型详解
接口 一.接口的基本概念 关键字为:Interface,在JAVA编程语言中是一个抽象类型,是抽象方法的集合.也是使用.java文件编写. 二.接口声明 命名规范:与类名的命名规范相同,通常情况下 ...
- 牛客网Java刷题知识点之Java 集合框架的构成、集合框架中的迭代器Iterator、集合框架中的集合接口Collection(List和Set)、集合框架中的Map集合
不多说,直接上干货! 集合框架中包含了大量集合接口.这些接口的实现类和操作它们的算法. 集合容器因为内部的数据结构不同,有多种具体容器. 不断的向上抽取,就形成了集合框架. Map是一次添加一对元素. ...
- Java集合框架概述和集合的遍历
第三阶段 JAVA常见对象的学习 集合框架概述和集合的遍历 (一) 集合框架的概述 (1) 集合的由来 如果一个程序只包含固定数量的且其生命周期都是已知的对象,那么这是一个非常简单的程序. 通常,程序 ...
- Java基础-Java中23种设计模式之常用的设计模式
Java基础-Java中23种设计模式之常用的设计模式 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.设计模式分类 设计模式是针对特定场景给出的专家级的解决方案.总的来说设 ...
- Java基础-JAVA中常见的数据结构介绍
Java基础-JAVA中常见的数据结构介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.什么是数据结构 答:数据结构是指数据存储的组织方式.大致上分为线性表.栈(Stack) ...
- java基础---->java中正则表达式二
跟正则表达式相关的类有:Pattern.Matcher和String.今天我们就开始Java中正则表达式的学习. Pattern和Matcher的理解 一.正则表达式的使用方法 一般推荐使用的方式如下 ...
- Java基础-Java中的堆内存和离堆内存机制
Java基础-Java中的堆内存和离堆内存机制 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任.
- Java基础-Java中的内存分配与回收机制
Java基础-Java中的内存分配与回收机制 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一. 二.
- Java基础-Java中的并法库之重入读写锁(ReentrantReadWriteLock)
Java基础-Java中的并法库之重入读写锁(ReentrantReadWriteLock) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 在学习Java的之前,你可能已经听说过读 ...
- Java基础-Java中的并法库之线程池技术
Java基础-Java中的并法库之线程池技术 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.什么是线程池技术 二.
随机推荐
- IOS UIWebView 随记
UIWebView中加载的网页尺寸太大,如何让网页适应屏幕大小 webview.scalesPageToFit = YES;
- gdb调试段错误及使用
在编程调试中,经常出现段错误,此时可用gdb调试.具体方法为注册段错误信号处理函数,在处理函数中启动gdb.具体代码如下: void segv_handler(int no) { ]; ]; FILE ...
- h5的复制功能
js+html5实现复制文字按钮 <div> <input type="text" name="guanfangaddress" id=&qu ...
- 不为客户连接创建子进程的并发回射服务器( select实现 )
前言 在此前,我已经介绍了一种并发回射服务器实现( 点此进入 ).它通过调用fork函数为每个客户请求创建一个子进程.同时,我还为此服务器添加了自动消除僵尸子进程的机制.现在请想想,在客户量非常大的情 ...
- c#生成试卷。。。
.net下,操作Word的插件有NPOI,Spire,一版大家经常用的是NPOI,我在着手开发的时候,优先考虑的也是NPOI,然而时间比较着急,没有找到NPOI支持2003版本, 就放弃了,从网上发行 ...
- 【BZOJ3052】[wc2013]糖果公园 带修改的树上莫队
[BZOJ3052][wc2013]糖果公园 Description Input Output Sample Input Sample Input Sample Output 84 131 27 84 ...
- 网络直播流媒体协议的选择讨论,RTSP,RTMP,HTTP,私有协议?
最近有不少人在EasyDarwin的交流群里面问关于花椒.映客手机直播技术的问题,还有RTSP.RTMP协议选择的问题,这里个人谈一下自己的愚见. 1.不管是RTSP/RTP.RTMP.HTTP,亦或 ...
- ECMAscript 没有对该方法进行标准化,因此反对使用它。 es 日期格式化
JavaScript substr() 方法 http://www.w3school.com.cn/jsref/jsref_substr.asp 注释:substr() 的参数指定的是子串的开始位置和 ...
- checkbox复选框的使用
复选框: <input type="checkbox" name="favor" value="唱歌"/>唱歌 <i ...
- handler message messagequeue详解
long when 应该被处理的时间值,如果是即时发送,即为当前时间,如果是延迟发送则为当前时间+延迟时间