集合是一个容器,是用来存储和获取数据;
集合类的特点长度可变;
我们学习的是面向对象的编程语言,面向对象的编程语言对事物的描述都是通过对象体现的,为了方便对多个对象进行操作,我们就必须把这多个对象进行存储,而要想存储多个对象,就不能是基本的变量了,应该是一个容器类型的变量。回顾我们学过的知识,有哪些是容器类型的呢?
数组,StringBuilder,它的结果是一个字符串,不一定满足我们的需求,所以我们只能选择数组了,而数组的长度固定,不能适应变化的需求,在这种情况下,Java就提供了集合类供我们使用。
由此可见,集合类的长度是可变的。
Collection是单列集合的顶层接口。
元素Collection 表示一组对象,这些对象也称为 collection 的
特点:
ArrayList()是创建Collection集合的对象,我们采用的是多态的方式,使用的是具体的ArrayList类。因为这个类是最常用的集合类。
Collection:是一种特殊的数据类型,泛型。这里我们会使用就可以了。如何使用呢?
代码:
package Java_study;
import java.util.ArrayList;
import java.util.Collection;
/**
*
* @author 只是甲
* @date 2021-07-02
* @remark 创建Collection集合对象并添加元素
*
*/
public class collection1 {
public static void main(String[] args) {
//创建Collection集合对象
//下面这种写法是JDK7的新特性,看懂就可以,但不建议这种写法,最好还是补全<>中的类型
//Collection c = new ArrayList<>(); //多态的方式
Collection c = new ArrayList<>();
c.add("hello");
c.add("world");
c.add("java");
//输出集合对象
//输出了集合中的元素按照指定格式拼接的内容,说明ArrayList重写了toString()方法
/* [hello, world, java] */
System.out.println(c);
}
}
测试记录:
[hello, world, java]
//添加元素
boolean add(E e);
//从集合中移除元素
boolean remove(Object o);
//清空集合中的元素
void clear();
//判断集合中是否存在指定的元素
boolean contains(Object o);
//判断集合是否为空
boolean isEmpty();
//集合的长度,也就是集合中元素的个数
int size();
代码:
package Java_study;
import java.util.Collection;
import java.util.ArrayList;
/**
*
* @author 只是甲
* @date 2021-07-02
* @remark Collection集合的成员方法
*
*/
public class collection2 {
public static void main(String[] args) {
//创建集合对象
Collection c = new ArrayList<>();
//1.添加元素
System.out.println("add:" + c.add("hello"));
System.out.println("add:" + c.add("world"));
System.out.println("----1------");
//2.从集合中删除元素
System.out.println("remove:" + c.remove("world"));
System.out.println("remove:" + c.remove("haha"));
System.out.println("-----2-----");
//3.清空集合
System.out.println(c);
c.clear();
System.out.println(c);
System.out.println("-----3-----");
//4.判断集合中是否存在指定的元素
c.add("world");
System.out.println("contains:" + c.contains("world"));
System.out.println("contains:" + c.contains("haha"));
System.out.println("-----4-----");
//5. 判断集合是否为空
System.out.println("isEmpty:" + c.isEmpty());
System.out.println("-----5-----");
//6. 集合的长度
System.out.println("size:" + c.size());
System.out.println("----6------");
//输出集合对象
System.out.println(c);
}
}
测试记录:
add:true
add:true
----1------
remove:true
remove:false
-----2-----
[hello]
[]
-----3-----
contains:false
contains:false
-----4-----
isEmpty:false
-----5-----
size:1
----6------
[world]
Collection集合的遍历:
Iterator:
代码:
package Java_study;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.NoSuchElementException;
/**
*
* @author 只是甲
* @date 2021-07-02
* @remark Collection集合的遍历
*
*/
public class collection3 {
public static void main(String[] args) {
//创建集合对象
Collection c1 = new ArrayList();
//添加元素
c1.add(new String("hello"));
c1.add("world");
c1.add("java");
//Iterator iterator()
Iterator it1 = c1.iterator();//返回的是跌抬起接口的实现类的对象
try {
System.out.println(it1.next());//hello
System.out.println(it1.next());//world
System.out.println(it1.next());//java
System.out.println(it1.next());//NoSuchElementException:没有这样的元素异常
} catch (NoSuchElementException e) {
System.out.println("NoSuchElementException:没有这样的元素异常");
}
/* 遍历方式一:iterator.hasNext()
boolean hasNext 如果还有下一个元素则返回true 否则跳出循环,避免NoSuchElementException
*/
System.out.println("===========方式一:iterator.hasNext()===============");
Iterator it2 = c1.iterator();
while (it2.hasNext()) {
String s = it2.next();
System.out.println(s);
}
/* 遍历方式二: try catch 获取NoSuchElementException异常 */
System.out.println("===========方式二:try catch===============");
Iterator it3 = c1.iterator();
iter:
while (true) {
try {
System.out.println(it3.next());
} catch (NoSuchElementException e) {
System.out.println("遍历结束");
break iter;
}
}
/* 遍历方式三: 增强for循环 c1.for根据提示自动生成*/
System.out.println("===========方式三:增强for循环===============");
for (String s : c1) {
System.out.println(s);
}
}
}
测试记录:
hello
world
java
NoSuchElementException:没有这样的元素异常
===========方式一:iterator.hasNext()===============
hello
world
java
===========方式二:try catch===============
hello
world
java
遍历结束
===========方式三:增强for循环===============
hello
world
java
List:
List集合的特点:
代码:
package Java_study;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
/**
*
* @author 只是甲
* @date 2021-07-02
* @remark List集合的特点
*
*/
public class list1 {
public static void main(String[] args) {
//创建集合对象
List list = new ArrayList();
//存储元素
list.add("hello");
list.add("world");
list.add("java");
//遍历集合
Iterator it = list.iterator();
while (it.hasNext()) {
String s = it.next();
System.out.println(s);
}
}
}
测试记录:
hello
world
java
//在指定位置添加元素
void add(int index,E element);
//删除指定位置的元素
E remove(int index);
//获取指定位置的元素
E get(int index);
//修改指定位置的元素
E set(int index,E element);
代码:
package Java_study;
import java.util.ArrayList;
import java.util.List;
/**
*
* @author 只是甲
* @date 2021-04-19
* @remark List集合的特有成员方法
*
*/
public class list2 {
public static void main(String[] args) {
//创建集合对象
List list = new ArrayList();
//使用集成Collection的添加功能
list.add("hello");
list.add("world");
list.add("java");
// 1. void add(int index,E element):在指定位置添加元素
System.out.println("==========1. void add(int index,E element):在指定位置添加元素==========");
try {
list.add(1, "javaee");
list.add(11, "javase");//IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据
} catch (IndexOutOfBoundsException e) {
System.out.println("IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据");
}
System.out.println(list);//[hello, javaee, world, java]
//2. E remove(int index):删除指定位置的元素,返回被删除的元素
System.out.println("==========2. E remove(int index):删除指定位置的元素,返回被删除的元素==========");
try {
System.out.println("remove: " + list.remove(2));
System.out.println("remove: " + list.remove(11));//IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据
} catch (IndexOutOfBoundsException e) {
System.out.println("IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据");
}
System.out.println(list);//[hello, javaee, java]
//3. E get(int index):获取指定位置的元素
System.out.println("==========3. E get(int index):获取指定位置的元素==========");
try {
System.out.println("get:" + list.get(1));
System.out.println("get:" + list.get(11));//IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据
} catch (IndexOutOfBoundsException e) {
System.out.println("IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据");
}
}
}
测试记录:
==========1. void add(int index,E element):在指定位置添加元素==========
IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据
[hello, javaee, world, java]
==========2. E remove(int index):删除指定位置的元素,返回被删除的元素==========
remove: world
IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据
[hello, javaee, java]
==========3. E get(int index):获取指定位置的元素==========
get:javaee
IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据
List集合的遍历方式:
代码:
package Java_study;
import java.util.ArrayList;
import java.util.List;
/**
*
* @author 只是甲
* @date 2021-07-02
* @remark List集合的普通for循环遍历
*
*/
public class list3 {
public static void main(String[] args) {
//创建集合对象
List list = new ArrayList();
//添加循环
list.add("hello");
list.add("world");
list.add("java");
// E get(int index):获取指定位置的元素
try {
System.out.println(list.get(0));
System.out.println(list.get(1));
System.out.println(list.get(2));
System.out.println(list.get(3));//IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据
} catch (IndexOutOfBoundsException e) {
System.out.println("IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据");
}
}
}
测试记录:
hello
world
java
IndexOutOfBoundsException,索引超出边界异常,list中没有这个数据
自定义一个学生类,给出成员变量name和age。
遍历集合的时候,在控制台输出学生对象的成员变量值。
两种方式遍历:
代码:
list4
package Java_study;
/**
*
* @author 只是甲
* @date 2021-07-05
* @remark 实体学生类 List集合的练习存储自定义对象并遍历
*
*/
public class list4 {
private String name;
private int age;
public list4(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;
}
/**
* @remark 重写Object类的toString方法
* @return 学生类的姓名,年龄信息
*/
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
list5
package Java_study;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
/**
*
* @author 只是甲
* @date 2021-07-05
* @remark 测试类 List集合的练习存储自定义对象并遍历
*
*/
public class list5 {
public static void main(String[] args) {
//创建集合对象
List list = new ArrayList();
//创建元素对象
list4 s1 = new list4("杜兰特",33);
list4 s2 = new list4("库里",33);
list4 s3 = new list4("詹姆斯",37);
//把元素添加到集合
list.add(s1);
list.add(s2);
list.add(s3);
// a.迭代器
System.out.println("=====a.迭代器=====");
Iterator it = list.iterator();
while (it.hasNext()) {
list4 s = it.next();
System.out.println(s.toString());
}
//b.普通for
System.out.println("=====b.普通for=====");
for (int i = 0; i < list.size(); i++) {
list4 s = list.get(i);
System.out.println(s.toString());
}
//c.增强for
System.out.println("=====c.增强for=====");
for (list4 s : list) {
System.out.println(s.toString());
}
}
}
测试记录:
=====a.迭代器=====
Student{name='杜兰特', age=33}
Student{name='库里', age=33}
Student{name='詹姆斯', age=37}
=====b.普通for=====
Student{name='杜兰特', age=33}
Student{name='库里', age=33}
Student{name='詹姆斯', age=37}
=====c.增强for=====
Student{name='杜兰特', age=33}
Student{name='库里', age=33}
Student{name='詹姆斯', age=37}
ListIterator:
特有功能:
注意:
代码:
package Java_study;
import java.util.ArrayList;
import java.util.List;
import java.util.ListIterator;
/**
*
* @author 只是甲
* @date 2021-07-05
* @remark 列表迭代器的特有功能ListIterator
*
*/
public class list6 {
public static void main(String[] args) {
//创建集合对象
List list = new ArrayList();
//添加元素
list.add("hello");
list.add("world");
list.add("java");
ListIterator lit = list.listIterator();
System.out.println("=====a.直接逆向遍历list数组=====");
while (lit.hasPrevious()) {
String s = lit.previous();
System.out.println(s);
}
System.out.println("=====b.先正向遍历list数组再逆向遍历list数组=====");
while (lit.hasNext()) {
String s = lit.next();
}
while (lit.hasPrevious()) {
String s = lit.previous();
System.out.println(s);
}
}
}
测试记录:
=====a.直接逆向遍历list数组=====
=====b.先正向遍历list数组再逆向遍历list数组=====
java
world
hello
需求:
ConcurrentModificationException:
产生的原因:
如何解决?
A : 迭代器遍历的时候,我可以通过迭代器修改集合中的元素
元素是跟在刚才迭代的元素后面的
B : 集合遍历的时候,我可以通过集合对象修改集合中的元素
元素是在最后添加的
代码:
package Java_study;
import java.util.*;
/**
*
* @author 并发修改异常产生的原因及解决方案只是甲
* @date 2021-07-05
* @remark
*
*/
public class list7 {
public static void main(String[] args) {
// 创建集合对象
List list = new ArrayList();
// 添加元素
list.add("hello");
list.add("world");
list.add("java");
System.out.println(list);
//我想判断里面有没有"world"这个元素,如果有,我就添加一个"javaee"元素
Iterator it = list.iterator();
//ConcurrentModificationException:添加失败,当方法检测到对象的并发修改,但不允许这种修改时,抛出此异常。
/*while (it.hasNext()) {
String s = it.next();
if (s.equals("world")) {
list.add("javaee");
}
System.out.println(s);
}*/
//A:迭代器遍历的时候,我可以通过迭代器修改集合中的元素(元素是跟在刚才迭代的元素后面的)
ListIterator lit = list.listIterator();
while (lit.hasNext()) {
String s = lit.next();
if (s.equals("world")) {
lit.add("javaee");
}
}
System.out.println(list);
//B:集合遍历的时候,我可以通过集合对象修改集合中的元素(元素是在最后添加的)
for (int x = 0; x < list.size(); x++) {
String s = list.get(x);
if (s.equals("world")) {
list.add("javaee");
}
}
System.out.println(list);
}
}
测试记录:
[hello, world, java]
[hello, world, javaee, java]
[hello, world, javaee, java, javaee]
什么是数据结构?
数据的组织方式;
栈 :FIFO(First In First Out) 先进先出
队列 :FILO(First In Last Out) 先进后出
数组 :存储同一种数据类型的多个元素的容器,有索引,方便我们获取元素;
应用 : ArrayList ;
链表 :由一个链子把多个节点连接起来的数据;
结点 :由 数据 和 下一个元素的地址 组成;
应用 : LinkedList ;
List
ArrayList : 底层数据结构是数组,查询快,增删慢
LinkedList : 底层数据结构是链表,查询慢,增删快
ArrayList存储字符串并遍历
A : 迭代器
B : 普通for
C : 增强for
代码:
package Java_study;
import java.util.ArrayList;
import java.util.Iterator;
/**
*
* @author 只是甲
* @date 2021-07-05
* @remark ArrayList集合存储字符串
*
*/
public class arraylist1 {
public static void main(String[] args) {
//创建集合对象
ArrayList arr = new ArrayList();
//添加元素
arr.add("hello");
arr.add("world");
arr.add("java");
//a.迭代器 arr.iterator
System.out.println("a.迭代器 arr.iterator");
Iterator it = arr.iterator();
while (it.hasNext()) {
System.out.println(it.next());
}
//b.普通for arr.fori
System.out.println("b.普通for arr.fori ");
for (int i = 0; i < arr.size(); i++) {
System.out.println(arr.get(i));
}
//c.增强for
for (String s: arr) {
System.out.println(s);
}
}
}
测试记录:
a.迭代器 arr.iterator
hello
world
java
b.普通for arr.fori
hello
world
java
hello
world
java
Collection接口 – List(接口,可重复,存取顺序一致) 、 Set(接口,唯一,存取顺序不一致)
List接口 – ArrayList(实现类,数据结构为数组)、LinkedList(实现类,数据结构为链表)
Set接口 – HashSet(实现类)
Set
元素唯一,存储元素无序 ;
一个不包含重复元素的 collection;
HashSet
它不保证 set 的迭代顺序,特别是它不保证该顺序恒久不变;
代码:
package Java_study;
import java.util.HashSet;
import java.util.Set;
/**
*
* @author 只是甲
* @date 2021-07-05
* @remark Set集合的特点
*
*/
public class set1 {
public static void main(String[] args) {
//创建集合对象
Set set = new HashSet<>();
//添加元素
set.add("hello");
set.add("world");
set.add("java");
//set集合的唯一性: 再次添加相同的元素,但集合该元素唯一
set.add("world");
//遍历集合 set.for
for (String s : set) {
System.out.println(s);
}
}
}
测试记录:
world
java
hello
HashSet它不保证 set 的迭代顺序,特别是它不保证该顺序恒久不变;
HashSet保证元素唯一性的原理?
通过查看add方法的源码,我们知道了添加功能的执行过程中,是进行了数据的判断的。
这个判断的流程是:
首先比较对象的哈希值是否相同,这个哈希值是根据对象的hashCode()计算出来的。
如果哈希值不同,就直接添加到集合中
如果哈希值相同,继续执行equals()进行比较,
返回的是true,说明元素重复,不添加。
返回的是false,说明元素不重复,就添加。
如果我们使用HashSet集合存储对象,你要想保证元素的唯一性,就必须重写hashCode()和equals()方法。
代码:
package Java_study;
import java.util.HashSet;
/**
*
* @author 只是甲
* @date 2021-07-05
* @remark HashSet保证元素唯一性的原理
*
*/
public class hashset1 {
public static void main(String[] args) {
//创建集合对象
HashSet hs = new HashSet<>();
//添加元素
hs.add("hello");
hs.add("world");
hs.add("java");
//set集合的唯一性: 再次添加相同元素,但集合内该元素只有一个
hs.add("world");
//遍历集合
for (String h : hs) {
System.out.println(h);
}
}
}
测试记录:
world
java
hello
需求
自定义一个学生类,给出成员变量name和age。
遍历集合的时候,在控制台输出学生对象的成员变量值。
遍历方式
a.迭代器
b.增强for
代码:
hashset2
package Java_study;
/**
*
* @author 只是甲
* @date 2021-07-05
* @remark 实体学生类 HashSet集合存储实体类
*
*/
public class hashset2 {
private String name;
private int age;
public hashset2(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 "{name='" + name + "'" + ", age=" + age + '}';
}
}
hashset3
package Java_study;
import java.util.HashSet;
import java.util.Iterator;
/**
*
* @author 只是甲
* @date 2021-07-05
* @remark 测试类 HashSet集合存储实体类
*
*/
public class hashset3 {
public static void main(String[] args) {
//创建集合对象
HashSet hs = new HashSet<>();
//创建元素对象
hashset2 s1 = new hashset2("杜兰特",33);
hashset2 s2 = new hashset2("库里",33);
hashset2 s3 = new hashset2("詹姆斯",37);
//把元素添加到集合
hs.add(s1);
hs.add(s2);
hs.add(s3);
//a.迭代器遍历 hs.iterator()
Iterator it = hs.iterator();
System.out.println("a.迭代器遍历 hs.iterator()");
while (it.hasNext()) {
hashset2 s = it.next();
System.out.println(s.toString());
}
//b.增强for遍历 it.next()
System.out.println("b.增强for遍历 it.next()");
for (hashset2 s:hs) {
System.out.println(s);
}
}
}
测试记录:
a.迭代器遍历 hs.iterator()
{name='詹姆斯', age=37}
{name='杜兰特', age=33}
{name='库里', age=33}
b.增强for遍历 it.next()
{name='詹姆斯', age=37}
{name='杜兰特', age=33}
{name='库里', age=33}
要求
如果对象的成员变量值相同,就认为是同一个元素。
提示
自定义一个学生类,给出成员变量name和age。遍历集合的时候,在控制台输出学生对象的成员变量值。
遍历方式
a.迭代器
b.增强for
因为我们存储的元素所属的类没有重写hashCode()和equals()方法,所以保证不了元素的唯一性。
而我想保证,怎么办呢?重写这两个方法就可以了。
如何重写呢?自动生成就可以了。
代码:
hashset4
package Java_study;
import java.util.Objects;
/**
*
* @author 只是甲
* @date 2021-07-06
* @remark 实体学生类 HashSet重写hashCode();equals();
*
*/
public class hashset4 {
private String name;
private int age;
public hashset4(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 "{name='" + name + "'" + ", age=" + age + '}';
}
/**
* @remark 因为我们存储的元素所属的类没有重写hashCode()和equals()方法,所以保证不了元素的唯一性。
* 而我想保证,怎么办呢?重写这两个方法就可以了。
* 如何重写呢?自动生成就可以了。alt+insert
* @param o
* @return
*/
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
hashset4 that = (hashset4) o;
return age == that.age && name.equals(that.name);
}
/**
* @remark 因为我们存储的元素所属的类没有重写hashCode()和equals()方法,所以保证不了元素的唯一性。
* 而我想保证,怎么办呢?重写这两个方法就可以了。
* 如何重写呢?自动生成就可以了。alt+insert
* @return
*/
@Override
public int hashCode() {
return Objects.hash(name, age);
}
}
hashset5
package Java_study;
import java.util.HashSet;
/**
*
* @author 只是甲
* @date 2021-07-06
* @remark 测试类 HashSet重写hashCode();equals();
*
*/
public class hashset5 {
public static void main(String[] args) {
HashSet hs = new HashSet<>();
hashset4 s1 = new hashset4("杜兰特",33);
hashset4 s2 = new hashset4("库里",33);
hashset4 s3 = new hashset4("詹姆斯",38);
hashset4 s4 = new hashset4("东契奇",22);
//把元素添加到集合
hs.add(s1);
hs.add(s2);
hs.add(s3);
hs.add(s4);
//增强for遍历集合 hs.for
for (hashset4 h:hs) {
System.out.println(h);
}
}
}
测试记录:
{name='杜兰特', age=33}
{name='詹姆斯', age=38}
{name='库里', age=33}
{name='东契奇', age=22}