【4】Java集合

一、集合概述

1. Java中的集合是工具类,可以存储任意数量的具有共同属性的对象。
   与数组不同,集合的长度可以动态改变,所以数组适合去存储固定长度的数据,集合适合去存储不固定长度的数据。
 
 
2. 集合的应用场景:
1)无法预测存储数据的数量。
2)存储具有一对一关系的数据(比如某件商品,其商品编号只对应这件商品)。
3)需要进行数据的增删。
4)数据重复问题。

二、集合框架的体系结构

1、collection:存储类的对象

Collection三个子接口:
1)List(序列)
  存放的数据有序、可以重复
  List常用实现类ArrayList(长度动态增长的数组)
 
2)Queue(队列)
  存放的数据有序、可以重复
  Queue实现类LinkedList(同时实现了List接口表示的是链表的内容)
 
3)Set(集)
  无序、不可以重复
  Set实现类HashSet(哈希集)

2、Map

键值对 Map实现类HashMap(哈希表,存储为键值对<key,value>)

 

 

三、List

1、List概述

1. List中元素有序且可以重复,称为序列。
2. List可以精确地控制每个元素的插入位置,或删除某个位置的元素。
3. List的两个主要实现类:ArrayList和LinkedList。
4. ArrayList底层是由数组实现的,长度动态增长。ArrayList中的元素可以为null且可以有多个。
在列表尾部插入或删除数据非常有效,但是在中间则需要耗费较多资源。所以更适合查找和更新元素。

2、list的常用方法

List list = new ArrayList();
list.contains();用来判断列表中是否包含要查找的对象,括号中写对象名
list.add();添加元素
list.remove();移除元素/移除元素有两种方法1.根据下标移除2.根据要移除的元素内容移除
list.get();获取列表中指定位置处的元素
list.size();获取列表长度
 
举例:
import java.util.ArrayList;
import java.util.List;

public class ListDemo1 {

    public static void main(String[] args) {
        // 用ArrayList存储编程语言的名称,并输出
        List list=new ArrayList();
        list.add("Java");
        list.add("C");
        list.add("C++");
        list.add("Go");
        list.add("swift");
        //输出列表中元素的个数
        System.out.println("列表中元素的个数为:"+list.size());
        
        //遍历输出所有编程语言
        System.out.println("**************************************");
        for(int i=0;i<list.size();i++){
            System.out.print(list.get(i)+",");
        }

        //移除列表中的C++
        System.out.println();
        //list.remove(2);
        list.remove("C++");
        System.out.println("**************************************");
        System.out.println("移除C++以后的列表元素为:");
        for(int i=0;i<list.size();i++){
            System.out.print(list.get(i)+",");
        }
    }

}
List

 3、List代码实例:添加公告

1、 需求
--公告的添加和显示
--在指定位置处插入公告
--删除公告 --修改公告
 
2、分析
1)公告类属性: --编号 id --标题 title -- 创建人 creator --创建时间 createTime
2)公告类方法 --构造方法 --获取和设置属性值的方法

import java.util.Date;

public class Notice {
    private int id;//ID
    private String title;//标题
    private String creator;//创建人
    private Date createTime;//创建时间
    public Notice(int id, String title, String creator, Date createTime) {
        super();
        this.id = id;
        this.title = title;
        this.creator = creator;
        this.createTime = createTime;
    }
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getTitle() {
        return title;
    }
    public void setTitle(String title) {
        this.title = title;
    }
    public String getCreator() {
        return creator;
    }
    public void setCreator(String creator) {
        this.creator = creator;
    }
    public Date getCreateTime() {
        return createTime;
    }
    public void setCreateTime(Date createTime) {
        this.createTime = createTime;
    }
    
    
}
Notice

 

import java.util.ArrayList;
import java.util.Date;

public class NoticeTest {

    public static void main(String[] args) {
        // 创建Notice类的对象,生成三条公告
        Notice notice1 = new Notice(1, "欢迎来到学校!", "管理员", new Date());
        Notice notice2 = new Notice(2, "请同学们按时提交作业!", "老师", new Date());
        Notice notice3 = new Notice(3, "考勤通知!", "老师", new Date());

        // 添加公告
        ArrayList noticeList = new ArrayList();
        noticeList.add(notice1);
        noticeList.add(notice2);
        noticeList.add(notice3);

        // 显示公告
        System.out.println("公告的内容为:");
        for (int i = 0; i < noticeList.size(); i++) {
            System.out.println(i + 1 + ":" + ((Notice) (noticeList.get(i))).getTitle());//这里需要强制转换,加上Notice
        }

        System.out.println("**************************************");
        // 在第一条公告后面添加一条新公告
        Notice notice4 = new Notice(4, "在线编辑器可以使用啦!", "管理员", new Date());
        noticeList.add(1, notice4);

        // 显示公告
        System.out.println("公告的内容为:");
        for (int i = 0; i < noticeList.size(); i++) {
            System.out.println(i + 1 + ":" + ((Notice) (noticeList.get(i))).getTitle());
        }

        System.out.println("**************************************");
        // 删除按时提交作业的公告
        noticeList.remove(2);
        // 显示公告
        System.out.println("删除公告后的内容为:");
        for (int i = 0; i < noticeList.size(); i++) {
            System.out.println(i + 1 + ":" + ((Notice) (noticeList.get(i))).getTitle());
        }
        
        //将第二条公告改为:Java在线编辑器可以使用啦!
        System.out.println("**************************************");
        //修改第二条公告中title的值
        notice4.setTitle("Java在线编辑器可以使用啦!");
        noticeList.set(1, notice4);
        System.out.println("修改后公告的内容为:");
        for (int i = 0; i < noticeList.size(); i++) {
            System.out.println(i + 1 + ":" + ((Notice) (noticeList.get(i))).getTitle());
        }
    }

}
NoticeTest

 

 

四、set

1、set概述

Set是元素无序并且不可以重复的集合,被称为集,Set是接口。
  HashSet是Set的一个重要实现类,称为哈希集
  HashSet中的元素无序并且不可以重复
  HashSet中只允许一个null元素 具有良好的存取和查找性能
  HashSet的底层其实是HashMap。 HashSet默认容量是16,默认的加载因子为0.75
 

2、补充知识:迭代器 ITerator

迭代器:Iterator接口可以以统一的方式对各种集合元素进行遍历 hasNext():检测集合中是否还有下一个元素 next():返回集合中的下一个元素
迭代器使用完会释放空间,再次使用需要给到新的引用 把集合添加到迭代器中:(导入包)Iterator 迭代器名=集合名.iterator()方法 遍历迭代器并输出:
 
代码实例:
import java.util.HashSet;
import java.util.Iterator;
import java.util.Set;

public class WordDemo {

    public static void main(String[] args) {
        // 将英文单词添加到HashSet中
        Set set = new HashSet();
        // 向集合中添加元素
        set.add("blue");
        set.add("red");
        set.add("black");
        set.add("yellow");
        set.add("white");
        // 显示集合的内容
        System.out.println("集合中的元素为:");
        Iterator it = set.iterator();
        // 遍历迭代器并输出元素
        while (it.hasNext()) {
            System.out.print(it.next() + "   ");
        }
        System.out.println();
        // 在集合中插入一个新的单词
        // set.add("green");
        set.add("white");
        it = set.iterator();
        // 遍历迭代器并输出元素
        System.out.println("**************************");
        System.out.println("插入重复元素后的输出结果为:");
        while (it.hasNext()) {
            System.out.print(it.next() + "   ");
        }
        //插入失败,但是不会报错
    }

}
set代码实例

 

3、HashSet实例:宠物猫信息管理

 
 1)需求与提示
*需求
--添加和显示宠物猫信息
--查找某只宠物猫的信息并输出
--修改宠物猫的信息
--删除宠物猫信息

宠物猫类的定义:
*属性
--名字 name
--年龄 month
--品种 species
*类中方法的定义具体如截图所示

方法:
构造方法
获取和设置属性值的方法
其他方法:tostring方法
View Code

 

2)代码
 
public class Cat {
    private String name; //名字
    private int month; //年龄
    private String species;//品种
    
    //构造方法
    public Cat(String name, int month, String species) {
        super();
        this.name = name;
        this.month = month;
        this.species = species;
    }
    //getter与setter方法
    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getMonth() {
        return month;
    }

    public void setMonth(int month) {
        this.month = month;
    }

    public String getSpecies() {
        return species;
    }

    public void setSpecies(String species) {
        this.species = species;
    }


    /**
     * toString()方法是Object类的方法,我们定义的所有类都是Object类的子类;
     如果输出该类的对象,就会自动调用toString()方法,这样会输出该对象的地址;
     如果想输出对象的内容,可以在该类中重写toString()方法
     *改写toString方法
     * 1\重写Object类的toString()方法,输出对象时,就可以自动调用toString()方法。
     * 这个方法只是为了方便输出,比如System.out.println(xx),括号里面的“xx”如果不是String类型的话,就自动调用xx的toString()方法。
     我们在Cat中重写toString方法,输出Cat对象的各个属性值。
     总而言之,它只是sun公司开发java的时候为了方便所有类的字符串操作而特意加入的一个方法。
     * @return
     */
    @Override
    public String toString() {
        return "[姓名:" + name + ", 年龄:" + month + ", 品种:" + species + "]";
    }
    @Override
    public int hashCode() {
        final int prime = 31;
        int result = 1;
        result = prime * result + month;
        result = prime * result + ((name == null) ? 0 : name.hashCode());
        result = prime * result + ((species == null) ? 0 : species.hashCode());
        return result;
    }
    @Override
    public boolean equals(Object obj) {
        //判断对象是否相等,相等则返回true,不用继续比较属性了
        if(this==obj)
            return true;
        //判断obj是否是Cat类的对象
        if(obj.getClass()==Cat.class){
            Cat cat=(Cat)obj;
            return cat.getName().equals(name)&&(cat.getMonth()==month)&&(cat.getSpecies().equals(species));
        }
        
        return false;
    }
    
    
}
Cat

 

import java.util.HashSet;
import java.util.Iterator;
import java.util.Set;

public class CatTest {

    public static void main(String[] args) {
        // 定义宠物猫对象
        Cat huahua = new Cat("花花", 12, "英国短毛猫");
        Cat fanfan = new Cat("凡凡", 3, "中华田园猫");
        // 将宠物猫对象放入HashSet中
        Set<Cat> set = new HashSet<Cat>();
        set.add(huahua);
        set.add(fanfan);
        // 显示宠物猫信息
        Iterator<Cat> it = set.iterator();//使用迭代器
        while (it.hasNext()) {
            System.out.println(it.next());
        }

        // 再添加一个与花花属性一样的猫
        Cat huahua01 = new Cat("花花", 12, "英国短毛猫");
        set.add(huahua01);
        System.out.println("**********************************");
        System.out.println("添加重复数据后的宠物猫信息:");
        it = set.iterator();
        while (it.hasNext()) {
            System.out.println(it.next());
        }

        System.out.println("**********************************");
        // 重新插入一个新宠物猫
        Cat huahua02 = new Cat("花花二代", 2, "英国短毛猫");
        set.add(huahua02);
        System.out.println("添加花花二代后的宠物猫信息:");
        it = set.iterator();
        while (it.hasNext()) {
            System.out.println(it.next());
        }

        System.out.println("**********************************");
        // 在集合中查找花花的信息并输出
        if (set.contains(huahua)) {
            System.out.println("花花找到了!");
            System.out.println(huahua);
        } else {
            System.out.println("花花没找到!");
        }
        // 在集合中使用名字查找花花的信息
        System.out.println("**********************************");
        System.out.println("通过名字查找花花信息");
        boolean flag = false;
        Cat c = null;
        it = set.iterator();
        while (it.hasNext()) {
            c = it.next();
            if (c.getName().equals("花花")) {
                flag = true;// 找到了
                break;
            }
        }
        if (flag) {
            System.out.println("花花找到了");
            System.out.println(c);
        } else {
            System.out.println("花花没找到");
        }

        // 删除花花二代的信息并重新输出
        for (Cat cat : set) {
            if ("花花二代".equals(cat.getName())) {
                set.remove(cat);
break;            }
        }
        System.out.println("**********************************");
        
        System.out.println("删除花花二代后的数据");
        for(Cat cat:set){
            System.out.println(cat);
        }
        //删除集合中的所有宠物猫信息
        System.out.println("**********************************");
        boolean flag1=set.removeAll(set);
        if(set.isEmpty()){
            System.out.println("猫都不见了。。。");
        }else{
            System.out.println("猫还在。。。");

        }
    }
}
Cattest

 

4、toString方法+重写equals和hashcode方法

 

1)为什么要在Cat类中重写toString方法呢?

这个方法只是为了方便输出,比如System.out.println(xx),括号里面的“xx”如果不是String类型的话,就自动调用xx的toString()方法。
我们在Cat中重写toString方法,输出Cat对象的各个属性值。
总而言之,它只是sun公司开发java的时候为了方便所有类的字符串操作而特意加入的一个方法。
 

2)HashSet中,是无法添加相同的对象的。那么什么是相同的对象?

问:用相同的属性对对象进行初始化,它们是相同的对象吗?
如:
Cat cat1 = new Cat("花花", 12, "英国短毛猫");
Cat cat2 = new Cat("花花", 12, "英国短毛猫");
 
答:不是。

所以这样的两个对象,是可以加入的HashSet中的。这往往与我们的期望是不符的,我们认为他们应该是"相同的",不应该重复加入。

程序是通过调用hashCode和equals两个方法来判断对象是否相等的。

所以,对于上面提到的问题,我们可以通过重写这两个方法来实现我们需求。

equals方法思路:

1)先判断对象是否相等,相等返回true,不用继续比较属性;

2)判断obj是否为目标类的对象,若是则通过强制类型转换为目标类对象后继续比较属性值是否相同;

3)以上都不满足则返回false:

    @Override
    public boolean equals(Object obj) {
        //判断对象是否相等,相等则返回true,不用继续比较属性了
        if(this==obj)
            return true;
        //判断obj是否是Cat类的对象
        if(obj.getClass()==Cat.class){
            Cat cat=(Cat)obj;
            return cat.getName().equals(name)&&(cat.getMonth()==month)&&(cat.getSpecies().equals(species));
        }
        
        return false;
    }

 

 

3)自动生成toString、hashCode和equals

右键Generate
 

 

一般情况下hashcode直接用equals需要手动修改下

4)其他

hashCode方法是对类中的数据再分类,例如:分成了3份, 不重写也没关系。

如果hashCode()值相同,还需要根据equals()方法判断是否为同一个对象。

hashCode的值不同时,则两个对象必定不同

 

五、Map

1、Map概述

1)Map中的数据是以键值对(key-value)的形式存储的;
2)key-value以Entry类型的对象实例存在;Entry是一个接口
3)可以通过key值快速地查找value;
4)一个映射不能包含重复的键;
5)每个键最对只能映射到一个值。 
 

2. HashMap

1)基于哈希表的Map接口的实现;
2)允许使用null值和null键;
3)key值不允许重复;
4)HashMap中的Entry对象是无序排列的。

 

3、 HashMap打印输出

 
Map 类型,Map 是一个接口,无法进行实例化操作,可以定义一个引用指向HashMap()的方法构造一个Map类,
向Map中添加元素的时,使用 对象名.put ( key,value) 的方法添加一个键值对。
 
打印输出有两种方式:
1)打印输出value的值(直接使用迭代器)
2)打印输输出key和value的值:通过entrySet方法

 

 

通过keySet()方法将map中所有key值的取出,返回值是个只存放key值的Set集合
通过entrySet()方法将map集合中的映射关系取出,返回映射所包含的映射关系的Set集合
虽然使用keyset及entryset来进行遍历能取得相同的结果,但两者的遍历速度是有差别的。
keySet():迭代后只能通过get()取key;再根据key值取value。
entrySet():迭代后可以e.getKey(),e.getValue()取key和value。
 

4、HashMap代码实例

import java.util.*;
import java.util.Map.Entry;

public class DictionaryDemo {

    public static void main(String[] args) {
        Map<String,String> animal=new HashMap<String,String>();
        System.out.println("请输入三组单词对应的注释,并存放到HashMap中");
        Scanner console=new Scanner(System.in);
        //添加数据
        int i=0;
        while(i<3){
            System.out.println("请输入key值(单词):");
            String key=console.next();
            System.out.println("请输入value值(注释):");
            String value=console.next();
            animal.put(key, value);
            i++;
        }
        //打印输出value的值(直接使用迭代器)
        System.out.println("*****************************************");
        System.out.println("使用迭代器输出所有的value:");
        Iterator<String> it=animal.values().iterator();
        while(it.hasNext()){
            System.out.print(it.next()+"    ");
        }
        System.out.println();
        System.out.println("*****************************************");
        //打印输出key和value的值
        //通过entrySet方法
        System.out.println("通过entrySet方法得到key-value:");
        Set<Entry<String, String>> entrySet=animal.entrySet();
        for(Entry<String, String> entry:entrySet){
            System.out.print(entry.getKey()+"-");;
            System.out.println(entry.getValue());
        }
        System.out.println();
        System.out.println("*****************************************");
        
        //通过单词找到注释并输出
        //使用keySet方法
        System.out.println("请输入要查找的单词:");
        String strSearch=console.next();
        //1.取得keySet
        Set<String> keySet=animal.keySet();
        //2.遍历keySet
        for(String key:keySet){
            if(strSearch.equals(key)){
                System.out.println("找到了!"+"键值对为:"+key+"-"+animal.get(key));
                break;
            }
        }
    }

}
DictionaryDemo

 

 

posted @ 2020-06-15 21:10  柠檬不萌!  阅读(257)  评论(0编辑  收藏  举报
浏览器标题切换
浏览器标题切换end