美文网首页
java学习笔记6

java学习笔记6

作者: 海洋_5ad4 | 来源:发表于2018-11-06 18:06 被阅读0次

    迭代器的原理及源码解析

    • A:迭代器原理
      • 迭代器原理:迭代器是对集合进行遍历,而每一个集合内部的存储结构都是不同的,所以每一个集合存和取都是不一样,那么就需要在每一个类中定义hasNext()和next()方法,这样做是可以的,但是会让整个集合体系过于臃肿,迭代器是将这样的方法向上抽取出接口,然后在每个类的内部,定义自己迭代方式,这样做的好处有二,第一规定了整个集合体系的遍历方式都是hasNext()和next()方法,第二,代码有底层内部实现,使用者不用管怎么实现的,会用即可
    • B:迭代器源码解析
      • 1,在eclipse中ctrl + shift + t找到ArrayList类
      • 2,ctrl+o查找iterator()方法
      • 3,查看返回值类型是new Itr(),说明Itr这个类实现Iterator接口
      • 4,查找Itr这个内部类,发现重写了Iterator中的所有抽象方法

    List集合的特有功能概述和测试

    • A:List集合的特有功能概述
      • void add(int index,E element)
      • E remove(int index)
      • E get(int index)
      • E set(int index,E element)
    package com.heima.list;
    
    import java.util.ArrayList;
    import java.util.List;
    
    public class Demo1_List {
        public static void main(String[] args) {
            List list = new ArrayList();
            list.add("a");
            list.add("b");
            list.add("c");
            list.add("d");
            list.set(1,"z");
            System.out.println(list);
        }
    
        public static void demo4() {
            List list = new ArrayList();
            list.add("a");
            list.add("b");
            list.add("c");
            list.add("d");
            //通过索引遍历List集合
            for(int i = 0;i < list.size(); i++) {
                System.out.println(list.get(i));
            }
        }
    
        public static void demo3() {
            List list = new ArrayList();
            list.add(111);
            list.add(222);
            list.add(333);
            
            list.remove(111);           //删除的时候不会自动装箱,把111当作索引
            System.out.println(list);
        }
    
        public static void demo2() {
            List list = new ArrayList();
            list.add("a");
            list.add("b");
            list.add("c");
            list.add("d");
            
            Object obj = list.remove(1);        //通过索引删除元素,将被删除的元素返回
            System.out.println(obj);
            System.out.println(list);
        }
    
        public static void demo1() {
            List list = new ArrayList();
            list.add("a");
            list.add("b");
            list.add("c");
            list.add("d");
            list.add(4,"e");    //index<=size并且>=0都不会报异常
            //list.add(10,"z"); //java.lang.IndexOutOfBoundsException,
            System.out.println(list);//当存储时使用不存在的索引时,就会出现索引越界异常
        }
    
    }
    

    List集合存储学生对象并遍历

    package com.heima.list;
    
    import java.util.ArrayList;
    import java.util.List;
    
    import com.heima.bean.Student;
    
    public class Demo2_List {
    
        public static void main(String[] args) {
            List list = new ArrayList();
            list.add(new Student("张三",23));
            list.add(new Student("李四",24));
            list.add(new Student("王五",25));
            list.add(new Student("赵六",26));
            
            for (int i = 0; i < list.size(); i++) {
                //System.out.println(list.get(i));
                Student s = (Student)list.get(i);
                System.out.println(s.getName());
            }
        }
    
    }
    

    并发修改异常产生的原因及解决方案

    • A:案例演示
      • 需求:我有一个集合,请问,我想判断里面有没有"world"这个元素,如果有,我就添加一个"javaee"元素,请写代码实现。
    • B:ConcurrentModificationException出现
      • 迭代器遍历,集合修改集合
    • C:解决方案
      • a:迭代器迭代元素,迭代器修改元素(ListIterator的特有功能add)
      • b:集合遍历元素,集合修改元素

    ListIterator

    • boolean hasNext()是否有下一个
    • boolean hasPrevious()是否有前一个
    • Object next()返回下一个元素
    • Object previous();返回上一个元素
    package com.heima.list;
    
    import java.util.ArrayList;
    import java.util.List;
    import java.util.ListIterator;
    
    public class Demo4_ListIterator {
    
        /**
         * @param args
         */
        public static void main(String[] args) {
            List list = new ArrayList();
            list.add("a");                                  //Object obj = new String();
            list.add("b");
            list.add("world");
            list.add("c");
            list.add("d");
            list.add("e");
            
            ListIterator lit = list.listIterator();         //获取迭代器
            while(lit.hasNext()) {
                System.out.println(lit.next());             //获取元素并将指针向后移动
            }
            
            System.out.println("-----------------");
            
            while(lit.hasPrevious()) {
                System.out.println(lit.previous());         //获取元素并将指针向前移动
            }
        }
    
    }
    

    Vector的特有功能

    • A:Vector类概述
    • B:Vector类特有功能
      • public void addElement(E obj)
      • public E elementAt(int index)
      • public Enumeration elements()
    package com.heima.list;
    
    import java.util.Enumeration;
    import java.util.Vector;
    
    public class Demo5_Vector {
    
        public static void main(String[] args) {
            Vector v = new Vector();
            v.addElement("a");
            v.addElement("b");
            v.addElement("c");
            v.addElement("d");
            
            Enumeration en = v.elements();              //获取枚举
            while(en.hasMoreElements()) {               //判断集合中是否有元素
                System.out.println(en.nextElement());   //获取集合中的元素
            }
        }
    
    }
    

    数据结构之数组和链表

    • A:数组
      • 查询快修改也快
      • 增删慢
    • B:链表
      • 查询慢,修改也慢
      • 增删快


        数组.jpg
        链表.jpg

    List的三个子类的特点

    • A:List的三个子类的特点

        ArrayList:
            底层数据结构是数组,查询快,增删慢。
            线程不安全,效率高。
      
        Vector:
            底层数据结构是数组,查询快,增删慢。
            线程安全,效率低。
        Vector相对ArrayList查询慢(线程安全的)
        Vector相对LinkedList增删慢(数组结构)
      
        LinkedList:
            底层数据结构是链表,查询慢,增删快。
            线程不安全,效率高。
      
        Vector和ArrayList的区别
            Vector是线程安全的,效率低
            ArrayList是线程不安全的,效率高
        共同点:都是数组实现的
      
        ArrayList和LinkedList的区别
            ArrayList底层是数组结果,查询和修改快
            LinkedList底层是链表结构的,增和删比较快,查询和修改比较慢
        共同点:都是线程不安全的
      
    • B:List有三个儿子,我们到底使用谁呢?
      查询多用ArrayList
      增删多用LinkedList
      如果都多ArrayList

    去除ArrayList中重复字符串元素方式

    package com.heima.list;
    
    import java.util.ArrayList;
    import java.util.Iterator;
    
    public class Demo1_Arraylist {
    
        public static void main(String[] args) {
            ArrayList list = new ArrayList();
            list.add("a");
            list.add("a");
            list.add("b");
            list.add("b");
            list.add("c");
            list.add("c");
            list.add("c");
            ArrayList newList = getSingle(list);
            System.out.println(newList);
        }
        
        public static ArrayList getSingle(ArrayList list) {
            ArrayList newList = new ArrayList();                //1,创建新集合
            Iterator it = list.iterator();          //2,根据传入的集合(老集合)获取迭代器
            while(it.hasNext()) {                   //3,遍历老集合
                Object obj = it.next();             //记录住每一个元素
                if(!newList.contains(obj)) {    //如果新集合中不包含老集合的元素,则将该元素添加
                    newList.add(obj);
                }
            }
            return newList;
        }
    
    }
    

    去除ArrayList中重复自定义对象元素

    package com.heima.list;
    
    import java.util.ArrayList;
    import java.util.Iterator;
    
    import com.heima.bean.Person;
    
    public class Demo2_ArrayList {
    
        /**
         * @param args
         */
        public static void main(String[] args) {
            ArrayList list = new ArrayList();               //创建集合对象
            list.add(new Person("张三", 23));
            list.add(new Person("张三", 23));
            list.add(new Person("李四", 24));
            list.add(new Person("王麻子", 25));
            
            //ArrayList newList = getSingle(list);
            //System.out.println(newList);
            list.remove(new Person("张三", 23));  //一次只能删除一个new Person("张三", 23)
            System.out.println(list);
        }
        
        public static ArrayList getSingle(ArrayList list) {
            ArrayList newList = new ArrayList();                //1,创建新集合
            Iterator it = list.iterator();          //2,根据传入的集合(老集合)获取迭代器
            while(it.hasNext()) {                   //3,遍历老集合
                Object obj = it.next();             //记录住每一个元素
                if(!newList.contains(obj)) {    //如果新集合中不包含老集合的元素,则将该元素添加
                    newList.add(obj);           //contains和remove底层依赖的是equals方法
                }
            }
            return newList;
        }
    }
    

    LinkedList的特有功能

    A:LinkedList类概述

    • B:LinkedList类特有功能
      • public void addFirst(E e)及addLast(E e)
      • public E getFirst()及getLast()
      • public E removeFirst()及public E removeLast()
      • public E get(int index);
    package com.heima.list;
    
    import java.util.LinkedList;
    
    public class Demo3_LinkedList {
    
        public static void main(String[] args) {
            LinkedList list = new LinkedList();
            list.addFirst("a");
            list.addFirst("b");
            list.addFirst("c");
            list.addLast("d");
            
            //System.out.println(list.getFirst());  //c 
            //System.out.println(list.getLast());       //d
            
            //System.out.println(list.removeFirst());   //c
            //System.out.println(list.removeLast());    //d
            
            System.out.println(list.get(1));            //b 
            System.out.println(list);
        }
    
    }
    

    栈和队列数据结构

      • 先进后出
    • 队列
      • 先进先出

    用LinkedList模拟栈数据结构的集合并测试

    package com.heima.list;
    
    import java.util.LinkedList;
    
    public class Demo4_LinkedList {
    
        public static void main(String[] args) {
            Stack s = new Stack();
            s.in("a");  //进栈
            s.in("b");
            s.in("c");
            s.in("d");
            
            while(!s.isEmpty()) {   //判断栈是否为空
                System.out.println(s.out());    //弹栈
            }
        }
    
        public static void demo1() {
            LinkedList list = new LinkedList();         //创建集合对象
            list.addLast("a");
            list.addLast("b");
            list.addLast("c");
            list.addLast("d");
            while(!list.isEmpty()) {
                System.out.println(list.removeLast());
            }
        }
    
    }
    
    package com.heima.list;
    
    import java.util.LinkedList;
    
    public class Stack {
        private LinkedList list = new LinkedList();
        //模拟进栈方法
        public void in(Object obj) {
            list.addLast(obj);
        }
        //模拟出栈
        public Object out() {
            return list.removeLast();
        }
        //模拟栈结构是否为空
        public boolean isEmpty() {
            return list.isEmpty();
        }
    }
    

    泛型概述和基本使用

    • A:泛型概述
    • B:泛型好处
      • 提高安全性(将运行期的错误转换到编译期)
      • 省去强转的麻烦
    • C:泛型基本使用
      • <>中放的必须是引用数据类型
    • D:泛型使用注意事项
      • 前后的泛型必须一致,或者后面的泛型可以省略不写(1.7的新特性菱形泛型)
    package com.heima.generic;
    
    import java.util.ArrayList;
    import java.util.Iterator;
    
    import com.heima.bean.Person;
    
    public class Demo1_Generic {
    
        /**
         * @param args
         */
        public static void main(String[] args) {
            //int[] arr = new byte[5];  //数组要保证前后的数据类型一致
            //ArrayList<Object> list = new ArrayList<Person>(); //集合的泛型要保证前后的数据类型一致 
            //ArrayList<Object> list = new ArrayList<>();   //1.7版本的新特性,菱形泛型
            ArrayList<Object> list = new ArrayList<>();
            //泛型最好不要定位成Object,因为任何类都是Object类的子类,所以和不加<>是一样的效果
            list.add("aaa");
            list.add(true);
        }
    
        public static void demo1() {
            ArrayList<Person> list = new ArrayList<Person>();
            //list.add(110);
            //list.add(true);
            list.add(new Person("张三",23));
            list.add(new Person("李四",24));
            
            Iterator<Person> it = list.iterator();
            while(it.hasNext()) {
                System.out.println(it.next().getName());//next方法会将指针向后移动
                //调用next方法多次,会将指针向后移动多次
            }
        }
    
    }
    

    ArrayList存储字符串和自定义对象并遍历泛型版

    package com.heima.generic;
    
    import java.util.ArrayList;
    import java.util.Iterator;
    
    import com.heima.bean.Person;
    
    public class Demo2_Generic {
    
        public static void main(String[] args) {
            ArrayList<Person> list = new ArrayList<>();
            list.add(new Person("张三",23));
            list.add(new Person("李四",24));
            list.add(new Person("王五",25));
            list.add(new Person("赵六",26));
            
            Iterator<Person> it = list.iterator();
            while(it.hasNext()) {
                Person p = it.next();       //将集合中的每一个元素用Person记录
                System.out.println(p);
            }
        }
    
        public static void demo1() {
            ArrayList<String> list = new ArrayList<>(); //创建集合对象
            list.add("a");list.add("b");list.add("c");list.add("d");
            
            Iterator<String> it = list.iterator();
            while(it.hasNext()) {
                System.out.println(it.next());
            }
        }
    
    }
    

    泛型的由来

    • A:案例演示
      • 泛型的由来:通过Object转型问题引入
      • 早期的Object类型可以接收任意的对象类型,但是在实际的使用中,会有类型转换的问题。也就存在这隐患,所以Java提供了泛型来解决这个安全问题。

    泛型类的概述及使用

    A:泛型类概述<T>
    * 把泛型定义在类上

    • B:定义格式
      • public class 类名<泛型类型1,…>
    • C:注意事项
      • 泛型类型必须是引用类型
    • D:案例演示
      • 泛型类的使用

    泛型方法的概述和使用

    • A:泛型方法概述
      • 把泛型定义在方法上
    • B:定义格式
      • public <泛型类型> 返回类型 方法名(泛型类型 变量名)
    • C:案例演示
      • 泛型方法的使用
    package com.heima.bean;
    
    public class Tool<Q> {
        private Q q;
        
        /*public void show(Q q) {   //方法泛型最好与类的泛型一致
            System.out.println(q);
        }*/
        
        public<T> void show(T t) {  //如果不一致,需要在方法上声明该泛型
            System.out.println(t);
        }
        public static<W> void print(W w) {  //静态方法必须声明自己的泛型
            System.out.println(w);  //因为类的泛型是类实例化的时候(生成对象的时候)传入的,静态方法是随着类的加载而加载,矛盾。
        }
    }
    

    泛型接口的概述和使用

    • A:泛型接口概述
      • 把泛型定义在接口上
    • B:定义格式
      • public interface 接口名<泛型类型>
    • C:案例演示
      • 泛型接口的使用
    package com.heima.generic;
    
    public class Demo4_Generic {
    
        public static void main(String[] args) {
    
        }
    
    }
    
    interface Inter<T> {
        public void show(T t);
    }
    
    /*class Demo implements Inter<String> {     //推荐用这种
    
        @Override
        public void show(String t) {
            System.out.println(t);
        }
        
    }*/         //第一种方式
    
    class Demo<T> implements Inter<T> {         //没有必须在实现接口的时候给自己类加泛型
    
        @Override
        public void show(T t) {
            System.out.println(t);
        }
        
    }
    

    泛型高级之通配符

    • A:泛型通配符<?>
      • 任意类型,如果没有明确,那么就是Object以及任意的Java类了
    • B:? extends E
      • 向下限定,E及其子类
    • C:? super E
      • 向上限定,E及其父类
    package com.heima.generic;
    
    import java.util.ArrayList;
    
    import com.heima.bean.Person;
    
    public class Demo5_Generic {
    
        public static void main(String[] args) {
            //List<?> list = new ArrayList<Integer>();//当右边的泛型时不确定时,左边也可以指定为?       
            ArrayList<Person> list1 = new ArrayList<>();
            list1.add(new Person("张三",23));
            list1.add(new Person("李四",24));
            list1.add(new Person("王五",25));
            
            ArrayList<Person> list2 = new ArrayList<>();
            list2.add(new Person("赵六",26));
            list2.add(new Person("周七",27));
            
            list1.addAll(list2);        //list2必须是list1的子类,子类提升为父类
            System.out.println(list1);
        }
    
    }
    

    增强for的概述和使用

    • A:增强for概述
      • 简化数组和Collection集合的遍历
    • B:格式:
      for(元素数据类型 变量 : 数组或者Collection集合) {
      使用变量即可,该变量就是元素
      }
    • C:案例演示
      • 数组,集合存储元素用增强for遍历
    • D:好处
      • 简化遍历
    package com.heima.jdk5;
    
    import java.util.ArrayList;
    
    import com.heima.bean.Person;
    //增强for循环底层依赖的是迭代器(Iterator)
    public class Demo1_Foreach {
    
        public static void main(String[] args) {
            ArrayList<Person> list = new ArrayList<>();
            list.add(new Person("张三", 23));
            list.add(new Person("李四", 24));
            list.add(new Person("王五", 25));
            list.add(new Person("赵六", 26));
            for (Person person : list) {
                System.out.println(person);
            }
        }
    
        public static void demo1() {
            int[] arr = {11,22,33,44,55};
            for (int i : arr) {
                System.out.println(i);
            }
            
            ArrayList<String> list = new ArrayList<>();
            list.add("a");list.add("b");list.add("c");
            for (String string : list) {
                System.out.println(string);
            }
        }
    
    }
    

    三种迭代的能否删除

    • 普通for循环,可以删除,但是索引要--
    • 迭代器,可以删除,但是必须使用迭代器自身的remove方法,否则会出现并发修改异常
    • 增强for循环不能删除
    package com.heima.jdk5;
    
    import java.util.ArrayList;
    import java.util.Iterator;
    
    import com.heima.bean.Person;
    //增强for循环底层依赖的是迭代器(Iterator)
    public class Demo1_Foreach {
    
        public static void main(String[] args) {
            ArrayList<String> list = new ArrayList<>();
            list.add("b");
            list.add("b");
            list.add("c");
            list.add("a");
            list.add("d");
            list.add("b");
            list.add("b");
            //1,普通for循环删除
            /*for(int i = 0;i < list.size(); i++) {
                if("b".equals(list.get(i))) {
                    list.remove(i--);   //删除的时候后面的所有元素都向前移一位,所以索引要--
                    //就算b出现在第一位也不会报错,因为执行i--后会执行i++。
                }
            }*/
            //2,迭代器删除
            /*for(Iterator<String> it2 = list.iterator(); it2.hasNext();) {
                if("b".equals(it2.next())) {
                    it2.remove();
                }
            }*/
            /*Iterator<String> it = list.iterator();
            while(it.hasNext()) {
                if("b".equals(it.next())) {
                    //list.remove("b"); //不能用集合的删除方法,因为迭代过程中如果集合修改会出现并发修改异常
                    it.remove();
                }
            }*/
            //3,增强for循环不能删除,只能遍历。因为增强for循环的底层是通过迭代器实现的,调用集合的remove方法会引发并发修改异常
            for (String string : list) {
                if("b".equals(string)) {
                    list.remove("b");
                }
            }
            System.out.println(list);
        }
        
        
        public static void demo2() {
            ArrayList<Person> list = new ArrayList<>();
            list.add(new Person("张三", 23));
            list.add(new Person("李四", 24));
            list.add(new Person("王五", 25));
            list.add(new Person("赵六", 26));
            for (Person person : list) {
                System.out.println(person);
            }
        }
    
        public static void demo1() {
            int[] arr = {11,22,33,44,55};
            for (int i : arr) {
                System.out.println(i);
            }
            
            ArrayList<String> list = new ArrayList<>();
            list.add("a");list.add("b");list.add("c");
            for (String string : list) {
                System.out.println(string);
            }
        }
    
    }
    

    静态导入的概述和使用

    • A:静态导入概述
    • B:格式:
      • import static 包名….类名.方法名;
      • 可以直接导入到方法的级别
    • C:注意事项
      • 方法必须是静态的,如果有多个同名的静态方法,容易不知道使用谁?
        这个时候要使用,必须加前缀。由此可见,意义不大,所以一般不用,但是要能看懂。
    package com.heima.jdk5;
    
    import static java.util.Arrays.sort;        //静态导入
    import static java.util.Arrays.toString;
    
    public class Demo2_StaticImport {
        //静态导入开发不用,但是需要能看懂
        public static void main(String[] args) {
            int[] arr = {55,22,33,44,11};
            //Arrays.sort(arr);             //排序
            sort(arr);
            //System.out.println(Arrays.toString(arr));
            //System.out.println(toString(arr));        //toString方法静态导入不能用
            System.out.println(arr);
        }
    
    }
    

    可变参数的概述和使用

    • A:可变参数概述
      • 定义方法的时候不知道该定义多少个参数
    • B:格式
      • 修饰符 返回值类型 方法名(数据类型… 变量名){}
    • C:注意事项:
      • 这里的变量其实是一个数组
      • 如果一个方法有可变参数,并且有多个参数,那么,可变参数肯定是最后一个
    package com.heima.jdk5;
    
    public class Demo3_ChangeableArgs {
    
        /**
         * @param args
         */
        public static void main(String[] args) {
            int[] arr = {11,22,33,44,55};
            //print(arr);
            print(11,22,33,44,55);      //只支持可变参数的形式
            //print();
        }
        
        /*public static void print(int[] arr) {
            for(int i = 0; i < arr.length; i++) {
                System.out.println(arr[i]);
            }
        }*/
        
        public static void print(int x,int ... arr) {           //可变参数其实是一个数组
            for (int i = 0; i < arr.length; i++) {
                System.out.println(arr[i]);
            }
        }
    
    }
    

    Arrays工具类的asList()方法的使用

    • A:案例演示
      • Arrays工具类的asList()方法的使用
      • Collection中toArray(T[] a)泛型版的集合转数组
    package com.heima.jdk5;
    
    import java.util.ArrayList;
    import java.util.Arrays;
    import java.util.List;
    
    public class Demo4_AsList {
    
        /**
         * 数组转换成集合虽然不能增加或减少元素,但是可以用集合的思想操作数组,也就是说可以使用其他集合中的方法
         */
        public static void main(String[] args) {
            //集合转数组,加泛型的
            ArrayList<String> list = new ArrayList<>();
            list.add("a");
            list.add("b");
            list.add("c");
            list.add("d");
            
            String[] arr = list.toArray(new String[10]);    
            for (String string : arr) {     //当集合转换数组时,数组长度如果是小于等于集合的size时,转换后的数组长度等于集合的size
                System.out.println(string); //如果数组的长度大于了size,分配的数组长度就和你指定的长度一样,自动补上null
            }
        }
    
        public static void demo2() {
            /*int[] arr = {11,22,33,44,55};
            List<int[]> list = Arrays.asList(arr);  //将整个数组放在集合里面,泛型是int型数组
            System.out.println(list);   //基本数据类型的数组转换成集合,会将整个数组当作一个对象转换
    */  
            Integer[] arr = {11,22,33,44,55};   //将数组转换成集合,数组必须是引用数据类型
            List<Integer> list = Arrays.asList(arr);
            System.out.println(list);
        }
    
        public static void demo1() {
            String[] arr = {"a","b","c"};
            List<String> list = Arrays.asList(arr);     //数组转换成集合
            //list.add("d");    //UnsupportedOperationException,不能添加
            System.out.println(list);
        }
    
    }
    

    集合嵌套之ArrayList嵌套ArrayList

    package com.heima.list;
    
    import java.util.ArrayList;
    
    import com.heima.bean.Person;
    
    public class Demo5_ArrayListArrayList {
    
        public static void main(String[] args) {
            ArrayList<ArrayList<Person>> list = new ArrayList<>();
            
            ArrayList<Person> first = new ArrayList<>();    //创建第一个班级
            first.add(new Person("杨幂",30));
            first.add(new Person("李冰冰",33));
            first.add(new Person("范冰冰",20));
            
            ArrayList<Person> second = new ArrayList<>();
            second.add(new Person("黄晓明",31));
            second.add(new Person("赵薇",33));
            second.add(new Person("陈坤",32));
            
            //将班级添加到学科集合中
            list.add(first);
            list.add(second);
            
            //遍历学科集合
            for (ArrayList<Person> a : list) {
                for (Person p : a) {
                    System.out.println(p);
                }
            }
        }
    
    }
    

    HashSet储存字符串并遍历

    package com.heima.set;
    
    import java.util.HashSet;
    
    public class Demo1_HashSet {
    
        /**
         * Set集合,无索引,不可以重复,无序(存取不一致)
         */
        public static void main(String[] args) {
            HashSet<String> hs = new HashSet<>();   //创建HashSet对象
            boolean b1 = hs.add("a");
            boolean b2 = hs.add("a");   //当向set集合中储存重复元素的时候返回为false
            hs.add("b");
            hs.add("c");
            hs.add("d");
            System.out.println(hs);     //HashSet的继承体系中有重写toString方法
            System.out.println(b1);
            System.out.println(b2);
            
            for (String string : hs) {  //只要能用迭代器迭代的,就可以使用增强for循环遍历
                System.out.println(string);
            }
        }
    
    }
    

    HashSet如何保证元素唯一性的原理

    • 1.HashSet原理
      • 我们使用set集合都是需要去掉重复元素的,如果在存储的时候逐个equals比较,效率较低,哈希算法提高了去重复的效率,降低了使用equals()方法的次数
      • 当HashSet调用add()方法存储对象的时候,先调用对象的hashCode()方法得到一个哈希值,然后再集合中查找是否有哈希值相同的对象
        • 如果没有哈希值相同的对象就直接存入集合
        • 如果有哈希值相同的对象,就和哈希值相同的对象逐个进行equals比较,比较结果为false就存入,true则不存
    • 2.将自定义类的对象存入HashSet去重复
      • 类中必须重写hashCode()和equals()方法
      • hashCode():属性相同的对象返回值必须相同,属性不同的返回值尽量不同(提高效率)
      • equals():属性相同返回true,属性不同返回false,返回false的时候储存
    package com.heima.bean;
    
    public class Person {
        private String name;
        private int age;    
        public Person() {
            super();
            
        }
        public Person(String name, int age) {
            super();
            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 + "]";
        }
        /*@Override
        public boolean equals(Object obj) {
            System.out.println("执行了吗");
            Person p = (Person)obj;
            return this.name.equals(p.name) && this.age == p.age;
        }
        @Override
        public int hashCode() {
            final int NUM = 38;
            return name.hashCode() * NUM + age;
        }*/
        /*
         * 为什么是31?
         * 1,31是一个质数,质数是能被1和自己本身整除的数
         * 2,31这个数既不大也不小
         * 3,31这个数好算,2的五次方-1,2向左移动5位减一
         */
        @Override
        public int hashCode() {
            final int prime = 31;
            int result = 1;
            result = prime * result + age;
            result = prime * result + ((name == null) ? 0 : name.hashCode());
            return result;
        }
        @Override
        public boolean equals(Object obj) {
            if (this == obj)    //调用的对象和传入的对象是同一个对象
                return true;    //直接返回true
            if (obj == null)    //传入的对象为null
                return false;   //返回false
            if (getClass() != obj.getClass())   //判断两个对象对应的字节码文件是否是同一个字节码
                return false;                   //如果不是直接返回false
            Person other = (Person) obj;        //向下转型
            if (age != other.age)               //调用对象的年龄不等于传入对象的年龄
                return false;                   //返回false
            if (name == null) {                 //调用对象的姓名为null
                if (other.name != null)         //传入对象的姓名不为null
                    return false;               //返回false
            } else if (!name.equals(other.name))//调用对象的姓名不为null,且不等于传入对象的姓名
                return false;                   //返回false
            return true;                        //返回true
        }
        
        
    }
    
    package com.heima.set;
    
    import java.util.HashSet;
    
    import com.heima.bean.Person;
    
    public class Demo1_HashSet {
    
        /**
         * Set集合,无索引,不可以重复,无序(存取不一致)
         */
        public static void main(String[] args) {
            HashSet<Person> hs = new HashSet<>();
            hs.add(new Person("张三", 23));
            hs.add(new Person("张三", 23));
            hs.add(new Person("李四", 24));
            hs.add(new Person("李四", 24));
            hs.add(new Person("李四", 24));
            hs.add(new Person("李四", 24));
            
            System.out.println(hs);
        }
    
        public static void demo1() {
            HashSet<String> hs = new HashSet<>();   //创建HashSet对象
            boolean b1 = hs.add("a");
            boolean b2 = hs.add("a");   //当向set集合中储存重复元素的时候返回为false
            hs.add("b");
            hs.add("c");
            hs.add("d");
            System.out.println(hs);     //HashSet的继承体系中有重写toString方法
            System.out.println(b1);
            System.out.println(b2);
            
            for (String string : hs) {  //只要能用迭代器迭代的,就可以使用增强for循环遍历
                System.out.println(string);
            }
        }
    
    }
    

    LinkedHashSet的使用

    package com.heima.set;
    
    import java.util.LinkedHashSet;
    
    public class Demo2_LinkedHashSet {
    
        /**
         * LinkedHashSet
         * 底层是链表实现的,是set集合中唯一一个能保证怎么存就怎么取的集合对象
         * 因为是HashSet的子类,所以也是保证元素唯一的,与HashSet的原理一样
         */
        public static void main(String[] args) {
            LinkedHashSet<String> lhs = new LinkedHashSet<>();
            lhs.add("a");
            lhs.add("a");
            lhs.add("b");
            lhs.add("c");
            lhs.add("a");
            lhs.add("b");
            lhs.add("c");
            System.out.println(lhs);
        }
    
    }
    

    练习

    package com.heima.test;
    
    import java.util.HashSet;
    import java.util.Random;
    
    public class Test1 {
    
        /**
         * 需求:编写一个程序,获取10个1至20的随机数,要求随机数不能重复。并把最终的随机数输出到控制台
         * 1,有Random类创建随机数对象
         * 2,需要存储10个随机数,而且不能重复,所以我们用HashSet集合
         * 3,如果HashSet的size是小于10就可以不断的存储,如果大于等于10就停止存储
         * 4,通过Random类中的nextInt(n)方法获取1到20之间的随机数,并将这些随机数存储在HashSet集合中
         * 5,遍历HashSet
         */
        public static void main(String[] args) {
            //1,有Random类创建随机数对象
            Random r = new Random();
            //2,需要存储10个随机数,而且不能重复,所以我们用HashSet集合
            HashSet<Integer> hs = new HashSet<>();
            //3,如果HashSet的size是小于10就可以不断的存储,如果大于等于10就停止存储
            while(hs.size() < 10) {
                //4,通过Random类中的nextInt(n)方法获取1到20之间的随机数,并将这些随机数存储在HashSet集合中
                hs.add(r.nextInt(20) + 1);
            }
            //5,遍历HashSet
            for (Integer integer : hs) {
                System.out.println(integer);
            }
        }
    
    }
    
    package com.heima.test;
    
    import java.util.HashSet;
    import java.util.Scanner;
    
    public class Test2 {
    
        /**
         * 使用Scanner从键盘读取一行输入,去掉其中重复字符,打印出不同的那些字符
         * aaaabbbcccddd
         * 分析:
         * 1,创建Scanner对象
         * 2,创建HashSet对象,将字符存储,去掉重复
         * 3,将字符串转换为字符数组,获取每一个字符存储在HashSet集合中,自动去除重复
         * 4,遍历HashSet,打印每一个字符
         */
        public static void main(String[] args) {
            //1,创建Scanner对象
            Scanner sc = new Scanner(System.in);
            System.out.println("请输入一行字符串:");
            //2,创建HashSet对象,将字符存储,去掉重复
            HashSet<Character> hs = new HashSet<>();
            //3,将字符串转换为字符数组,获取每一个字符存储在HashSet集合中,自动去除重复
            String line = sc.nextLine();
            char[] arr = line.toCharArray();
            
            for (char c : arr) {        //char可以换成Character,自动装箱
                hs.add(c);
            }
            //4,遍历HashSet,打印每一个字符
            for(Character ch : hs) {    //Character可以换成char,自动拆箱
                System.out.print(ch);
            }
        }
    
    }
    
    package com.heima.test;
    
    import java.util.ArrayList;
    import java.util.LinkedHashSet;
    import java.util.List;
    
    public class Test3 {
    
        /**
         * 需求:将集合中的重复元素去掉
         * 分析:
         * 1,创建List集合存储若干个重复元素
         * 2,单独定义方法去除重复
         * 3,打印一下List集合
         */
        public static void main(String[] args) {
            //1,创建List集合存储若干个重复元素
            ArrayList<String> list = new ArrayList<>();
            list.add("a");
            list.add("a");
            list.add("b");
            list.add("b");
            list.add("c");
            list.add("c");
            list.add("d");
            list.add("d");
            //2,单独定义方法去除重复
            getSingle(list);
            
            //3,打印一下List集合
            System.out.println(list);
        }
        /*
         * 去除List集合中的重复元素
         * 1,创建一个LinkedHashSet集合
         * 2,将List集合中所有的元素添加到LinkedHashSet集合
         * 3,将List集合中的元素清除
         * 4,将LinkedHashSet集合中的元素添加回List集合中
         */
        private static void getSingle(List<String> list) {
            //1,创建一个LinkedHashSet集合
            LinkedHashSet<String> lhs = new LinkedHashSet<>();
            //2,将List集合中所有的元素添加到LinkedHashSet集合
            lhs.addAll(list);
            //3,将List集合中的元素清除
            list.clear();
            //4,将LinkedHashSet集合中的元素添加回List集合中
            list.addAll(lhs);
        }
    
    }
    

    相关文章

      网友评论

          本文标题:java学习笔记6

          本文链接:https://www.haomeiwen.com/subject/pubuxqtx.html