java数组容器的分析数组、List和Set、Map
java中常用的数据结构
java中常用的数据结构
Java中常用的数据结构有:
1. 数组(Array):一组具有相同类型的数据元素的集合,通
过索引来访问元素。
2. 链表(LinkedList):由若干个节点组成,每个节点包含数
据和指向下一个节点的指针。
3. 栈(Stack):一种后进先出(LIFO)的数据结构,只允许
在栈顶进行插入和删除操作。
4. 队列(Queue):一种先进先出(FIFO)的数据结构,只允
许在队头和队尾进行插入和删除操作。
5. 集合(Set):一种不允许重复元素的数据结构,常见的实
现类有HashSet和TreeSet。
6. 列表(List):一种有序的数据结构,允许重复元素,常见
的实现类有ArrayList和LinkedList。
7. 字典(Map):一种键值对的数据结构,以键作为唯一标识
符来存储和访问元素,常见的实现类有HashMap和TreeMap。
8. 堆(Heap):一种可以快速找到最大值(或最小值)的数
据结构,常用于优先队列的实现。
9. 树(Tree):一种层次关系的数据结构,包含根节点、子节
点和叶子节点等。
10. 图(Graph):由节点和节点之间的关系(边)组成的数据结构,常用于描述网络等复杂关系。
这些数据结构在Java中都有对应的类或接口,可以根据具体
的需求选择合适的数据结构来使用。
Java中常用的容器类有哪些?它们有什么特点?
Java中常用的容器类有哪些?它们有什么特点?2023年的今天,Java作为一门跨平台的面向对象编程语言,在软件开发领域继续得到广泛的应用。
而在Java中,容器类是不可或缺的一部分,它们为程序员提供了方便易用的数据结构和算法,进而帮助他们快速实现各种功能。
本文将详细介绍Java中常用的容器类,并探讨它们的特点和使用方式。
1. ListList是Java中最基本的容器类之一,它也是最常用的容器之一。
List容器可以按顺序存储一组对象,并允许有重复元素。
List容器提供了一系列的操作方法,包括增加、删除、获取、修改等操作。
在Java中,常见的List容器包括ArrayList、LinkedList和Vector 等。
- ArrayListArrayList是用数组实现的List容器。
它的优点是支持随机访问,插入和删除元素时效率高,而缺点则是在删除元素时需要移动数组的其他元素。
因此,ArrayList适合用于索引域比较小的场合。
- LinkedList相比ArrayList,LinkedList则是通过双向链表的方式实现的容器。
LinkedList支持插入和删除元素时具有较高的效率,而随机访问效率较低。
因此,当需要频繁地进行插入和删除操作时,LinkedList 可以提供更好的性能。
- VectorVector是一个同步的List容器,在多线程环境中更加稳定。
Vector与ArrayList类似,也是用数组实现的容器,并且支持随机访问。
2. SetSet是另一种常用的容器类,它用于存储一组无序的元素。
Set容器具有不允许有重复元素的特点,并提供一系列的操作方法,包括增加、删除、包含等操作。
在Java中,常见的Set容器包括HashSet、LinkedHashSet和TreeSet等。
- HashSetHashSet是使用哈希表实现的Set容器,它具有快速查找元素的特点。
HashSet容器不允许重复元素,因此它适合用于元素去重的场合。
浅谈JAVA集合框架及其应用
2020年第10期104计算机应用信息技术与信息化浅谈JAVA 集合框架及其应用段莎莉* DUAN Sha-li摘 要 在计算机应用技术领域的学习中,灵活运用各种各样的数据结构可以说是一项基本技能,了解各种数据结构的底层源码将有助于用户更好地使用各种开源框架,java 语言凭借其跨平台的先天优势及桌面端、WEB 端、移动端等多场景的应用成为许多程序员追捧的对象,本文将结合jdk6.0源码分析java 集合框架中常用的接口和实现类对应的底层数据结构及其特点,便于在实际业务开发中做出更好地选择。
关键词 java 集合框架;数据结构;应用doi:10.3969/j.issn.1672-9528.2020.10.032* 山西国际商务职业学院 山西太原 0300311 java 集合框架简介java 集合框架位于java.util 包中,作为对数组结构的补充,当元素个数即将超过容器定义的长度时,它会乘以系数进行自动扩容。
根据存储元素种类的不同可分为两大类,有存储元素为单个引用型数据的Collection 集合和存储元素为<键,值>组合的Map 集合,二者本身没有任何联系,但是Map集合中元素的key 是无序不可重复的,其底层是Col-lection 集合中的Set 集合,这样就可以通过Collection 集合的迭代器遍历Map 集合的key,从而遍历到整个集合的元素记录。
它们之间的实现、继承关系见图1、图2所示。
图1 Collection 接口中常用集合类继承关系图图2 Map 接口中常用集合类的继承关系图计算机应用信息技术与信息化2 集合底层的数据结构及其特点2.1 Collection集合Collection集合中常用的子接口有List和Set。
List中的元素是有序可重复的,常用的实现子类有Ar-rayList、LinkedList和Vector。
ArrayList和Vector底层均采用数组的线性结构,每个元素有唯一的下标索引便于查找,与此同时,元素的位置相对固定造成在增、删元素时需要大量地移动元素才能进行,影响效率,但add()方法是在集合尾部增加元素,这是不受影响的。
Java中常见数据结构:list与map
Java中常见数据结构:list与map1 1:集合2 Collection(单列集合)3 List(有序,可重复)4 ArrayList5底层数据结构是数组,查询快,增删慢6线程不安全,效率⾼7 Vector8底层数据结构是数组,查询快,增删慢9线程安全,效率低10 LinkedList11底层数据结构是链表,查询慢,增删快12线程不安全,效率⾼13 Set(⽆序,唯⼀)14 HashSet15底层数据结构是哈希表。
16哈希表依赖两个⽅法:hashCode()和equals()17执⾏顺序:18⾸先判断hashCode()值是否相同19是:继续执⾏equals(),看其返回值20是true:说明元素重复,不添加21是false:就直接添加到集合22否:就直接添加到集合23最终:24⾃动⽣成hashCode()和equals()即可2526 LinkedHashSet27底层数据结构由链表和哈希表组成。
28由链表保证元素有序。
29由哈希表保证元素唯⼀。
30 TreeSet31底层数据结构是红⿊树。
(是⼀种⾃平衡的⼆叉树)32如何保证元素唯⼀性呢?33根据⽐较的返回值是否是0来决定34如何保证元素的排序呢?35两种⽅式36⾃然排序(元素具备⽐较性)37让元素所属的类实现Comparable接⼝38⽐较器排序(集合具备⽐较性)39让集合接收⼀个Comparator的实现类对象40 Map(双列集合)41 A:Map集合的数据结构仅仅针对键有效,与值⽆关。
42 B:存储的是键值对形式的元素,键唯⼀,值可重复。
4344 HashMap45底层数据结构是哈希表。
线程不安全,效率⾼46哈希表依赖两个⽅法:hashCode()和equals()47执⾏顺序:48⾸先判断hashCode()值是否相同49是:继续执⾏equals(),看其返回值50是true:说明元素重复,不添加51是false:就直接添加到集合52否:就直接添加到集合53最终:54⾃动⽣成hashCode()和equals()即可55 LinkedHashMap56底层数据结构由链表和哈希表组成。
JAVA数组操作的常用工具类
JAVA数组操作的常用工具类在Java中,数组是一种包含固定数量元素的数据结构。
数组操作是在数组中执行各种任务的过程,并且在编程中经常遇到。
为了简化数组操作的过程,Java提供了许多常用的工具类来处理数组。
下面是Java数组操作的常用工具类。
1. Arrays类:Arrays类提供了许多静态方法来处理数组,其中一些常用的方法包括:- sort(:对数组进行排序。
- binarySearch(:在排序后的数组中查找指定元素的索引。
- toString(:将数组转换为字符串。
- equals(:比较两个数组是否相等。
- fill(:将数组的所有元素设置为指定值。
2. System类:System类提供了一些用于操作数组的静态方法,其中一些常用的方法包括:- arraycopy(:将一个数组的部分元素复制到另一个数组。
- identityHashCode(:获取数组的哈希码。
3. Collections类:Collections类是Java集合框架的一部分,但也可以用于处理数组。
其中一些常用的方法包括:- sort(:对数组进行排序。
- binarySearch(:在排序后的数组中查找指定元素的索引。
4. Arrays类与Collection接口转换:Arrays类和Collection接口之间提供了一些静态方法,可以将数组转换为Collection,并将Collection转换为数组。
其中一些常用的方法包括:- asList(:将数组转换为List。
- toArray(:将Collection转换为数组。
- toList(:将Collection转换为List。
5. Arrays类与Stream接口转换:Java 8引入了Stream接口,可以对数组进行处理。
Arrays类提供了一些方法来将数组转换为Stream,并从Stream转换为数组。
其中一些常用的方法包括:- stream(:将数组转换为Stream。
《Java程序设计》课程教学大纲
《Java程序设计》课程教学大纲课程编号:08120031课程名称:Java程序设计/JAVA Programming总学时/学分:48/3(其中理论32学时,实验16学时)适用专业:计算机科学与技术一、课程目标通过本课程学习,学生应达到如下目标:目标1. 识别Java语言特点、基本语法、语言机制。
目标2. 将面向对象方法知识运用在程序设计案例中,能使用JAVA常用类、枚举、lambda 表达式、容器、泛型进行实例编程验证。
目标3. 利用JavaSE中的异常处理、输入输出等技术来表达处理程序应用问题。
目标4. 将图形用户界面和数据库编程技术运用中综合应用程序设计中。
目标5. 搭建Java开发环境,能设计实现各种Java技术的应用程序,且能测试运行。
二、课程目标对毕业要求的支撑三、教学过程安排四、实验或上机内容五、课程目标达成方法六、考核标准本门课程考核包括6个部分,分别为考试、作业、实验、讨论和测验。
具体要求及评分方法如下:1、期末考试试卷知识点要求2、作业3、实验4、讨论设置讨论课一次,要求学生按照讨论题目分组查阅资料,归纳总结,撰写报告。
5、测验随堂测验,老师给出题目,学生回答。
具体有任课老师给出评分标准。
七、教材及主要参考资料[1] 黑马程序员. Java基础入门(第2版)[M]. 清华大学出版社, 2018.[2] 郑人杰、马素霞、殷人昆. 软件工程概论(第2版)[M]. 机械工业出版社,2016.[3] Gay S.Horstmann. Java核心技术(第10版)[M]. 机械工业出版社,2016.[4] Y.Daniel Liang(美). Java语言程序设计(第10版)[M]. 机械工业出版社.2015.[5] 李刚. 疯狂Java讲义(第4版)[M]. 电子工业出版社,2018.[6] 封亚飞. 揭秘Java虚拟机[M]. 电子工业出版社,2017.[7] Bruce Eckel(美). Java编程思想(第4版)[M]. 机械工业出版社,2007.。
2023上半年 大数据分析师(中级)考前冲刺题A1卷
2023上半年 大数据分析师(中级)考前冲刺题A1卷1.【单选题】HBase的三层结构不包括()。
A:Zookeeper文件B:ROOT表C:.META.表D:Region正确答案:D答案解析:HBase的三层结构分别是Zookeeper文件,ROOT表,.META.表2.【单选题】HBase与Hadoop生态系统其他组件的关系描述不正确的是()。
A:利用HDFS进行数据存储B:利用Zookeeper进行协同服务管理C:利用Spark进行并行计算D:利用Sqoop进行数据导入与导出正确答案:C答案解析:HBase的并行计算不使用Spark3.【单选题】Hive中having子句通常会和()子句一起出现。
A:selectB:fromC:whereD:group by正确答案:D答案解析:having是分组后进行筛选,需要和group by一起使用4.【单选题】Hive中把弧度转换为角度的函数是()。
A:degreesB:radiansC:todegreesD:toradians正确答案:A5.【单选题】Hive中的函数last_day的作用是()。
A:得到参数所指定时间的后一天B:得到参数所指定时间所在月份的最后一天C:得到参数所指定时间所在年份的最后一天D:得到参数所指定时间的前一天正确答案:B6.【单选题】Hive中加载数据到表中,指定文件路径的关键字是()。
A:pathB:inpathC:directoryD:local正确答案:B答案解析:inpath指定要导入的文件7.【单选题】MySQL中,()不是查询语句中的关键字。
A:group byB:havingC:limitD:update正确答案:D8.【单选题】MySQL中,select ascii('A');的查询结果是()。
A:'A'C:65D:64正确答案:C答案解析:A的编码是659.【单选题】MySQL中,查询所有学生的sno和name的SQL语句是()。
Java中常用的集合类有哪些?它们的使用场景是什么?
Java中常用的集合类有哪些?它们的使用场景是什么?Java作为目前最为流行的编程语言之一,其优越的面向对象编程思想和强大的类库使其成为了广大编程爱好者和专业开发者的首选语言之一。
在Java开发中,常用的集合类具有广泛的应用场景,可以大大简化我们代码的编写和维护。
在本篇文章中,我们将介绍Java中常用的集合类有哪些,它们的使用场景是什么,以及如何选择合适的集合类来应对各种场景。
一、Java中常用的集合类Java中常用的集合类包括List、Set、Map等,具体如下:1.ListList是Java中最基础和最常用的集合类之一,它是一个有序的集合,可以存储重复的元素。
List提供了一系列的方法用来操作列表中的元素,如添加、删除、获取、修改等。
常见的List有ArrayList 和LinkedList。
2.SetSet是Java中的另一个基础集合类,它是一个无序的集合,不允许存储重复的元素。
Set提供了一系列的方法用来操作集合中的元素,如添加、删除、获取等。
常见的Set有HashSet、TreeSet。
3.MapMap是Java中常用的映射关系集合,它存储键值对,支持通过键来访问值。
Map提供了一系列的方法用来操作映射关系,如添加、删除、获取、修改等。
常见的Map有HashMap、TreeMap、ConcurrentHashMap等。
二、Java中常用集合类的使用场景不同的集合类有不同的使用场景,我们需要根据具体的业务需求来选择合适的集合类。
下面我们来介绍几种常见的使用场景及其对应的集合类。
1.需要随机访问元素的情况:ArrayListArrayList是Java中常用的集合类之一,它支持随机访问,通过索引访问元素的时间复杂度为O(1),是处理元素数量较大的情况下的较好选择。
2.需要频繁插入或删除元素的情况:LinkedListLinkedList是另一个常用的集合类,它支持快速的插入和删除操作,通过节点互相关联实现。
Java核心数据结构(List、Map、Set)原理与使用技巧
Java核⼼数据结构(List、Map、Set)原理与使⽤技巧JDK提供了⼀组主要的数据结构实现,如List、Set等常⽤数据结构。
这些数据都继承⾃java.util.Collection接⼝,并位于java.util包内。
⼀、List接⼝最重要的三种List接⼝实现:ArrayList、Vector、LinkedList。
它们的类图如下:可以看到,3种List均来⾃AbstratList的实现。
⽽AbstratList直接实现了List接⼝,并扩展⾃AbstratCollection。
ArrayList和Vector使⽤了数组实现,可以认为,ArrayList封装了对内部数组的操作。
⽐如向数组中添加、删除、插⼊新的元素或数组的扩展和重定义。
对ArrayList或者Vector的操作,等价于对内部对象数组的操作。
ArrayList和Vector⼏乎使⽤了相同的算法,它们的唯⼀区别可以认为是对多线程的⽀持。
ArrayList没有对⼀个⽅法做线程同步,因此不是线程安全的。
Vector中绝⼤多数⽅法都做了线程同步,是⼀种线程安全的实现。
因此ArrayList和Vector的性能特性相差⽆⼏。
LinkedList使⽤了循环双向链表数据结构。
LinkedList由⼀系列表项连接⽽成。
⼀个表项总是包含3个部分:元素内容、前驱表项和后驱表项。
如图所⽰:LinkedList的表项源码:private static class Node<E> {E item;Node<E> next;Node<E> prev;Node(Node<E> prev, E element, Node<E> next) {this.item = element;this.next = next;this.prev = prev;}}⽆论LinkedList是否为空,链表都有⼀个header表项,它既是链表的开始,也表⽰链表的结尾。
Java集合排序及java集合类详解(Collection、List、Map、Set)
Java集合排序及java集合类详解(Collection, List, Set, Map)摘要内容Java里面最重要,最常用也就是集合一部分了。
能够用好集合和理解好集合对于做Java程序的开发拥有无比的好处。
本文详细解释了关于Java中的集合是如何实现的,以及他们的实现原理。
关键字:Collection , List ,Set , Map , 集合,框架。
目录1 集合框架 (2)1.1 集合框架概述 (2)1.1.1 容器简介 (2)1.1.2 容器的分类 (4)1.2 Collection (6)1.2.1 常用方法 (6)1.2.2 迭代器 (8)1.3 List (10)1.3.1 概述 (10)1.3.2 常用方法 (11)1.3.3 实现原理 (15)1.4 Map (18)1.4.1 概述 (18)1.4.2 常用方法 (18)1.4.3 Comparable 接口 (23)1.4.4 实现原理 (25)1.4.5 覆写hashCode() (29)1.5 Set (33)1.5.1 概述 (33)1.5.2 常用方法 (34)1.5.3 实现原理 (38)1.6 总结:集合框架中常用类比较 (39)2 练习 (40)3 附录:排序 (41)1集合框架1.1集合框架概述1.1.1容器简介到目前为止,我们已经学习了如何创建多个不同的对象,定义了这些对象以后,我们就可以利用它们来做一些有意义的事情。
举例来说,假设要存储许多雇员,不同的雇员的区别仅在于雇员的身份证号。
我们可以通过身份证号来顺序存储每个雇员,但是在内存中实现呢?是不是要准备足够的内存来存储1000个雇员,然后再将这些雇员逐一插入?如果已经插入了500条记录,这时需要插入一个身份证号较低的新雇员,该怎么办呢?是在内存中将500条记录全部下移后,再从开头插入新的记录? 还是创建一个映射来记住每个对象的位置?当决定如何存储对象的集合时,必须考虑如下问题。
list、set、map的区别
set --其中的值不允许重复,无序的数据结构list --其中的值允许重复,因为其为有序的数据结构map--成对的数据结构,健值必须具有唯一性(键不能同,否则值替换)List按对象进入的顺序保存对象,不做排序或编辑操作。
Set对每个对象只接受一次,并使用自己内部的排序方法(通常,你只关心某个元素是否属于Set,而不关心它的顺序--否则应该使用List)。
Map同样对每个元素保存一份,但这是基于"键"的,Map也有内置的排序,因而不关心元素添加的顺序。
如果添加元素的顺序对你很重要,应该使用 LinkedHashSet或者LinkedHashMap.List的功能方法实际上有两种List: 一种是基本的ArrayList,其优点在于随机访问元素,另一种是更强大的LinkedList,它并不是为快速随机访问设计的,而是具有一套更通用的方法。
List : 次序是List最重要的特点:它保证维护元素特定的顺序。
List为Collection添加了许多方法,使得能够向List中间插入与移除元素(这只推荐LinkedList使用。
)一个List可以生成ListIterator,使用它可以从两个方向遍历List,也可以从List中间插入和移除元素。
ArrayList : 由数组实现的List。
允许对元素进行快速随机访问,但是向List中间插入与移除元素的速度很慢。
ListIterator只应该用来由后向前遍历ArrayList,而不是用来插入和移除元素。
因为那比LinkedList开销要大很多。
LinkedList : 对顺序访问进行了优化,向List中间插入与删除的开销并不大。
随机访问则相对较慢。
(使用ArrayList代替。
)还具有下列方法:addFirst(), addLast(), getFirst(), getLast(), removeFirst() 和 removeLast(), 这些方法(没有在任何接口或基类中定义过)使得LinkedList可以当作堆栈、队列和双向队列使用。
java集合总结
java集合总结一、数组、集合数组、集合:都是一种容器,用一个对象管理多个对象;数组:不能自动增长;只能存放同类型的元素集合:能自动扩容;部分集合允许存放不同类型的元素;二、学习这些集合类要把握哪些东西:1〕怎样得到〔选择〕集合对象;2〕怎样添加元素3〕怎样删除元素4〕怎样循环遍历没一个元素三、list、set、mapcollection:父接口;Set:接口一个实现类:HashSetList:接口三个实现类:LinkedList,Vector,ArrayListSortedSet:接口实现类:TreeSet1、List:List:有序列表,允许存放重复的元素;实现类:ArrayList:数组实现,查询快,增删慢,线程担心全,轻量级;下标也是从0开头;LinkedList:链表实现,增删快,查询慢Vector:数组实现,线程平安,重量级2.Set:无序集合,不允许存放重复的元素;实现类HashSet:equals返回true,hashCode返回相同的整数;哈希表;子接口SortedSet:对Set排序实现类:TreeSet:二叉树实现的;看API:E泛型:表示一个对象;Iterator:接口,迭代器;java.util;hasNext();next();remove();Iterable:可迭代的,访问的;ng;实现了可迭代的接口就可以用迭代的方式访问;只需实现iterator();方法即可;Iteratoriterator();三种循环的访问方式:只有实现了Iterable接口的才能用第三种;能用其次种的也肯定能用第三种;ArrayList:自动扩容,是数组照搬过来的;3.MapHashMap:键值对,key不能重复,但是value可以重复;key 的实现就是HashSet;value对应着放;HashSet的后台有一个HashMap;初始化后台容量;只不过生成一个HashSet的话,系统只供应key的访问;假如有两个Key重复,那么会掩盖之前的;Hashtable:线程平安的Properties:java.util.Properties;key和value都是String 类型,用来读配置文件;HashMap与Hashtable区分:HashMap线程担心全的,允许null作为key或value;Hashtable线程平安的,不允许null作为key或value;TreeMap:对key排好序的Map;key就是TreeSet,value对应每个key;key要实现Comparable接口或TreeMap有自己的构造器;HashSet:remove(Objecto)的原则看这个对象O的Hashcode和equals是否相等,并不是看是不是一个对象;定义一个Map;key是课程名称,value是Integer表示选课人数;map.put(cou,map.get(cou)+newInteger(1));四、Hashtable、Properties1,Hashtable:实现了Map接口,此类实现一个哈希表,作用和HashMap相同。
数组与容器差别
数组是 Java 语言内置的类型,除此之外, Java 有多种保存对象引用的方式。
Java 类库提供了一套相当完整的容器类,使用这些类的方法可以保存和操纵对象。
下面分别进行讨论,在研究Java 容器类之前,先了解一下Java 数组的基本功能和特性。
1. 数组的基本特性数组与其它种类的容器 (List/Set /Map) 之间的区别在于效率、确定的类型和保存基本类型数据的能力。
数组是一种高效的存储和随机访问对象引用序列的方式,使用数组可以快速的访问数组中的元素。
但是当创建一个数组对象 ( 注意和对象数组的区别 ) 后,数组的大小也就固定了,当数组空间不足的时候就再创建一个新的数组,把旧的数组中所有的引用复制到新的数组中。
Java 中的数组和容器都需要进行边界检查,如果越界就会得到一个 RuntimeException 异常。
这点和 C++ 中有所不同, C++ 中 vector 的操作符 [] 不会做边界检查,这在速度上会有一定的提高, Java 的数组和容器会因为时刻存在的边界检查带来一些性能上的开销。
Java 中通用的容器类不会以具体的类型来处理对象,容器中的对象都是以 Object 类型处理的,这是 Java 中所有类的基类。
另外,数组可以保存基本类型,而容器不能,它只能保存任意的 Java 对象。
一般情况下,考虑到效率与类型检查,应该尽可能考虑使用数组。
如果要解决一般化的问题,数组可能会受到一些限制,这时可以使用 Java 提供的容器类。
2. 操作数组的实用功能在 java .util.Arrays 类中,有许多 static 静态方法,提供了操作数组的一些基本功能:equals() 方法 ---- 用于比较两个数组是否相等,相等的条件是两个数组的元素个数必须相等,并且对应位置的元素也相等。
fill() 方法 ---- 用以某个值填充整个数组,这个方法有点笨。
asList() 方法 ---- 接受任意的数组为参数,将其转变为 List 容器。
JAVA超的笔试题目
1.(单选题)如果要按照特定顺序保存元素,应当使用的容器是()A.HashMapB.ArrayListC.TreeSetD.数组正确答案:C本题考查TreeSet保存元素,会以特定的顺序保存元素选项C正确。
2.(单选题)下列关于变量作用域的说法中,正确的是()。
A.在main()方法中任何位置定义的变量,其作用域为整个main()方法B.块中定义的变量,在块外也是可以使用的C.变量的作用域为:从定义处开始,到变量所在块结束位置D.变量的作用域不受块的限制正确答案:C本题考查变量的作用域 Java中变量的作用域以语句块为标志,变量只在定义它的语句块内有效。
选项C正确。
3.(单选题)在Java中,如下代码的正确结果为()。
double dou = 5/2; System.out.println(dou);A.2B.2.5C.0D.2.0正确答案:D此题目考查的是Java中的算术运算符。
在本题是,因为5和2都是int类型,所以相除的结果会舍弃小数位,只保留整数位的2,将int类型的2赋值给double 型的变量,进而结果为2.0,所以本题选项D是正确的。
4.(单选题)关于构造方法,下面说法正确的是()。
A.构造方法不能带有参数B.构造方法的名称必须和类名相同C.构造方法可以定义返回值类型D.构造方法不能重载正确答案:B本题考查构造方法的概念 Java中的构造方法,是用来初始化成员变量和创建对象的。
构造方法的语法要求如下: * 构造方法方法名必须与类名相同 * 构造方法可以有参数,可以根据传参的方式,初始化对象的成员变量 * 构造方法的定义不能有返回值类型,即使void也不可以使用 * 一个类可以定义多个构造方法,方法名都与类名相同,参数不同,即为重载根据以上描述,本题只有选项B正确5.(单选题)在Java中,关于抽象方法的说法正确的是()。
A.抽象方法可以有方法体。
B.抽象类中的方法都是抽象方法。
C.抽象方法可以出现在非抽象类中。
Java8List对象转Set、Map(高级)、排序、分组、统计
Java8List对象转Set、Map(⾼级)、排序、分组、统计实体类import lombok.Getter;import lombok.Setter;@Getter@Setterpublic class Student {private int id;private String name;private String score;private int classNo;public Student(int id, String name, String score, int classNo) {this.id = id; = name;this.score = score;this.classNo = classNo;}}Mainimport com.demo.entity.Student;import java.util.*;import java.util.stream.Collectors;public class Main {private static List<String> simpleList = new ArrayList<>();private static List<Student> normalList = new ArrayList<>();static {simpleList.add("apple");simpleList.add("apple");simpleList.add("banana");simpleList.add("orange");normalList.add(new Student(1, "Emma", "A", 701));normalList.add(new Student(2, "Larissa", "S", 701));normalList.add(new Student(3, "Sophia", "B", 701));normalList.add(new Student(4, "Ashley", "B", 702));normalList.add(new Student(5, "May", "C", 702));normalList.add(new Student(6, "Hailey", "D", 702));normalList.add(new Student(7, "Kelly", "S", 703));normalList.add(new Student(8, "Amy", "A", 703));normalList.add(new Student(9, "Wesley", "C", 703));}public static void main(String[] args){// TODO}}List<String> 转 Set<String>System.out.println("----------------简单List---------------");simpleList.forEach(System.out::println);System.out.println("----------------简单List转Set---------------");Set<String> simpleSet = new HashSet<>(simpleList);simpleSet.forEach(System.out::println);输出:----------------简单List---------------appleapplebananaorange----------------简单List转Set---------------bananaorangeappleList<Student> 转 Set<Integer>System.out.println("----------------普通List---------------");normalList.forEach(System.out::println);System.out.println("----------------普通List转Set---------------");Set<Integer> normalSet = normalList.stream().map(Student::getClassNo).collect(Collectors.toSet());normalSet.forEach(System.out::println);输出:----------------普通List---------------Student{id=1, name='Emma', score='A', classNo=701}Student{id=2, name='Larissa', score='S', classNo=701}Student{id=3, name='Sophia', score='B', classNo=701}Student{id=4, name='Ashley', score='B', classNo=702}Student{id=5, name='May', score='C', classNo=702}Student{id=6, name='Hailey', score='D', classNo=702}Student{id=7, name='Kelly', score='S', classNo=703}Student{id=8, name='Amy', score='A', classNo=703}Student{id=9, name='Wesley', score='C', classNo=703}----------------普通List转Set---------------701702703List<Student> 转 List<String>System.out.println("----------------普通List转List---------------");List<String> list = normalList.stream().map(Student::getName).collect(Collectors.toList());list.forEach(System.out::println);输出:----------------普通List转List---------------EmmaLarissaSophiaAshleyMayHaileyKellyAmyWesleyList<Student> 转 Map<Integer,Student>System.out.println("----------------普通List转Map---------------");Map<Integer,Student> normalMap = normalList.stream().collect(Collectors.toMap(Student::getId,(b)->b));normalMap.forEach((id, student) -> {System.out.println(id + "::" + student);});输出:----------------普通List转Map---------------1::Student{id=1, name='Emma', score='A', classNo=701}2::Student{id=2, name='Larissa', score='S', classNo=701}3::Student{id=3, name='Sophia', score='B', classNo=701}4::Student{id=4, name='Ashley', score='B', classNo=702}5::Student{id=5, name='May', score='C', classNo=702}6::Student{id=6, name='Hailey', score='D', classNo=702}7::Student{id=7, name='Kelly', score='S', classNo=703}8::Student{id=8, name='Amy', score='A', classNo=703}9::Student{id=9, name='Wesley', score='C', classNo=703}复杂⼀点的转换:(List转Map处理重复key)List<Student> students = new ArrayList<>(normalList);System.out.println("----------------原数据---------------");students.forEach(System.out::println);System.out.println("----------------List<Student>转Map<String, Student>重复key只保留前者---------------");// 重复key处理 (s1, s2) -> s1)Map<Integer, Student> classStudentMap = students.stream().collect(Collectors.toMap(Student::getClassNo, s -> s, (s1, s2) -> s1));classStudentMap.forEach((classNo, student) -> System.out.println(classNo + "::" + student));System.out.println("----------------List<Student>转Map<String, List<Student>>---------------");// 重复key处理成⼀个集合Map<Integer, List<Student>> listMap = students.stream().collect(Collectors.toMap(Student::getClassNo, s -> {List<Student> l = new ArrayList<>();l.add(s);return l;}, (List<Student> s1, List<Student> s2) -> {s1.addAll(s2);return s1;}));listMap.forEach((learn, student) -> System.out.println(learn + "::" + student));输出:----------------原数据---------------Student{id=1, name='Emma', score='A', classNo=701}Student{id=2, name='Larissa', score='S', classNo=701}Student{id=3, name='Sophia', score='B', classNo=701}Student{id=4, name='Ashley', score='B', classNo=702}Student{id=5, name='May', score='C', classNo=702}Student{id=6, name='Hailey', score='D', classNo=702}Student{id=7, name='Kelly', score='S', classNo=703}Student{id=8, name='Amy', score='A', classNo=703}Student{id=9, name='Wesley', score='C', classNo=703}----------------List<Student>转Map<String, Student>重复key只保留前者---------------701::Student{id=1, name='Emma', score='A', classNo=701}702::Student{id=4, name='Ashley', score='B', classNo=702}703::Student{id=7, name='Kelly', score='S', classNo=703}----------------List<Student>转Map<String, List<Student>>---------------701::[Student{id=1, name='Emma', score='A', classNo=701}, Student{id=2, name='Larissa', score='S', classNo=701}, Student{id=3, name='Sophia', score='B', classNo=701}] 702::[Student{id=4, name='Ashley', score='B', classNo=702}, Student{id=5, name='May', score='C', classNo=702}, Student{id=6, name='Hailey', score='D', classNo=702}] 703::[Student{id=7, name='Kelly', score='S', classNo=703}, Student{id=8, name='Amy', score='A', classNo=703}, Student{id=9, name='Wesley', score='C', classNo=703}]Map<String, List<Student>>转List<Student> 和 Map<String, List<Student>>转List<List<Student>>List<Student> students = new ArrayList<>(normalList);// 先分组,准备好数据Map<String, List<Student>> grouping = students.stream().collect(Collectors.groupingBy(Student::getScore));System.out.println("----------------Map<String, List<Student>>转List<Student>---------------");// 把map的values全部拆出来List<Student> collect = grouping.entrySet().stream().flatMap(map -> map.getValue().stream()).collect(Collectors.toList());collect.forEach(System.out::println);System.out.println("----------------Map<String, List<Student>>转List<List<Student>>---------------");// 只要map的value,但是不改变格式grouping.values().forEach(System.out::println);输出:----------------Map<String, List<Student>>转List<Student>---------------Student{id=1, name='Emma', score='A', classNo=701}Student{id=8, name='Amy', score='A', classNo=703}Student{id=3, name='Sophia', score='B', classNo=701}Student{id=4, name='Ashley', score='B', classNo=702}Student{id=5, name='May', score='C', classNo=702}Student{id=9, name='Wesley', score='C', classNo=703}Student{id=2, name='Larissa', score='S', classNo=701}Student{id=7, name='Kelly', score='S', classNo=703}Student{id=6, name='Hailey', score='D', classNo=702}----------------Map<String, List<Student>>转List<List<Student>>---------------[Student{id=1, name='Emma', score='A', classNo=701}, Student{id=8, name='Amy', score='A', classNo=703}][Student{id=3, name='Sophia', score='B', classNo=701}, Student{id=4, name='Ashley', score='B', classNo=702}][Student{id=5, name='May', score='C', classNo=702}, Student{id=9, name='Wesley', score='C', classNo=703}][Student{id=2, name='Larissa', score='S', classNo=701}, Student{id=7, name='Kelly', score='S', classNo=703}][Student{id=6, name='Hailey', score='D', classNo=702}]分组List<Student> students = new ArrayList<>(normalList);System.out.println("----------------分组---------------");// 根据key分组Map<String, List<Student>> grouping = students.stream().collect(Collectors.groupingBy(Student::getScore));grouping.forEach((score, student) -> System.out.println(score + "::" + student));System.out.println("----------------按照多个属性分组---------------");// 根据多个key的组合分组grouping = students.stream().collect(Collectors.groupingBy( e -> e.getClassNo() + "_" + e.getScore()));grouping.forEach((learn, student) -> System.out.println(learn + "::" + student));输出:A::[Student{id=1, name='Emma', score='A', classNo=701}, Student{id=8, name='Amy', score='A', classNo=703}] B::[Student{id=3, name='Sophia', score='B', classNo=701}, Student{id=4, name='Ashley', score='B', classNo=702}] C::[Student{id=5, name='May', score='C', classNo=702}, Student{id=9, name='Wesley', score='C', classNo=703}] S::[Student{id=2, name='Larissa', score='S', classNo=701}, Student{id=7, name='Kelly', score='S', classNo=703}] D::[Student{id=6, name='Hailey', score='D', classNo=702}]----------------按照多个属性分组---------------702_C::[Student{id=5, name='May', score='C', classNo=702}]703_A::[Student{id=8, name='Amy', score='A', classNo=703}]702_B::[Student{id=4, name='Ashley', score='B', classNo=702}]701_S::[Student{id=2, name='Larissa', score='S', classNo=701}]703_C::[Student{id=9, name='Wesley', score='C', classNo=703}]703_S::[Student{id=7, name='Kelly', score='S', classNo=703}]702_D::[Student{id=6, name='Hailey', score='D', classNo=702}]701_B::[Student{id=3, name='Sophia', score='B', classNo=701}]701_A::[Student{id=1, name='Emma', score='A', classNo=701}]排序List<Student>根据名字排序System.out.println("----------------List排序---------------");// 虽然这⾥是浅拷贝,但是只影响修改⽽不影响排序List<Student> students = new ArrayList<>(normalList);Collections.sort(students, paring(Student::getName));// ⽐上⾯更简洁// students.sort(paring(Student::getName));students.forEach(System.out::println);输出:----------------List排序---------------Student{id=8, name='Amy', score='A', classNo=703}Student{id=4, name='Ashley', score='B', classNo=702}Student{id=1, name='Emma', score='A', classNo=701}Student{id=6, name='Hailey', score='D', classNo=702}Student{id=7, name='Kelly', score='S', classNo=703}Student{id=2, name='Larissa', score='S', classNo=701}Student{id=5, name='May', score='C', classNo=702}Student{id=3, name='Sophia', score='B', classNo=701}Student{id=9, name='Wesley', score='C', classNo=703}List<String>排序System.out.println("----------------简单List排序---------------");List<String> list = new ArrayList<>(simpleList);System.out.println("----------------正序---------------");list.sort((a,b) -> pareTo(b));// 更简洁的⽅式// list.sort(Comparator.naturalOrder());list.forEach(System.out::println);System.out.println("----------------倒序---------------");list.sort(Comparator.reverseOrder());list.forEach(System.out::println);输出:----------------简单List排序-------------------------------正序---------------appleapplebananaorange----------------倒序---------------orangebananaappleappleMap排序List<Student> students = new ArrayList<>(normalList);// 先按照成绩分组,准备好数据Map<String, List<Student>> grouping = students.stream().collect(Collectors.groupingBy(Student::getScore)); System.out.println("----------------Map<String, List<Student>>排序---------------");Map<String, List<Student>> result = new LinkedHashMap<>();// Map的key有特殊处理grouping.entrySet().stream().sorted((o1,o2) -> {Integer k1 = getWeight(o1.getKey());Integer k2 = getWeight(o2.getKey());return pareTo(k2);}).forEachOrdered(x -> result.put(x.getKey(), x.getValue()));result.forEach((learn, student) -> System.out.println(learn + "::" + student));System.out.println("----------------");Map<String, List<Student>> result2 = new LinkedHashMap<>();// 仅仅按照key排序grouping.entrySet().stream().sorted(paringByKey()).forEachOrdered(x -> result2.put(x.getKey(), x.getValue()));result2.forEach((learn, student) -> System.out.println(learn + "::" + student));System.out.println("----------------");Map<String, List<Student>> result3 = new LinkedHashMap<>();// 等价第⼀个,只是省去了getKey⽅法grouping.entrySet().stream().sorted(paringByKey((o1,o2) -> {Integer k1 = getWeight(o1);Integer k2 = getWeight(o2);return pareTo(k2);})).forEachOrdered(x -> result3.put(x.getKey(), x.getValue()));result3.forEach((learn, student) -> System.out.println(learn + "::" + student));权重⽅法:/*** 不同成绩有不同的排序权重* @param score* @return*/public static Integer getWeight(String score){switch (score){case "S": return 1;case "A": return 2;case "B": return 3;case "C": return 2;case "D": return 2;}}输出:----------------Map<String, List<Student>>排序---------------S::[Student{id=2, name='Larissa', score='S', classNo=701}, Student{id=7, name='Kelly', score='S', classNo=703}]A::[Student{id=1, name='Emma', score='A', classNo=701}, Student{id=8, name='Amy', score='A', classNo=703}]C::[Student{id=5, name='May', score='C', classNo=702}, Student{id=9, name='Wesley', score='C', classNo=703}]D::[Student{id=6, name='Hailey', score='D', classNo=702}]B::[Student{id=3, name='Sophia', score='B', classNo=701}, Student{id=4, name='Ashley', score='B', classNo=702}]----------------A::[Student{id=1, name='Emma', score='A', classNo=701}, Student{id=8, name='Amy', score='A', classNo=703}]B::[Student{id=3, name='Sophia', score='B', classNo=701}, Student{id=4, name='Ashley', score='B', classNo=702}]C::[Student{id=5, name='May', score='C', classNo=702}, Student{id=9, name='Wesley', score='C', classNo=703}]D::[Student{id=6, name='Hailey', score='D', classNo=702}]S::[Student{id=2, name='Larissa', score='S', classNo=701}, Student{id=7, name='Kelly', score='S', classNo=703}]----------------S::[Student{id=2, name='Larissa', score='S', classNo=701}, Student{id=7, name='Kelly', score='S', classNo=703}]A::[Student{id=1, name='Emma', score='A', classNo=701}, Student{id=8, name='Amy', score='A', classNo=703}]C::[Student{id=5, name='May', score='C', classNo=702}, Student{id=9, name='Wesley', score='C', classNo=703}]D::[Student{id=6, name='Hailey', score='D', classNo=702}]B::[Student{id=3, name='Sophia', score='B', classNo=701}, Student{id=4, name='Ashley', score='B', classNo=702}]如果你要倒序的话System.out.println("----------------倒序----------------");Map<String, List<Student>> result4 = new LinkedHashMap<>();// 仅仅按照key排序grouping.entrySet().stream().sorted(Map.Entry.<String, List<Student>>comparingByKey().reversed()).forEachOrdered(x -> result4.put(x.getKey(), x.getValue()));result4.forEach((learn, student) -> System.out.println(learn + "::" + student));System.out.println("----------------");Map<String, List<Student>> result5 = new LinkedHashMap<>();// 等价第⼀个,只是省去了getKey⽅法grouping.entrySet().stream().sorted(Map.Entry.<String, List<Student>>comparingByKey((o1,o2) -> {Integer k1 = getWeight(o1);Integer k2 = getWeight(o2);return pareTo(k2);}).reversed()).forEachOrdered(x -> result5.put(x.getKey(), x.getValue()));result5.forEach((learn, student) -> System.out.println(learn + "::" + student));输出:----------------倒序----------------S::[Student{id=2, name='Larissa', score='S', classNo=701}, Student{id=7, name='Kelly', score='S', classNo=703}]D::[Student{id=6, name='Hailey', score='D', classNo=702}]C::[Student{id=5, name='May', score='C', classNo=702}, Student{id=9, name='Wesley', score='C', classNo=703}]B::[Student{id=3, name='Sophia', score='B', classNo=701}, Student{id=4, name='Ashley', score='B', classNo=702}]A::[Student{id=1, name='Emma', score='A', classNo=701}, Student{id=8, name='Amy', score='A', classNo=703}]----------------B::[Student{id=3, name='Sophia', score='B', classNo=701}, Student{id=4, name='Ashley', score='B', classNo=702}]A::[Student{id=1, name='Emma', score='A', classNo=701}, Student{id=8, name='Amy', score='A', classNo=703}]C::[Student{id=5, name='May', score='C', classNo=702}, Student{id=9, name='Wesley', score='C', classNo=703}]D::[Student{id=6, name='Hailey', score='D', classNo=702}]S::[Student{id=2, name='Larissa', score='S', classNo=701}, Student{id=7, name='Kelly', score='S', classNo=703}]统计String s = "15, 11, 01, 03, 07, 12, 15, 12, 15, 02, 07, 14, 01, 03, 04, 05, 09, 04, 06, 12, 04, 07, 13, 10, 04, 14, 13, 11, 10, 16, 16, 04, 15, 03, 16, 08, 10, 05, 08, 11, 16, 04, 13, 07, 14, 06, 14, 10, 15, 02, 09, 16, 08, 11, 10, 01, 16, 12, 06, 01, 12, 0 List<Integer> list = new ArrayList<>();for(String i : s.split(", ")){list.add(Integer.valueOf(i));}// 平均值System.out.println(list.stream().mapToDouble(value -> value).average());// 总值System.out.println(list.stream().mapToDouble(value -> value).sum());// 数据数量System.out.println(list.stream().mapToDouble(value -> value).count());// 最⼤System.out.println(list.stream().mapToDouble(value -> value).max());// 最⼩System.out.println(list.stream().mapToDouble(value -> value).min());// 去重后计算总值System.out.println(list.stream().mapToDouble(value -> value).distinct().sum());也有其他数据类型的⽅法结果:OptionalDouble[9.121212*********]602.066OptionalDouble[16.0]OptionalDouble[1.0]136.0=========================================分割线=================================================Lambda表达式格式:(params) -> {expression}⽐如我们实现⼀个Runnable接⼝Runnable run = new Runnable() {@Overridepublic void run() {// TODO}};使⽤Lambda表达式Runnable run = () -> {// TODO};⽽可以使⽤Lambda的接⼝也是有限制的,即只能有⼀个⽅法。
List、Set、Map有什么异同(详解)
List、Set、Map有什么异同(详解)引⾔:Java集合框架提供了⼀套性能优良、使⽤⽅便的接⼝和类,它们位于java.util包中Java集合框架(常⽤接⼝):Collection 接⼝存储⼀组不唯⼀,⽆序的对象(⽗类接⼝)List 接⼝存储⼀组不唯⼀,有序(插⼊顺序)的对象Set 接⼝存储⼀组唯⼀,⽆序的对象Map接⼝存储⼀组键值对象,提供key到value的映射⼀、List接⼝List是⼀个继承于Collection的接⼝,即List是集合中的⼀种。
List是有序的队列,List中的每⼀个元素都有⼀个索引;第⼀个元素的索引值是0,往后的元素的索引值依次+1。
和Set 不同,List中允许有重复的元素。
实现List接⼝的集合主要有:ArrayList、LinkedList、Vector、Stack。
1.可以允许重复的对象。
2.可以插⼊多个null元素。
3.是⼀个有序容器,保持了每个元素的插⼊顺序,输出的顺序就是插⼊的顺序。
4.常⽤的实现类有 ArrayList、LinkedList 和 Vector。
ArrayList 最为流⾏,它提供了使⽤索引的随意访问,⽽LinkedList 则对于经常需要从 List 中添加或删除元素的场合更为合适。
ArrayListArrayList是⼀个动态数组,也是我们最常⽤的集合。
它允许任何符合规则的元素插⼊甚⾄包括null。
每⼀个ArrayList都有⼀个初始容量:private static final int DEFAULT_CAPACITY = 10;随着容器中的元素不断增加,容器的⼤⼩也会随着增加。
在每次向容器中增加元素的同时都会进⾏容量检查,当快溢出时,就会进⾏扩容操作。
所以如果我们明确所插⼊元素的多少,最好指定⼀个初始容量值,避免过多的进⾏扩容操作⽽浪费时间、效率。
size、isEmpty、get、set、iterator 和 listIterator 操作都以固定时间运⾏。
java list和set使用场景
java list和set使用场景Java中的List和Set是两种常用的集合类型,它们各自有着不同的使用场景和特点。
本文将分别介绍List和Set的使用场景,并对其进行详细解析。
一、List的使用场景List是一种有序的集合,可以存储重复的元素。
List的使用场景如下:1. 数据存储:List可以用来存储一组有序的数据,比如存储学生成绩、员工工资等。
List中的元素可以按照插入的顺序进行访问,也可以根据索引进行随机访问。
2. 数据检索:List提供了丰富的方法来检索、访问和修改集合中的元素。
可以根据索引、元素值或自定义条件来进行检索。
3. 数据排序:List可以对集合中的元素进行排序,通过调用Collections.sort(List)方法可以对List进行升序排序。
4. 数据操作:List提供了一系列方法来操作集合中的元素,比如添加元素、删除元素、替换元素等。
5. 数据重复:List允许存储重复的元素,因此适用于需要存储重复数据的场景。
二、Set的使用场景Set是一种不允许存储重复元素的集合,它保证了集合中的元素是唯一的。
Set的使用场景如下:1. 数据去重:Set可以用来去除重复的数据,比如从一个数组或List中去除重复元素。
2. 数据筛选:Set可以用来筛选出满足特定条件的元素,比如筛选出一组不重复的IP地址或手机号码。
3. 数据集合:Set可以用来将多个集合合并成一个集合,并且保证合并后的集合中没有重复的元素。
4. 数据判断:Set提供了判断元素是否存在于集合中的方法,可以快速判断一个元素是否在集合中。
5. 数据去序:Set中的元素是无序的,因此适用于不关心元素顺序的场景。
三、List和Set的区别List和Set在使用上有一些区别,主要体现在以下几个方面:1. 元素顺序:List是有序的,可以根据元素的插入顺序进行访问;而Set是无序的,不保证元素的顺序。
2. 元素重复:List允许存储重复的元素;而Set不允许存储重复的元素,保证集合中的元素是唯一的。
java集合使用场景
java集合使用场景Java集合是Java编程中常用的数据结构,它提供了一组方法和类来处理和存储数据。
Java集合框架主要包括List、Set、Map、Queue等接口和它们的实现类,每个类都有其自身的使用场景。
下面将介绍一些常见的Java集合使用场景。
1. List集合:List是有序、可重复的集合。
常用的List实现类有ArrayList和LinkedList。
使用场景包括但不限于:- 当需要按照特定顺序存储多个元素,并且需要能够随机访问元素时,可以使用ArrayList。
- 当需要频繁执行插入、删除操作,而不是随机访问元素时,可以使用LinkedList。
2. Set集合:Set是无序、不可重复的集合。
常用的Set实现类有HashSet和TreeSet。
使用场景包括但不限于:- 当需要存储一组唯一的元素,并且不关心元素的顺序时,可以使用HashSet。
- 当需要按照一定顺序存储元素,并且不希望有重复元素时,可以使用TreeSet。
3. Map集合:Map是存储键值对的集合,每个键对应一个唯一的值。
常用的Map实现类有HashMap和TreeMap。
使用场景包括但不限于:- 当需要根据键快速查找对应的值时,可以使用HashMap。
- 当需要按照键的顺序存储键值对时,可以使用TreeMap。
4. Queue集合:Queue是一种特殊的集合,它遵循先进先出(FIFO)的原则。
常用的Queue实现类有LinkedList和ArrayDeque。
使用场景包括但不限于:- 当需要按照先后顺序处理元素,并且从队列的一端插入、另一端删除元素时,可以使用LinkedList。
- 当需要高效地在队列两端插入、删除元素时,可以使用ArrayDeque。
除了以上常见的使用场景外,还可以根据具体需求选择其他集合类。
例如,如果需要对集合进行高效的查找操作,可以使用HashSet或TreeSet实现类;如果需要对集合进行高效的插入、删除操作,可以使用LinkedList或ArrayDeque实现类;如果需要对集合进行高效的键值对查找或按照键排序的操作,可以使用HashMap或TreeMap实现类等。
List,set,Map 的用法和区别等
├List│├LinkedList│├ArrayList│└Vector│└Stack└SetMap├Hashtable├HashMap└WeakHashMapCollection接口Collection是最基本的集合接口,一个Collection代表一组Object,即Collection的元素(Elements)。
一些Collection允许相同的元素而另一些不行。
一些能排序而另一些不行。
Java SDK不提供直接继承自Collection的类,Java SDK提供的类都是继承自Collection 的“子接口”如List和Set。
所有实现Collection接口的类都必须提供两个标准的构造函数:无参数的构造函数用于创建一个空的Collection,有一个Collection参数的构造函数用于创建一个新的Collection,这个新的Collection与传入的Collection有相同的元素。
后一个构造函数允许用户复制一个如何遍历Collection中的每一个元素?不论Collection的实际类型如何,它都支持一个iterator()的方法,该方法返回一个迭代子,使用该迭代子即可逐一访问Collection中每一个元素。
典型的用法如下:Iterator it = collection.iterator(); // 获得一个迭代子while(it.hasNext()) {Object obj = it.next(); // 得到下一个元素}由Collection接口派生的两个接口是List和Set。
List接口List是有序的Collection,使用此接口能够精确的控制每个元素插入的位置。
用户能够使用索引(元素在List中的位置,类似于数组下标)来访问List中的元素,这类似于Java的数组。
和下面要提到的Set不同,List允许有相同的元素。
除了具有Collection接口必备的iterator()方法外,List还提供一个listIterator()方法,返回一个ListIterator接口,和标准的Iterator接口相比,ListIterator多了一些add()之类的方法,允许添加,删除,设定元素,还能向前或向后遍历。
List、Map、set的加载因子,默认初始容量和扩容增量
List、Map、set的加载因⼦,默认初始容量和扩容增量⾸先,这三个概念说下。
初始⼤⼩,就是创建时可容纳的默认元素个数;加载因⼦,表⽰某个阀值,⽤0~1之间的⼩数来表⽰,当已有元素占⽐达到这个阀值后,底层将进⾏扩容操作;扩容⽅式,即指定每次扩容后的⼤⼩的规则,⽐如翻倍等。
当底层实现涉及到扩容时,容器或重新分配⼀段更⼤的连续内存(如果是离散分配则不需要重新分配,离散分配都是插⼊新元素时动态分配内存),要将容器原来的数据全部复制到新的内存上,这⽆疑使效率⼤⼤降低。
加载因⼦的系数⼩于等于1,意指即当元素个数超过容量长度*加载因⼦的系数时,进⾏扩容。
另外,扩容也是有默认的倍数的,不同的容器扩容情况不同。
List 元素是有序的、可重复ArrayList、Vector默认初始容量为10Vector:线程安全,但速度慢 底层数据结构是数组结构 加载因⼦为1:即当元素个数超过容量长度时,进⾏扩容 扩容增量:原容量的 1倍 如 Vector的容量为10,⼀次扩容后是容量为20ArrayList:线程不安全,查询速度快 底层数据结构是数组结构 扩容增量:原容量的 0.5倍 如 ArrayList的容量为10,⼀次扩容后是容量为15Set(集) 元素⽆序的、不可重复。
HashSet:线程不安全,存取速度快 底层实现是⼀个HashMap(保存数据),实现Set接⼝ 默认初始容量为16(为何是16,见下⽅对HashMap的描述) 加载因⼦为0.75:即当元素个数超过容量长度的0.75倍时,进⾏扩容 扩容增量:原容量的 1 倍 如 HashSet的容量为16,⼀次扩容后是容量为32构造⽅法摘要HashSet()HashSet(int initialCapacity)构造⼀个新的空 set,其底层 HashMap 实例具有指定的初始容量和默认的加载因⼦(0.75)。
HashSet hs=new HashSet(1);所以可见 HashSet类,创建对象的时候是可以的制定容量的⼤⼩的,期中第⼆个就具有这个⼯功能。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
Java容器分析--数组数组是Java语言内置的类型,除此之外,Java有多种保存对象引用的方式。
Java类库提供了一套相当完整的容器类,使用这些类的方法可以保存和操纵对象。
下面分别进行讨论,在研究Java容器类之前,先了解一下Java数组的基本功能和特性。
1. 数组的基本特性数组与其它种类的容器(List/Set/Map)之间的区别在于效率、确定的类型和保存基本类型数据的能力。
数组是一种高效的存储和随机访问对象引用序列的方式,使用数组可以快速的访问数组中的元素。
但是当创建一个数组对象(注意和对象数组的区别)后,数组的大小也就固定了,当数组空间不足的时候就再创建一个新的数组,把旧的数组中所有的引用复制到新的数组中。
Java中的数组和容器都需要进行边界检查,如果越界就会得到一个RuntimeException 异常。
这点和C++中有所不同,C++中vector的操作符[]不会做边界检查,这在速度上会有一定的提高,Java的数组和容器会因为时刻存在的边界检查带来一些性能上的开销。
Java中通用的容器类不会以具体的类型来处理对象,容器中的对象都是以Object类型处理的,这是Java中所有类的基类。
另外,数组可以保存基本类型,而容器不能,它只能保存任意的Java对象。
一般情况下,考虑到效率与类型检查,应该尽可能考虑使用数组。
如果要解决一般化的问题,数组可能会受到一些限制,这时可以使用Java提供的容器类。
2. 操作数组的实用功能在java.util.Arrays类中,有许多static静态方法,提供了操作数组的一些基本功能:equals()方法----用于比较两个数组是否相等,相等的条件是两个数组的元素个数必须相等,并且对应位置的元素也相等。
fill()方法----用以某个值填充整个数组,这个方法有点笨。
asList()方法----接受任意的数组为参数,将其转变为List容器。
binarySearch()方法----用于在已经排序的数组中查找元素,需要注意的是必须是已经排序过的数组。
当Arrays.binarySearch()找到了查找目标时,该方法将返回一个等于或大于0的值,否则将返回一个负值,表示在该数组目前的排序状态下此目标元素所应该插入的位置。
负值的计算公式是“-x-1”。
x指的是第一个大于查找对象的元素在数组中的位置,如果数组中所有的元素都小于要查找的对象,则x = a.size()。
如果数组中包含重复的元素,则无法保证找到的是哪一个元素,如果需要对没有重复元素的数组排序,可以使用TreeSet或者LinkedHashSet。
另外,如果使用Comparator排序了某个对象数组,在使用该方法时必须提供同样的Comparator类型的参数。
需要注意的是,基本类型数组无法使用Comparator进行排序。
sort()方法----对数组进行升序排序。
在Java标准类库中,另有static方法System.arraycopy()用来复制数组,它针对所有类型做了重载。
3. 数组的排序在Java1.0和1.1两个版本中,类库缺少基本的算法操作,包括排序的操作,Java2对此进行了改善。
在进行排序的操作时,需要根据对象的实际类型执行比较操作,如果为每种不同的类型各自编写一个不同的排序方法,将会使得代码很难被复用。
一般的程序设计目标应是“将保持不变的事物与会发改变的事物相分离”。
在这里,不变的是通用的排序算法,变化的是各种对象相互比较的方式。
Java有两种方式来实现比较的功能,一种是实现parable接口,该接口只有一个compareTo()方法,并以一个Object类为参数,如果当前对象小于参数则返回负值,如果相等返回零,如果当前对象大于参数则返回正值。
另一种比较方法是采用策略(strategy)设计模式,将会发生变化的代码封装在它自己的类(策略对象)中,再将策略对象交给保持不变的代码中,后者使用此策略实现它的算法。
因此,可以为不同的比较方式生成不同的对象,将它们用在同样的排序程序中。
在此情况下,通过定义一个实现了Comparator接口的类而创建了一个策略,这个策略类有compare()和equals()两个方法,一般情况下实现compare()方法即可。
使用上述两种方法即可对任意基本类型的数组进行排序,也可以对任意的对象数组进行排序。
再提示一遍,基本类型数组无法使用Comparator进行排序。
Java标准类库中的排序算法针对排序的类型进行了优化——针对基本类型设计了“快速排序”,针对对象设计的“稳定归并排序”。
一般不用担心其性能。
Java容器分析--List和Set容器类可以大大提高编程效率和编程能力,在Java2中,所有的容器都由SUN公司的Joshua Bloch进行了重新设计,丰富了容器类库的功能。
Java2容器类类库的用途是“保存对象”,它分为两类:Collection----一组独立的元素,通常这些元素都服从某种规则。
List必须保持元素特定的顺序,而Set不能有重复元素。
Map----一组成对的“键值对”对象,即其元素是成对的对象,最典型的应用就是数据字典,并且还有其它广泛的应用。
另外,Map可以返回其所有键组成的Set和其所有值组成的Collection,或其键值对组成的Set,并且还可以像数组一样扩展多维Map,只要让Map中键值对的每个“值”是一个Map即可。
1.迭代器迭代器是一种设计模式,它是一个对象,它可以遍历并选择序列中的对象,而开发人员不需要了解该序列的底层结构。
迭代器通常被称为“轻量级”对象,因为创建它的代价小。
Java中的Iterator功能比较简单,并且只能单向移动:(1) 使用方法iterator()要求容器返回一个Iterator。
第一次调用Iterator的next()方法时,它返回序列的第一个元素。
(2) 使用next()获得序列中的下一个元素。
(3) 使用hasNext()检查序列中是否还有元素。
(4) 使用remove()将迭代器新返回的元素删除。
Iterator是Java迭代器最简单的实现,为List设计的ListIterator具有更多的功能,它可以从两个方向遍历List,也可以从List中插入和删除元素。
2.List的功能方法List(interface): 次序是List最重要的特点;它确保维护元素特定的顺序。
List为Collection添加了许多方法,使得能够向List中间插入与移除元素(只推荐LinkedList使用)。
一个List 可以生成ListIterator,使用它可以从两个方向遍历List,也可以从List中间插入和删除元素。
ArrayList: 由数组实现的List。
它允许对元素进行快速随机访问,但是向List中间插入与移除元素的速度很慢。
ListIterator只应该用来由后向前遍历ArrayList,而不是用来插入和删除元素,因为这比LinkedList开销要大很多。
LinkedList: 对顺序访问进行了优化,向List中间插入与删除得开销不大,随机访问则相对较慢(可用ArrayList代替)。
它具有方法addFirst()、addLast()、getFirst()、getLast()、removeFirst()、removeLast(),这些方法(没有在任何接口或基类中定义过)使得LinkedList可以当作堆栈、队列和双向队列使用。
3.Set的功能方法Set(interface): 存入Set的每个元素必须是唯一的,因为Set不保存重复元素。
加入Set的Object必须定义equals()方法以确保对象的唯一性。
Set与Collection有完全一样的接口。
Set 接口不保证维护元素的次序。
HashSet: 为快速查找而设计的Set。
存入HashSet的对象必须定义hashCode()。
TreeSet: 保持次序的Set,底层为树结构。
使用它可以从Set中提取有序的序列。
LinkedHashSet: 具有HashSet的查询速度,且内部使用链表维护元素的顺序(插入的次序)。
于是在使用迭代器遍历Set时,结果会按元素插入的次序显示。
HashSet采用散列函数对元素进行排序,这是专门为快速查询而设计的;TreeSet采用红黑树的数据结构进行排序元素;LinkedHashSet内部使用散列以加快查询速度,同时使用链表维护元素的次序,使得看起来元素是以插入的顺序保存的。
需要注意的是,生成自己的类时,Set需要维护元素的存储顺序,因此要实现Comparable接口并定义compareTo()方法。
Java容器分析--Map标准的Java类库中包含了几种类型的Map,它们都拥有同样的基本接口Map,但是行为特性各不相同,主要表现在效率、键值对的保存、元素呈现次序、对象的保存周期和判定键是否等价的策略等方面。
1.Map的功能方法Map(interface): 维护label和value的关联性,使得可以通过label查找value。
HashMap: Map基于散列表的实现,取代了Hashtable。
插入和查询label/value的开销是固定的,并且可以通过构造器设置容量和负载因子,以调整容器的性能。
LinkedHashMap: 在HashMap的基础上做了一些改进,在迭代遍历它时,取得label/value的顺序是其插入的次序,或者是最近最少使用(LRU)的次序,速度上比HashMap要慢一点,但在迭代访问时速度会更快,主要原因是它使用了链表维护内部次序。
TreeMap: 查看label或label/value时,元素会被排序,其次序由Comparable或Comparator 决定,因此查询所得到的结果是经过排序的。
另外,它是唯一带有subMap()方法的Map具体类,即返回一个子树。
它也是SortedMap接口的唯一实现,subMap()方法也是从该接口继承的。
WeakHashMap: Weak Key映射,允许释放映射所指向的对象。
当映射之外没有引用指向某个label时,此label可以被垃圾收集器回收。
IdentityHashMap: 使用==代替equals()对label进行比较的散列映射。
2.hashCode()当使用标准库中的类Integer作为HashMap的label时,程序能够正常运行,但是使用自己创建的类作为HashMap的label时,通常犯一个错误。
在HashMap中通过label查找value时,实际上是计算label对象地址的散列码来确定value的。
一般情况下,我们是使用基类Object的方法hashCode()来生成散列码,它默认是使用对象的地址来计算的,因此由第一个对象new Apple(5)和第二个对象new Apple(5)生成的散列码是不同的,不能完成正确的查找。