容器


title: 容器
date: 2020-08-15 18:01:28
tags: [java基础,泛型,ArrayList]

容器/集合的基本概念

数组就是一种容器,可以在其中放置对象或基本类型数据

数组的优势:是一个简单的线性序列,可以快速地访问数组元素,效率高。如果从效率和类型检查的角度讲,数组是最好的

数组的劣势:不灵活。容量需要事先定义好,不能随着需求的变化而扩容。比如:我们在一个用户管理系统中,要吧今天注册的所有用户取出来,那么这样的用户有多少个?我们在写程序时时无法确定的。因此,在这里我们就不能使用数组

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-eDW9CS8I-1597492653247)(https://s1.ax1x.com/2020/08/15/dkUxdP.png)]

泛型

泛型是JDK1.5以后增加的,他可以帮助我们建立类型安全的集合。

泛型的本质就是“数据类型的参数化”。我们可以吧“泛型”理解为数据类型的一个占位符(形式参数),即告诉编译器,在调用泛型时必须传入实际类型。

    package cn.yishan.collection;

    /**
    * 测试泛型
    * @author yishan
    */
    public class TestGeneric {
        public static void main(String[] args) {
            MyCollection<String> mc = new MyCollection<String>();

            mc.set("杨亦山",0);

            String b = mc.get(0);
        }
    }

    class MyCollection<E>{
        Object[] objs = new Object[5];

        public void set(E e,int index){
            objs[index] = e;
        }
        public E get(int index){
            return (E) objs[index];
        }
    }

容器中方法

    package cn.yishan.collection;

    import java.util.*;

    /**
    * 测试Collection接口中的方法
    * @author yishan
    */
    public class TestList {
        public static void main(String[] args) {           
            test01();
        }
        public static void test01(){
            Collection<String> c = new ArrayList<>();

            System.out.println(c.size());
            System.out.println(c.isEmpty());

            c.add("杨大侠");
            c.add("杨大帅");
            System.out.println(c);
            System.out.println(c.size());
            //判断是否包含字符串内容
            System.out.println(c.contains("杨大侠"));
            Object[] obj = c.toArray();
            System.out.println(obj);
            //移除容器里面的对象,并不是删除
            c.remove("杨大帅");
            System.out.println(c);
            //移除容器内所有的元素
            c.clear();
            System.out.println(c.size());
        }       
    }

ArrayList_操作多个List_并集和交集

public static void test02(){
            List<String> list01 = new ArrayList<>();
            list01.add("aa");
            list01.add("bb");
            list01.add("cc");

            List<String> list02 = new ArrayList<>();
            list02.add("aa");
            list02.add("dd");
            list02.add("ee");

            System.out.println("list01:"+list01);
            //把另外一个集合的所有的元素都加进来
            //list01.addAll(list02);
            //移除两个集合相同的部分
            //list01.removeAll(list02);
            //取交集
            list01.retainAll(list02);

            System.out.println("list01:"+list01);
            //查看一个集合是否包含另外一个集合
            System.out.println(list01.containsAll(list02));

        }

ArrayList_索引和顺序相关方法

List是有序,可重复的容器

有序:List中每个元素都有索引标记。可以根据元素的索引标记(在List中的位置)访问元素,从而精确控制这些元素。

可重复:List允许加入重复的元素。更确切的讲,List通常允许满足e1.equals(e2)的元素重复加入容器。

List接口常用的实现类有3个:ArrayList(底层实现是数组)、LinkedList(底层是链表)和Vector(底层也是数组,线程安全)。

    private static void test03() {
            List<String> list = new ArrayList<>();
            list.add("A");
            list.add("B");
            list.add("C");
            list.add("D");
            System.out.println(list);
            //在指定的相应的索引处插入字符
            list.add(2,"我");
            System.out.println(list);
            //在指定的相应的索引处删除字符
            list.remove(2);
            System.out.println(list);
            //在指定的相应的索引处修改字符
            list.set(2,"杨大帅");
            System.out.println(list);
            //获取指定的相应索引处的字符
            System.out.println(list.get(2));

            list.add("C");
            list.add("B");
            list.add("A");
            System.out.println(list);
            //查找该字符,返回该字符第一次出现的索引位置,不存在返回-1
            System.out.println(list.indexOf("B"));
            //查找该字符,返回该字符最后一次出现的索引位置,不存在返回-1
            System.out.println(list.lastIndexOf("B"));
        }

ArrayList 源码解读

ArrayList底层是用数组实现的存储。特点:查询效率高,增删效率低,线程不安全。我们一般使用它。

数组长度是有限的,而ArrayList是可以存放任意数量的对象,长度不受限制,那么他是怎么实现的呢? 数组扩容

  • 扩容长度为 原数组长度右移一位,相当于扩容原数组长度的一半

      private void grow(int minCapacity) {
              // overflow-conscious code
              int oldCapacity = elementData.length;
              int newCapacity = oldCapacity + (oldCapacity >> 1);
              if (newCapacity - minCapacity < 0)
                  newCapacity = minCapacity;
              if (newCapacity - MAX_ARRAY_SIZE > 0)
                  newCapacity = hugeCapacity(minCapacity);
              // minCapacity is usually close to size, so this is a win:
              elementData = Arrays.copyOf(elementData, newCapacity);
          }.
    
remove方法
  • 数组拷贝 数组下标控制位置,自我拷贝,覆盖需要删除的元素

      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;
          }
    
clear方法
  • 循环遍历,依次赋值为null,即清空所有元素

      public void clear() {
              modCount++;
    
              // clear to let GC do its work
              for (int i = 0; i < size; i++)
                  elementData[i] = null;
    
              size = 0;
          }
    
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值