赞
踩
当我们需要保存一组一样(类型相同)的元素的时候,我们应该使用一个容器来存储,数组就是这样一个容器。
● 数组的特点:
数组是一组数据类型相同的元素集合;
创建数组时,必须给定长度,而且一旦创建长度不能改变;
一旦数组装满元素,就需要创建一个新的数组,将元素复制过去;
● 数组的局限:
如果装满了,就需要数组复制;
当我们需要从数组中删除或添加一个元素,需要移动后面的元素;
● 集合的作用:
在开发实践中,我们需要一些能够动态增长长度的容器来保存我们的数据,java中为了解决数据存储单一的情况,java中就提供了不同结构的集合类,可以让我们根据不同的场景进行数据存储的选择,如Java中提供了 数组实现的集合,链表实现的集合,哈希结构,树结构等。
分类
● 单列集合:一次放进去一个值 ( 对象 )
Collection接口:定义了单列集合共有的方法,其子接口Set和List分别定义了存储方式
● List:可以有重复元素
● Set:不可以有重复元素
● 双列集合: 键 值
集合体系概述: Java的集合框架是由很多接口、抽象类、具体类组成的,都位于java.util包中。
概念:List继承了Collection接口,有三个实现的类,分别是:
● ArrayList 数组列表
● LinkedList 链表列表
● Vector 数组列表 (是线程安全的)
● 底层有一个数组,可以动态扩展数组长度,并提供了一系列方法操作。
特点: 查询快,在中间增加 / 删除慢
注意:集合容器类中默认可以添加Object类型,但是一般建议一个集合对象只保存同一种类型,若保存多个类型后期处理时涉及类型转换问题;所以我们可以通过泛型来加以控制。
● 泛型:声明类型时可以自定义参数类型
ArraysList <E>
ArrayList的常用方法:
- ArrayList<String> arrayList = new ArrayList();
- arrayList.add("a");
- arrayList.add("b");
- arrayList.add("c");
- arrayList.add(3,"a");
-
- arrayList.remove("a"); //根据内容删除匹配的第一个元素,返回值为boolean
- arrayList.remove(1); //删除并返回指定位置的元素
- arrayList.get(1); //获取指定位置的元素
- arrayList.set(1,"X"); //替换并返回指定位置的元素
- arrayList.clear(); //清空集合中的元素
- arrayList.isEmpty(); //判断集合元素是否为空
- arrayList.contains("c"); //判断是否包含指定元素
- arrayList.size(); //返回集合中元素个数
● 底层是一个链表结构,查询效率低,但增加 / 删除元素快
LinkedList中方法及功能和ArrayList中的方法大致相同,只不过LinkedList中多了关于队列和栈相关的操作方法。
和ArrayList一样,底层也是数组实现,不同的是Vector的方法默认加了锁,线程是安全的。
List集合遍历方式有四种:
● 1. for循环遍历 ● 2. 增强for循环遍历
- //List集合遍历方式1:for循环
- for (int i = 0; i < arrayList.size(); i++) {
- if ("a".equals(arrayList.get(i))){
- arrayList.remove("a");//允许修改集合元素
- i--; // 要注意索引的变化与元素位置的移动
- }
-
- }
- System.out.println(arrayList);
- System.out.println("----------------");
-
- //List集合遍历方式2:增强for循环
- for (String s:arrayList) {
- System.out.println(s); //不能修改集合元素
- }
● 3. 迭代器遍历 (Iterator)
- //List集合遍历方式3:使用迭代器遍历
- //获得集合对象的迭代器对象
- Iterator<String> it = arrayList.iterator();
- while(it.hasNext()){
- String s = it.next();//获取下一个院系
- if (s.equals("a")) {
- it.remove(); //使用迭代器对象删除元素
- }
- }
- System.out.println(arrayList);
● 4. ListIterator迭代器:
- public static void main(String[] args) {
- ArrayList<String> arrayList = new ArrayList();
- arrayList.add("a");
- arrayList.add("b");
- arrayList.add("c");
- arrayList.add("d");
- /*
- ListIterator迭代器只能对List接口下的实现遍历
- ListIterator(index)可以从指定位置开始向前向后遍历
- */
- ListIterator<String> listIterator= arrayList.listIterator(arrayList.size());
- while(listIterator.hasPrevious()){
- System.out.println(listIterator.previous());//逆序遍历
- }
- }
● Set接口也继承了Collection接口,Set中所储存的元素是不重复的,但是是无序的,Set中的元素没有索引。
● 1. HashSet类中的元素不能重复
- public static void main(String[] args) {
- HashSet set =new HashSet<>();
- set.add("a");
- set.add("a");
- set.add("b");
- set.add("c"); //元素是不重复的
- System.out.println(set);
- }
● 2. HashSet类中的元素是无序的
- public static void main(String[] args) {
- HashSet set =new HashSet<>();
- set.add("c");
- set.add("s");
- set.add("x");
- set.add("d"); //元素是无序的
- System.out.println(set);
-
- set.remove("s");//没有索引,只能根据内容遍历
- set.iterator();//没有索引,要用迭代器遍历
-
- }
在底层会先调用hashCode(),注意,Object中的hashCode()返回的是对象的地址,此时并不会调用;此时调用的是类中重写的hashCode(),返回的是根据内容计算的哈希值,遍历时,会用哈希值先比较是否相等,会提高比较的效率;但哈希值会存在问题:内容不同,哈希值相同;这种情况下再调equals比较内容,这样既保证效率又确保安全。
例:
● 这是错误写法,此时默认调用的是Object类中hashCode( ),返回对象地址
- import java.util.HashSet;
- import java.util.Objects;
-
- public class Student {
- private String name ;
- private String num;
-
-
- public Student(String name, String num) {
- this.name = name;
- this.num = num;
- }
-
- @Override
- public String toString() {
- return "Student{" +
- "name='" + name + '\'' +
- ", num='" + num + '\'' +
- '}';
- }
-
- public static void main(String[] args) {
- HashSet<Student> set = new HashSet<>();
- Student s1 = new Student("小王1","10001");
- Student s2 = new Student("小王2","10002");
- Student s3 = new Student("小王3","10003");
- Student s4 = new Student("小王1","10001");
- set.add(s1);
- set.add(s2);
- set.add(s3);
- set.add(s4);
- System.out.println(set);
-
- }
- }
● 正确写法应该是,在Student类中重写hashCode()和equals()
- package Demo;
-
- import java.util.HashSet;
- import java.util.Objects;
-
- public class Student {
- private String name ;
- private String num;
-
- @Override
- public boolean equals(Object o) {
- if (this == o) return true;
- if (o == null || getClass() != o.getClass()) return false;
- Student student = (Student) o;
- return num.equals(student.num);
- }
-
- @Override
- public int hashCode() {
- return Objects.hash(num);
- }
-
- public Student(String name, String num) {
- this.name = name;
- this.num = num;
- }
-
- @Override
- public String toString() {
- return "Student{" +
- "name='" + name + '\'' +
- ", num='" + num + '\'' +
- '}';
- }
-
- public static void main(String[] args) {
- HashSet<Student> set = new HashSet<>();
- Student s1 = new Student("小王1","10001");
- Student s2 = new Student("小王2","10002");
- Student s3 = new Student("小王3","10003");
- Student s4 = new Student("小王1","10001");
- set.add(s1);
- set.add(s2);
- set.add(s3);
- set.add(s4);
- System.out.println(set);
-
- }
- }
-
这样就能避免重复了(此图是s1,s4重复,但只输出s1)
● 补充:如何快速生成hashCode()和equals()的重写方法?
右键选择Generate,选择equals()and hashCode(),选择重写的属性。
......敬请期待^_^
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。