学习JavaEE的日子 Day23 迭代器,LinkedList,Vector,Stack,HashSet,LinkedHashSet

news2025/1/17 3:05:17

Day23

1.迭代器

含义:遍历集合中的数据

分类:Iterator 和 ListIterator

Iterator 和 ListIterator 区别

Iterator :Collection接口下所有的实现类都可以获取的迭代器,可以在遍历时删除元素

ListIterator :List接口下所有的实现类可以获取的迭代器,可以在遍历时删除、替换、添加元素,也可以指定下标开始遍历,还可以倒叙遍历

1.1 研究foreach/增强for循环

注意:foreach底层是由Iterator实现

public class Test01 {
	public static void main(String[] args) {
		
		ArrayList<String> list = new ArrayList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		
		for (String element : list) {
			System.out.println(element);
		}
		
		//foreach遍历集合的底层实现:
//		String element;
//		for(Iterator it = list.iterator();it.hasNext();System.out.println(element))
//			element = (String)it.next();	
		
	}
}

注意:
1.泛型只在编码阶段有效,不会编译到class文件

2.泛型只支持引用数据类型

1.2 研究Iterator遍历ArrayList集合的底层原理

注意:

研究底层原理,就势必会看底层源码

底层源码必须找场景去研究,不能从上往下去理解(会越看越乱码)

public class Test02 {
	public static void main(String[] args) {
		
		ArrayList<String> list = new ArrayList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		
		Iterator<String> it = list.iterator();
		while (it.hasNext()) {
			String element = it.next();
			System.out.println(element);
		}
	}
}

思考题:为什么添加和删除才会把modCount++? 为什么修改和查询不会把modCount++?

分析:

集合用于存储数据、管理数据 – 增删改查

进行添加或删除操作时,会改变集合的元素个数

size相等于指针移动,modCount是记录操作(添加和删除)的次数

ArrayList添加元素的过程:

1.判断是否扩容(modCount++)

2.将元素添加到一维数组指定下标上的位置

面试官:ArrayList的数据结构是什么?

一维数组的容器

问你的是元素是存在ArrayList哪里的

为什么研究迭代器的底层???

1.让大家更深入理解面相对象程序设计的思想

2.应付面试

研究源码,必须找场景!!!

//场景
ArrayList<String> list = new ArrayList<>();
		
list.add("aaa");
list.add("bbb");
list.add("ccc");
list.add("ddd");

Iterator<String> it = list.iterator();
while (it.hasNext()) {
    String element = it.next();
    System.out.println(element);
}
public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {
    
    //外部操作数
    //作用:记录修改元素的次数(添加、删除会让该变量++)
    protected transient int modCount = 0;//modCount - 5
}
public class ArrayList<E> extends AbstractList<E> implements List<E>{
    //元素个数
    private int size;//size - 4
    //数据容器 - ["aaa","bbb","ccc","ddd",null,null,null,null,null,null]
    transient Object[] elementData;
    
    //e - ddd
    public boolean add(E e) {
        ensureCapacityInternal(size + 1);//判断是否扩容
        elementData[size++] = e;
        return true;
    }
    
    private void ensureCapacityInternal(int minCapacity) {
        if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
            minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
        }
        ensureExplicitCapacity(minCapacity);
    }

    private void ensureExplicitCapacity(int minCapacity) {
        modCount++;
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    }
    
    public E remove(int index) {
        rangeCheck(index);

        modCount++;
        E oldValue = elementData(index);

        int numMoved = size - index - 1;
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        elementData[--size] = null; // clear to let GC do its work

        return oldValue;
    }
    
    public Iterator<E> iterator() {
        return new Itr();
    }
    
    
    //ArrayList类的内部类 -- 实现了遍历元素的功能
    private class Itr implements Iterator<E> {
        
        int cursor;       // 游标 - 4
        int lastRet = -1; // 当前元素的下标 - 3
        int expectedModCount = modCount;//内部操作数 - 5

        public boolean hasNext() {
            return cursor != size;//4 - 4
        }

        @SuppressWarnings("unchecked")
        public E next() {
            
            /**
            	思考题:为什么在获取元素时,会先判断外部操作数是否等于内部操作数
            		考虑到遍历元素时,如果添加或删除元素,会导致数据个数变化
            		遍历时就有可能出现脏数据,如果外部操作数和内部操作数不相同就以为数据不同意,就报错!
            */
            checkForComodification();//判断外部操作数是否等于内部操作数,如果不等于就报错
            int i = cursor;//i - 3
            if (i >= size)
                throw new NoSuchElementException();
            //获取外部类的成员属性 -- elementData
            Object[] elementData = ArrayList.this.elementData;
            if (i >= elementData.length)
                throw new ConcurrentModificationException();
            cursor = i + 1;
            
            //elementData -- 获取的是ArrayList(外部类的)elementData -> Object[] elementData;
            //Object类型的元素强转为集合中真实类型的元素
            // --> elementData[3] --> Object类型的数据需要强转为String类型
            return (E) elementData[lastRet = i];//elementData[3]
        }
        
        final void checkForComodification() {
            if (modCount != expectedModCount)
                throw new ConcurrentModificationException();
        }

        //重写了Iterator接口中的remove()
        public void remove() {
            //判断当前元素的下标是否小于0
            if (lastRet < 0)
                throw new IllegalStateException();
            
            //判断外部操作数是否和内部操作数一致,不一致就会报错
            checkForComodification();

            try {
                //利用ArraList类的remove()去删除元素
                ArrayList.this.remove(lastRet);
                //把当前元素的下标赋值给游标
                cursor = lastRet;
                //把-1赋值给lastRet
                lastRet = -1;
                //重新把外部操作数赋值给内部操作数
                expectedModCount = modCount;
            } catch (IndexOutOfBoundsException ex) {
                throw new ConcurrentModificationException();
            }
        }
        
    }
    
}

思考题:为什么迭代器是一个接口,不是类???
因为Java提供了很多的集合,不同的集合实现增删改查的原理是不一样的,
所以,不同的集合都实现了各自遍历元素的代码(实现了各自的迭代器)

//迭代器的接口
public interface Iterator<E> {
   
    //判断是否有可迭代的元素
    boolean hasNext();

    //获取下一个元素
    E next();

    //删除元素的默认方法(作用:给实现类去重写,如果实现类可以选择不重写,意味着遍历时不能删除元素,当然实现类可以选择重写,那么就意味着这个迭代器可以在遍历元素时删除元素)
    default void remove() {
        //抛出异常 -- 删除异常
        throw new UnsupportedOperationException("remove");
    }
}

需求:使用Iterator遍历元素,遍历到“bbb”时删除该元素

public class Test03 {
	public static void main(String[] args) {
		
		ArrayList<String> list = new ArrayList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		
		Iterator<String> it = list.iterator();
		while (it.hasNext()) {
			String element = it.next();
			if(element.equals("bbb")){
	//注意:在Iterator遍历时,不能使用list.remove(),会报错
	//list.remove(element);
				
	//注意:在Iterator遍历时,如果要删除元素,使用迭代器中的remove()
	//思路:1.告诉ArrayList需要删除的元素 2.迭代器内部在更新现有数据
				it.remove();
			}
		}
		
		for (String element : list) {
			System.out.println(element);
		}
	}
}

1.3 研究ListIterator遍历

需求:使用ListIterator遍历元素

public class Test04 {
	public static void main(String[] args) {
		
		ArrayList<String> list = new ArrayList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		
		ListIterator<String> listIterator = list.listIterator();
		while(listIterator.hasNext()){
			String element = listIterator.next();
			System.out.println(element);
		}
	}
}

底层源码:

public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {
    
    //外部操作数
    //作用:记录修改元素的次数(添加、删除会让该变量++)
    protected transient int modCount = 0;//modCount - 5
}
public class ArrayList<E> extends AbstractList<E> implements List<E>{
    //元素个数
    private int size;//size - 4
    //数据容器 - ["aaa","bbb","ccc","ddd",null,null,null,null,null,null]
    transient Object[] elementData;
    
    //e - ddd
    public boolean add(E e) {
        ensureCapacityInternal(size + 1);//判断是否扩容
        elementData[size++] = e;
        return true;
    }
    
    private void ensureCapacityInternal(int minCapacity) {
        if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
            minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
        }
        ensureExplicitCapacity(minCapacity);
    }

    private void ensureExplicitCapacity(int minCapacity) {
        modCount++;
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    }
    
    public E remove(int index) {
        rangeCheck(index);

        modCount++;
        E oldValue = elementData(index);

        int numMoved = size - index - 1;
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        elementData[--size] = null; // clear to let GC do its work

        return oldValue;
    }
    
    public Iterator<E> iterator() {
        return new Itr();
    }
    
    //ArrayList类的内部类 -- 实现了遍历元素的功能
    private class Itr implements Iterator<E> {
        
        int cursor;       // 游标 - 4
        int lastRet = -1; // 当前元素的下标 - 3
        int expectedModCount = modCount;//内部操作数 - 5

        public boolean hasNext() {
            return cursor != size;//4 - 4
        }

        @SuppressWarnings("unchecked")
        public E next() {
            
            /**
            	思考题:为什么在获取元素时,会先判断外部操作数是否等于内部操作数
            		考虑到遍历元素时,如果添加或删除元素,会导致数据个数变化
            		遍历时就有可能出现脏数据,如果外部操作数和内部操作数不相同就以为数据不同意,就报错!
            */
            checkForComodification();//判断外部操作数是否等于内部操作数,如果不等于就报错
            int i = cursor;//i - 3
            if (i >= size)
                throw new NoSuchElementException();
            //获取外部类的成员属性 -- elementData
            Object[] elementData = ArrayList.this.elementData;
            if (i >= elementData.length)
                throw new ConcurrentModificationException();
            cursor = i + 1;
            
            //elementData -- 获取的是ArrayList(外部类的)elementData -> Object[] elementData;
            //Object类型的元素强转为集合中真实类型的元素
            // --> elementData[3] --> Object类型的数据需要强转为String类型
            return (E) elementData[lastRet = i];//elementData[3]
        }
        
        final void checkForComodification() {
            if (modCount != expectedModCount)
                throw new ConcurrentModificationException();
        }

        //重写了Iterator接口中的remove()
        public void remove() {
            //判断当前元素的下标是否小于0
            if (lastRet < 0)
                throw new IllegalStateException();
            
            //判断外部操作数是否和内部操作数一致,不一致就会报错
            checkForComodification();

            try {
                //利用ArraList类的remove()去删除元素
                ArrayList.this.remove(lastRet);
                //把当前元素的下标赋值给游标
                cursor = lastRet;
                //把-1赋值给lastRet
                lastRet = -1;
                //重新把外部操作数赋值给内部操作数
                expectedModCount = modCount;
            } catch (IndexOutOfBoundsException ex) {
                throw new ConcurrentModificationException();
            }
        }
        
    }
    
    public ListIterator<E> listIterator() {
        return new ListItr(0);//0 - 设置游标的位置
    }
    
    //指定游标位置
    public ListIterator<E> listIterator(int index) {
        return new ListItr(index);
    }
    
    private class ListItr extends Itr implements ListIterator<E> {
        
        ListItr(int index) {
            super();
            cursor = index;
        }

        //判断是否有上一个元素
        public boolean hasPrevious() {
            return cursor != 0;
        }

        //获取下一个元素的下标
        public int nextIndex() {
            return cursor;
        }

        //获取上一个元素的下标
        public int previousIndex() {
            return cursor - 1;
        }

        //获取上一个元素
        @SuppressWarnings("unchecked")
        public E previous() {
            checkForComodification();
            int i = cursor - 1;
            if (i < 0)
                throw new NoSuchElementException();
            Object[] elementData = ArrayList.this.elementData;
            if (i >= elementData.length)
                throw new ConcurrentModificationException();
            cursor = i;
            return (E) elementData[lastRet = i];
        }

        //设置元素到当前下标上
        public void set(E e) {
            if (lastRet < 0)
                throw new IllegalStateException();
            checkForComodification();

            try {
                //ListIterator的set功能还是依赖于ArrayList
                ArrayList.this.set(lastRet, e);
            } catch (IndexOutOfBoundsException ex) {
                throw new ConcurrentModificationException();
            }
        }

        //添加元素到当前下标上
        public void add(E e) {
            checkForComodification();

            try {
                int i = cursor;
                //ListIterator的add功能还是依赖于ArrayList
                ArrayList.this.add(i, e);
                cursor = i + 1;
                lastRet = -1;
                expectedModCount = modCount;
            } catch (IndexOutOfBoundsException ex) {
                throw new ConcurrentModificationException();
            }
        }
    }
}

需求:使用ListIterator遍历元素,遍历到“bbb”时删除该元素

public class Test05 {
	public static void main(String[] args) {
		
		ArrayList<String> list = new ArrayList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		
		ListIterator<String> listIterator = list.listIterator();
		while(listIterator.hasNext()){
			String element = listIterator.next();
			if(element.equals("bbb")){
				listIterator.remove();
			}
		}
		
		for (String element : list) {
			System.out.println(element);
		}
	}
}

需求:使用ListIterator遍历元素,遍历到“bbb”时替换该元素

public class Test06 {
	public static void main(String[] args) {
		
		ArrayList<String> list = new ArrayList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		
		ListIterator<String> listIterator = list.listIterator();
		while(listIterator.hasNext()){
			String element = listIterator.next();
			if(element.equals("bbb")){
				listIterator.set("王益升");
			}
		}
		
		for (String element : list) {
			System.out.println(element);
		}
	}
}

需求:使用ListIterator遍历元素,遍历到“bbb”时添加元素

public class Test07 {
	public static void main(String[] args) {
		
		ArrayList<String> list = new ArrayList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		
		ListIterator<String> listIterator = list.listIterator();
		while(listIterator.hasNext()){
			String element = listIterator.next();
			if(element.equals("bbb")){
				listIterator.add("王益升");
			}
		}
		
		for (String element : list) {
			System.out.println(element);
		}
	}
}

需求:使用ListIterator指定下标遍历元素

public class Test08 {
	public static void main(String[] args) {
		
		ArrayList<String> list = new ArrayList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		
		ListIterator<String> listIterator = list.listIterator(2);
		while(listIterator.hasNext()){
			String element = listIterator.next();
			System.out.println(element);
		}
		
	}
}

需求:使用ListIterator倒序遍历元素

public class Test09 {
	public static void main(String[] args) {
		
		ArrayList<String> list = new ArrayList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		
		ListIterator<String> listIterator = list.listIterator(list.size());//把指针移动到最后
		while(listIterator.hasPrevious()){//判断是否有上一个元素
			
			String element = listIterator.previous();//获取上一个元素
			System.out.println(element);
		}
		
	}
}

小结

Iterator:
理解:Collection接口下所有的实现类都能获取的迭代器
作用:
遍历元素
删除元素

ListIterator:
理解:List接口下所有的实现类才能获取的迭代器
注意:ListIterator继承Iterator
作用:
遍历元素
删除元素
替换元素
添加元素
指定下标开始遍历
倒序遍历

2.LinkedList(双向链表)

在这里插入图片描述

2.1 LinkedList的使用

注意:

LinkedList和ArrayList使用是一模一样的,这要归功于Conllection和List接口

因为接口是定义标准的!!!

只不过LinkedList和ArrayList底层实现原理是不一样的,LinkedList底层使用双向链表去实现对于数据的操作(增删改查)

public class Test01 {
	public static void main(String[] args) {
		
		//创建LinkedList集合的对象
		LinkedList<String> list = new LinkedList<>();
		
		//添加元素
		list.add("小希");
		list.add("小空");
		list.add("小丽");
		list.add("小光");
		list.add("小李");
		list.add("小阳");
		list.add("小川");
		
		//设置指定下标上的元素
		list.set(1, "小王");
		
		//获取指定下标上的元素
		String str = list.get(1);
		System.out.println("获取指定下标上的元素:" + str);//小王
		
		//获取元素个数
		int size = list.size();
		System.out.println("获取元素个数:" + size);//7
		
		//将元素添加到指定下标的位置
		list.add(4, "小北");
		
		LinkedList<String> newList1 = new LinkedList<>();
		Collections.addAll(newList1, "aaa","bbb","ccc","ccc");//利用Collections工具类给集合做批量添加
		list.addAll(newList1);//将newList1中所有的元素添加到list集合的末尾
		
		LinkedList<String> newList2 = new LinkedList<>();
		Collections.addAll(newList2, "xxx","yyy","zzz","zzz");//利用Collections工具类给集合做批量添加
		list.addAll(4, newList2);//将newList2中所有的元素添加到list集合指定下标的位置
		
		//清空集合中所有的元素
		//list.clear();
		
		System.out.println("判断集合中是否包含指定元素:" + list.contains("王益升"));//true
		System.out.println("判断集合中是否包含子集合中所有的元素:" + list.containsAll(newList1));//true
		
		System.out.println("获取集合中第一次出现该元素的下标:" + list.indexOf("ccc"));//13
		System.out.println("获取集合中最后一次出现该元素的下标:" + list.lastIndexOf("ccc"));//14
		
		System.out.println("判断集合中是否没有元素:" + list.isEmpty());//false
		
		list.remove(0);//根据下标删除元素
		list.remove("小光");//根据元素删除元素
		list.removeAll(newList1);//删除list中包含newList1的元素(去除交集)
		
		list.retainAll(newList2);//保留list中包含newList2的元素(保留交集)
		
		List<String> subList = list.subList(1, 3);//截取开始下标(包含)到结束下标(排他)处的元素,返回新的List集合
		
		//将集合转成数组
		Object[] objs = subList.toArray();
		System.out.println(Arrays.toString(objs));
		
		//将集合转成数组
		String[] ss = new String[2];
		subList.toArray(ss);
		System.out.println(Arrays.toString(ss));
		
		System.out.println("--------------------------------");
		
		//遍历数据 -- for循环
		for (int i = 0; i < list.size(); i++) {
			System.out.println(list.get(i));
		}
		
		System.out.println("--------------------------------");
		
		//遍历数据 -- foreach
		for (String element : list) {
			System.out.println(element);
		}
		
		System.out.println("--------------------------------");
		
		//遍历数据 -- Iterator
		Iterator<String> it = list.iterator();
		while(it.hasNext()){//判断是否有可迭代的元素
			
			String element = it.next();//返回下一个元素
			System.out.println(element);
		}
		
		System.out.println("--------------------------------");
		
		//遍历数据 -- ListIterator
		ListIterator<String> listIterator = list.listIterator();
		while(listIterator.hasNext()){//判断是否有可迭代的元素
			
			String element = listIterator.next();//返回下一个元素
			System.out.println(element);
		}
			
	}
}

2.2 LinkedList独有的方法

public class Test02 {
	public static void main(String[] args) {
		
		LinkedList<String> list = new LinkedList<>();
		
		list.add("小希");
		list.add("abc");
		list.add("小空");
		list.add("abc");
		list.add("小丽");
		list.add("abc");
		list.add("小光");
		
		//将元素添加到首部
		list.addFirst("小王");
		list.offerFirst("小新");
		list.push("小杨");
		
		
		//将元素添加到尾部
		list.addLast("xxx");
		list.offerLast("yyy");
		list.offer("zzz");
		
		System.out.println("获取第一个元素:" + list.element());
		System.out.println("获取第一个元素:" + list.getFirst());
		System.out.println("获取第一个元素:" + list.peek());
		System.out.println("获取第一个元素:" + list.peekFirst());

		System.out.println("获取最后一个元素:" + list.getLast());
		System.out.println("获取最后一个元素:" + list.peekLast());
		
		list.pop();//删除第一个元素
		list.poll();//删除第一个元素
		list.pollFirst();//删除第一个元素
		list.removeFirst();
		
		list.pollLast();//删除最后一个元素
		list.removeLast();//删除最后一个元素		
		
		list.removeFirstOccurrence("abc");//删除第一次出现的元素
		list.removeLastOccurrence("abc");//删除最后一个出现的元素
		
		//倒序遍历
//		Iterator<String> descendingIterator = list.descendingIterator();
//		while(descendingIterator.hasNext()){//hasNext()底层是判断是否有上一个元素
//			String element = descendingIterator.next();//next()底层是获取上一个元素
//			System.out.println(element);
//		}
		
		System.out.println("----------------------------");
		
		for (String element : list) {
			System.out.println(element);
		}
		
	}
}

2.3 LinkedList 的 队列模式(先进先出)

public class Test03 {
	public static void main(String[] args) {
		
		LinkedList<String> list = new LinkedList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		list.add("eee");
		
		while(!list.isEmpty()){//判断是否没有元素
			
			String element = list.removeFirst();//删除第一个元素,并返回
			System.out.println(element);
			
		}
		
		System.out.println("获取元素个数:" + list.size());//0
	}
}

2.4 LinkedList 的 栈模式(先进后出 或 后进先出)

public class Test04 {
	public static void main(String[] args) {
		
		LinkedList<String> list = new LinkedList<>();
		
		list.add("aaa");
		list.add("bbb");
		list.add("ccc");
		list.add("ddd");
		list.add("eee");
		
		while(!list.isEmpty()){
			
			String element = list.removeLast();//删除最后一个元素,并返回
			System.out.println(element);// eee ddd ccc bbb aaa
			
		}
		
		System.out.println("获取元素个数:" + list.size());//0
	}
}

3.Vector(一维数组+线程安全)

3.1 Vector的使用

注意:

Vector和ArrayList使用是一模一样的,这要归功于Conllection和List接口

因为接口是定义标准的!!!

只不过Vector和ArrayList底层实现原理是不一样的,Vector底层使用一维数组+线程安全去实现对于数据的操作(增删改查)

3.2 研究Vector的老方法

Vector的历史:

Vector是元老级别的类(JDK1.0),集合框架是JDK1.2开始才推出的概念,

当时大部分程序员已经习惯Vector去存储数据,为了更好的推广集合框架和保留Vector,

Java让Vector又多实现了List接口

经验:老的方法大多数使用element这个单词

public class Test02 {
	public static void main(String[] args) {
		
		Vector<String> v = new Vector<>();
		
		v.addElement("aaa");
		v.addElement("bbb");
		v.addElement("ccc");
		v.addElement("ddd");
		v.addElement("eee");
		
		//清空
		//v.removeAllElements();
		
		v.removeElementAt(0);//根据下标删除元素
		v.removeElement("ccc");//根据元素删除元素
		
		v.setElementAt("小王", 1);//将元素设置到指定下标的位置
		
		Enumeration<String> elements = v.elements();
		while(elements.hasMoreElements()){
			
			String element = elements.nextElement();
			System.out.println(element);
		}
		
	}
}

4.Stack

4.1 研究Stack的特点 – 栈模式(先进后出)

注意:

该类继承Vector,所以Stack可以使用父类所有的非私有的方法

public class Test01 {
	public static void main(String[] args) {
		
		Stack<String> stack = new Stack<>();
		
		//将元素添加到栈顶
		stack.push("111");
		stack.push("222");
		stack.push("333");
		stack.push("444");
		stack.push("555");
		
		System.out.println("获取栈顶元素:" + stack.peek());//555
		
		int search = stack.search("222");
		System.out.println("获取元素距离栈顶的个数(基于1):" + search);//4
		
		//遍历取出元素
		while(!stack.empty()){
			
			//删除栈顶元素,并返回
			String element = stack.pop();
			System.out.println(element);//555 444 333 222 111
		}
	}
}

5.HashSet

在这里插入图片描述

5.1 HashSet的使用

注意:

HashSet实现Set接口,Set接口继承Collection接口

ArrayList实现List接口,List接口继承Collection接口

以为ArrayList里的方法除了针对于下标操作的,其余的都在Set接口中

public class Test01 {
	public static void main(String[] args) {
		
		//创建HashSet集合的对象
		HashSet<String> set = new HashSet<>();
		
		//添加元素
		set.add("小希");
		set.add("小空");
		set.add("小丽");
		set.add("小光");
		set.add("小李");
		set.add("小林");
		set.add("小川");
		set.add("xxx");
		set.add("yyy");
		set.add("zzz");
		
		//获取元素个数
		int size = set.size();
		System.out.println("获取元素个数:" + size);//10
		
		HashSet<String> newSet1 = new HashSet<>();
		Collections.addAll(newSet1, "aaa","bbb","ccc","ccc");//利用Collections工具类给集合做批量添加
		set.addAll(newSet1);//将newSet1中所有的元素添加到set集合的末尾
		
		//清空集合中所有的元素
		//set.clear();
		
		System.out.println("判断集合中是否包含指定元素:" + set.contains("小王"));//true
		System.out.println("判断集合中是否包含子集合中所有的元素:" + set.containsAll(newSet1));//true
		
		System.out.println("判断集合中是否没有元素:" + set.isEmpty());//false
		
		set.remove("小光");//根据元素删除元素
		set.removeAll(newSet1);//删除set中包含newset1的元素(去除交集)
		
		HashSet<String> newSet2 = new HashSet<>();
		Collections.addAll(newSet2, "xxx","yyy","zzz","zzz");//利用Collections工具类给集合做批量添加
		set.retainAll(newSet2);//保留set中包含newset2的元素(保留交集)
		
		//将集合转成数组
		Object[] objs = set.toArray();
		System.out.println(Arrays.toString(objs));
		
		//将集合转成数组
		String[] ss = new String[set.size()];
		set.toArray(ss);
		System.out.println(Arrays.toString(ss));
		
		System.out.println("--------------------------------");
		
		//遍历数据 -- foreach
		for (String element : set) {
			System.out.println(element);
		}
		
		System.out.println("--------------------------------");
		
		//遍历数据 -- Iterator
		Iterator<String> it = set.iterator();
		while(it.hasNext()){//判断是否有可迭代的元素
			
			String element = it.next();//返回下一个元素
			System.out.println(element);
		}
		
	}
}

5.1 HashSet的特点

特点:无序 + 去重

无序的原因:存入顺序的规则和取出顺序的规则不一致

去重的原因:元素的hash值相同,再判断hash&&==||equals,如果相同就不存入到集合中

public class Test02 {
	public static void main(String[] args) {
		
		HashSet<String> set = new HashSet<>();
		
		set.add("aaa");
		set.add("bbb");
		set.add("ccc");
		set.add("ddd");
		set.add("ddd");
		set.add("Aa");
		set.add("BB");
		
		for (String element : set) {
			System.out.println(element);
		}
		
	}
}

在这里插入图片描述

存入顺序:

1.获取元素的hash值 – hashCode()

2.通过hash值计算在数组的下标

3.判断下标是否有元素

​ 没有 – 直接添加元素

有 – 判断元素是否相同(hash && == || equals)

​ 是 – 不存储(达到了去重的效果)

​ 不是 – 添加元素(JDK1.7 头插法,JDK1.8 尾插法)

取出法:

遍历数组 + 单向链表

6.LinkedHashSet

6.1 LinkedHashSet的使用

注意:

LinkedHashSet 继承 HashSet

3.2 LinkedHashSet的特点

特点:有序 + 去重

有序的原因:在HashSet的基础上添加了双向链表

去重的原因:元素的hash值相同,再判断hash&&==||equals,如果相同就不存入到集合中

public class Test02 {
	public static void main(String[] args) {
		
		LinkedHashSet<String> set = new LinkedHashSet<>();
		
		set.add("aaa");
		set.add("bbb");
		set.add("ccc");
		set.add("ddd");
		set.add("ddd");
		set.add("Aa");
		set.add("BB");
		
		for (String element : set) {
			System.out.println(element);
		}
		
	}

}

在这里插入图片描述

存入顺序:

1.获取元素的hash值 – hashCode()

2.通过hash值计算在数组的下标

3.判断下标是否有元素

​ 没有 – 直接添加(记录上一个添加元素的地址,上一个元素会记录选择元素的地址,双向链表)

​ 有 – 判断元素是否相同(hash && == || equals)

​ 是 – 不添加(达到了去重的效果)

​ 不是 – 添加元素(JDK1.7 头插法,JDK1.8 尾插法)

取出法:

1.找到第一个添加的元素

2.再向下找到下一个元素

注意

4.各种集合的应用场景

ArrayList:存数据,线程不安全

LinkedList:队列模式、栈模式,线程不安全

Vector:弃用,线程安全

Stack:弃用,线程安全

HashSet:去重+无序,线程不安全

LinkedHashSet:去重+有序,线程不安全

TreeSet:排序,线程不安全

HashMap:存key+value,key去重,无序,线程不安全

LinkedHashMap:存key+value,key去重,有序,线程不安全

Hashtable:弃用,存key+value,key去重,无序,线程安全,方法加锁-效率低

ConcurrentHashMap:存key+value,key去重,无序,线程安全,局部加锁、CAS-效率高

TreeMap:存key+value,针对于Key排序

Properties:配置文件

总结

撕迭代器底层

LinkedList:
和ArrayList使用一致
栈模式、队列模式

Vector:
和ArrayList的数据结构一致(一维数组),但是Vector是线程安全的(加锁)
Vector有老的方法

Stack:
Vector的子类
栈模式

HashSet:
数据结构:一维数组+单向链表
特点:无序+去重
注意:理解为什么无序,为什么去重

LinkedHashSet:
数据结构:一维数组+单向链表+双向链表
特点:有序+去重
注意:理解为什么有序,为什么去重

比较器接口

作用:排序时使用

分类:

​ 内置比较器:Comparable - compareTo()

​ 外置比较器:Comparator - compare()

使用场景:

​ 内置比较器:对象要想存入TreeSet、TreeMap中,对象所属的类必须要实现内置比较器

​ 外置比较器:当内置比较的规则不满足现在的需求,但又不能改动内置比较器规则时

优先级别:外置比较器 > 内置比较器

3.判断下标是否有元素

​ 没有 – 直接添加(记录上一个添加元素的地址,上一个元素会记录选择元素的地址,双向链表)

​ 有 – 判断元素是否相同(hash && == || equals)

​ 是 – 不添加(达到了去重的效果)

​ 不是 – 添加元素(JDK1.7 头插法,JDK1.8 尾插法)

取出法:

1.找到第一个添加的元素

2.再向下找到下一个元素

注意

4.各种集合的应用场景

ArrayList:存数据,线程不安全

LinkedList:队列模式、栈模式,线程不安全

Vector:弃用,线程安全

Stack:弃用,线程安全

HashSet:去重+无序,线程不安全

LinkedHashSet:去重+有序,线程不安全

TreeSet:排序,线程不安全

HashMap:存key+value,key去重,无序,线程不安全

LinkedHashMap:存key+value,key去重,有序,线程不安全

Hashtable:弃用,存key+value,key去重,无序,线程安全,方法加锁-效率低

ConcurrentHashMap:存key+value,key去重,无序,线程安全,局部加锁、CAS-效率高

TreeMap:存key+value,针对于Key排序

Properties:配置文件

总结

撕迭代器底层

LinkedList:
和ArrayList使用一致
栈模式、队列模式

Vector:
和ArrayList的数据结构一致(一维数组),但是Vector是线程安全的(加锁)
Vector有老的方法

Stack:
Vector的子类
栈模式

HashSet:
数据结构:一维数组+单向链表
特点:无序+去重
注意:理解为什么无序,为什么去重

LinkedHashSet:
数据结构:一维数组+单向链表+双向链表
特点:有序+去重
注意:理解为什么有序,为什么去重

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1496680.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

京东数据分析平台(京东店铺数据分析工具)推荐

京东店铺数据分析能够帮助商家了解自己的经营状况&#xff0c;优化商品策略&#xff0c;提高销售效率。以下是京东店铺数据分析的一些基本步骤和方法&#xff1a; 首先&#xff0c;在进行京东店铺数据分析时&#xff0c;我们需要借助一些电商数据分析工具来获取相关数据&#…

一次一对一服务引起的沉思和笑话(微信号Stefan)

前情提要 客户需求&#xff1a; 分析页面代码和接口请求协议和参数需求&#xff0c;将人工下载视频怎么获得最终的视频链接&#xff0c;这一逻辑清晰的展示并讲解清除。我询问了是否需要成品爬虫&#xff0c;他说代码他自己能搞定。 我给的价格选择&#xff1a; 第一种、首…

基于状态机的按键消抖实现

摸鱼记录 Day_14 !(^O^)y review 在day_13中以按键状态判断为例学习了状态分析基于状态机的按键消抖原理-CSDN博客 分析得到了下图&#xff1a; 今日任务&#xff1a;完成此过程 !(^O^)y 小梅哥对应视频&#xff1a; 15B 基于状态机的按键消抖Verilog实现_哔哩哔哩…

[C#]winform基于C2PNet算法实现室内和室外图像去雾

【CP2Net框架】 https://github.com/YuZheng9/C2PNet 【CP2Net介绍】 Abstract 考虑到不适定的性质&#xff0c;发展了单图像去模糊的对比正则化&#xff0c;引入了来自负图像的信息作为下界。然而&#xff0c;对比样本是非一致的&#xff0c;因为阴性通常距离清晰&#xff…

二维码门楼牌管理系统应用场景:赋能市场研究与城市决策的新动力

文章目录 前言一、市场研究的新视角&#xff1a;门牌数据揭示市场趋势二、城市规划的得力助手&#xff1a;门牌数据指导资源分配三、决策制定的科学依据&#xff1a;门牌数据提升决策准确性四、未来展望&#xff1a;二维码门楼牌管理系统的更多可能性 前言 随着信息技术的飞速…

国创证券|存钱有窍门,掌握这五个窍门利息高出不少?

大部分居民会把手中闲置的资金存入银行&#xff0c;享用安稳的收益&#xff0c;其间存在以下五个诀窍&#xff1a; 1、比照不同银行的利率 储户在去银行存钱时&#xff0c;能够比照不同银行的利率&#xff0c;尽量地挑选利率较高的银行存&#xff0c;来获取较多的收益&#x…

UML类图详解

目录 概述1 什么是UMLUML模型和图形UML2.2中一共定义了14种图示。 2 UML类图作用3 类图格式4类与类之间的关系表达 概述 我们在阅读开源项目时&#xff0c;总是希望能比较高效的整理清楚项目中的各个类之间的关系&#xff0c;那么有没有相应的工具能高效、简洁的表示清楚类关系…

CMake:开始

文章目录 在计算机上获取和安装 CMake目录结构基本 CMake 用法CMakeLists文件适用于 CMake 的 Hello World配置和生成运行 CMake GUI运行 ccmake Curses 接口从命令行运行 CMake将编译器指定为 CMake构建配置构建项目 在计算机上获取和安装 CMake 在使用 CMake 之前&#xff0…

Sharding-JDBC源码解析与vivo的定制开发

作者&#xff1a;vivo IT 平台团队 - Xiong Huanxin Sharding-JDBC是在JDBC层提供服务的数据库中间件&#xff0c;在分库分表场景具有广泛应用。本文对Sharding-JDBC的解析、路由、改写、执行、归并五大核心引擎进行了源码解析&#xff0c;并结合业务实践经验&#xff0c;总结…

Unity 轮转图, 惯性, 自动回正, 点击选择

简单的实现 2D 以及 3D 的轮转图, 类似于 Web 中无限循环的轮播图那样. 文中所有代码均已同步至 github.com/SlimeNull/UnityTests 3D 轮转图: Assets/Scripts/Scenes/CarouselTestScene/Carousel.cs2D 轮转图: Assets/Scripts/Scenes/CarouselTestScene/UICarousel.cs 主要逻…

Nodejs web服务器之GET、POST请求初次体验

一、认识http请求 步骤 1.DNS解析域名&#xff0c;找到ip地址&#xff0c;建立TCP连接&#xff0c;发起http请求 2.服务器接收到http请求&#xff0c;进行处理&#xff0c;返回数据 3.客户端接收到返回的数据&#xff0c;处理数据&#xff08;比如渲染页面&#xff09; 二、no…

新生儿睡眠抖动:温馨抚慰宝宝的安稳梦乡

引言 新生儿的睡眠过程常常伴随着轻微的抖动&#xff0c;对于许多父母来说&#xff0c;这可能会引起一些担忧。在这篇文章中&#xff0c;我们将探讨新生儿睡眠抖动的原因和注意事项&#xff0c;帮助父母更好地理解和处理宝宝的这种行为&#xff0c;为宝宝提供安心的睡眠环境。…

基于FPGA的HyeperRam接口设计与实现

一 HyperRAM 针对一些低功耗、低带宽应用&#xff08;物联网、消费产品、汽车和工业应用等&#xff09;&#xff0c;涉及到外部存储&#xff0c;HyperRAM提供了更简洁的内存解决方案。 HyperRAM具有以下特性&#xff1a; 1、超低功耗&#xff1a;200MHz工作频率下读写不到50mW…

UE5数字孪生系列笔记(一)

智慧城市数字孪生系统 虚幻引擎连接数据库 将自己的mysql版本的libmysql.dll替换掉插件里面的libmysql.dll 然后将这个插件目录复制到虚幻项目目录下 然后添加这个插件即可 新建一个UMG&#xff0c;添加一个按钮试试&#xff0c;数据库是否连接 将UI添加到视口 打印是否连接…

自研在线CAD系统介绍

去年调研了已有的在线的CAD系统(悟空CAD、维杰地图、梦想控件)&#xff0c;基本上都是按年收费&#xff0c;还相当的贵&#xff0c;基于此&#xff0c;就萌生了自己研发CAD系统的想法&#xff0c;从技术选型、框架设计、代码实现基本为都是自研实现。已经有了初步的成果。 10M…

chatGPT的耳朵!OpenAI的开源语音识别AI:Whisper !

语音识别是通用人工智能的重要一环&#xff01;可以说是AI的耳朵&#xff01; 它可以让机器理解人类的语音&#xff0c;并将其转换为文本或其他形式的输出。 语音识别的应用场景非常广泛&#xff0c;比如智能助理、语音搜索、语音翻译、语音输入等等。 然而&#xff0c;语音…

如何选择乐歌升降台,一张图带你了解全型号参数功能

在现代办公环境中&#xff0c;久坐已成为一种常态&#xff0c;而这种生活方式带来的不良影响日益凸显。乐歌办公升降电脑台应运而生&#xff0c;不仅是一种办公家具&#xff0c;更是健康办公的有力助手。让我们从多个角度深入了解这款产品的功能意义。 1. 台面层数 乐歌办公升…

mysql bug( InnoDB: Error number 22),表突然不能读取

mysql bug&#xff08; InnoDB: Error number 22&#xff09;&#xff0c;表突然不能读取 bug最开始的bug&#xff1a;表突然不能读取关闭mysql容器&#xff0c;再次重启失败 解决方案不重建容器的几种可能措施重建容器重建如果懒得打命令或者忘记命令可能的run bug&#xff1a…

MATLAB中设置输出格式

目录 设置输出中行间距的格式 设置浮点数格式 使代码换行以适应窗口宽度 隐藏输出 按页查看输出 清空命令行窗口 MATLAB 同时在命令行窗口和实时编辑器中显示输出。可以使用提供的多个选项为输出显示设置格式。 设置输出中行间距的格式 默认情况下&#xff0c;MATLAB 会…

【Qt学习】QProgressBar的使用(进度条的实现)

文章目录 1. 介绍2. 实例2.1 按钮启动进度条2.2 更改进度条样式2.3 资源文件 1. 介绍 详细的 QProgressBar 内容可以通过 查阅Qt官方文档 &#xff0c;这里进行简要的总结&#xff1a; QProgressBar 是Qt框架中的一个控件&#xff0c;用于显示进度条&#xff1a; QProgressBar…