Java基礎之淺談集合

漁夫會扎蛙發表於2022-04-07

前言

  • 兄弟們,終於更到Java集合了?。
  • 學完集合我們就可以刷演算法題了?。
  • 但是目前博主還沒有這個實力?。

一、Java集合框架介面

  • Java集合其實就是別人寫好的資料結構,分別為棧、佇列、連結串列、動態擴容陣列、雜湊表、二叉樹等等
  • 資料結構呢,在這篇文章中不作為重點,我們主要介紹實現了此資料結構的集合。
  • 現在我們開始吧?(Φ皿Φ)

1.1Collection介面

  • 我想大家也看到上面的圖了,在Java類庫中,集合類的基本介面是Collection介面。

  • Collection介面有三個子介面分別為List、Set

  • Collection介面中有很多的方法,其中有一種利與集合遍歷的方法,就是使用迭代器iterator遍歷。

  • boolean add(E e)確保此集合包含指定的元素

  • boolean addAll(Collection<? extends E> c)將指定集合中所有元素新增到此集合

  • void clear()從此集合中刪除所有元素

  • boolean contains(Object o)如果此集合包含指定集合中的所有元素,則返回true

  • boolean equals(Object o)將指定的物件與此集合進行比較以獲得相等性

  • int hashCode() 返回此集合的雜湊碼值

  • boolean isEmpty()如果此集合不包含元素,則返回true

  • Iterator iterator()返回此集合中的元素的迭代器

  • boolean remove(Object o)從該集合中刪除指定元素的單個例項

  • boolean removeAll(Collection<?> c)刪除指定集合中包含的所有此集合的元素

  • boolean retainAll(Collection<?> c)僅保留此集合中包含在指定集合中的元素

  • int size() 返回此集合中的元素數

  • Object[] toArray()返回一個包含此集合中所有元素的陣列

  • T[] toArray(T[] a)返回包含此集合中所有元素的陣列,返回的陣列的執行時型別時指定陣列的執行時型別

1.2Iterator迭代器

  • Collection也繼承了Iterable介面,具體操作看如下或者上圖。
  • 我給大家寫一段對於集合使用迭代器遍歷的基本操作。
public class Test1 {
    public static void main(String[] args) {
        ArrayList<Integer> list = new ArrayList<>();//首先我們建立一個ArrayList集合
        list.add(1);//向集合中新增元素
        list.add(2);
        list.add(3);
        Iterator<Integer> iter = list.iterator();//呼叫集合中的迭代器方法
        while(iter.hasNext()) {//判斷集合中是否還有元素
            int it = iter.next();//把集合中的元素取出來,然後跳過當前元素,指向下一個元素
            System.out.println(it);//輸出?
        //這就是迭代器使用的基本操作啦!前面講過,在迭代器操作過程中,我們不能在迴圈裡對元素進行操作的o?
        }
    }
}
  • 我給大家畫一個迭代器移動的圖?
  • 這就是我們迭代器移動的原理,其實它使指在了兩個元素中間進行移動的。
for(Integer value:list) {
    System.out.println(value);//當然使用增強for迴圈,可謂是對我們集合非常的方便簡潔。
}
    Iterator<Integer> iter = list.iterator();
    //我們可以利用lambda表示式。
    //直接呼叫forEachRemaining方法。(但是並不推薦使用此方法)
    iter.forEachRemaining(value-> System.out.println(list));
  • 對於Iterator介面還有一個remove()方法,見名知意刪除,用法也沒什麼大區別。
  • 具體的Collection方法可以檢視API文件進行學習,在這裡我不一一介紹了。
  • boolean hasNext()如果集合中還有元素,則返回true
  • E next()返回迭代中的下一個元素
  • default void remove()刪除此迭代器返回的當前元素
  • default void forEachRemaining(Consumer<? super E> action)對每個剩餘元素執行給定的操作。

二、List介面✍

2.1List介面的特點

  1. 對於前面的學習我們已經知道了,集合的兩個基本介面為:Collection和Map。
  2. List是一個有序集合,當我們使用List介面的實現類時,元素都是按順序新增進來的,就跟我們的陣列一樣。
  3. 同時List集合中可以新增重複的元素,每個元素有自己的下標。
  4. 所以我們可以採用兩種訪問方式:①使用迭代器訪問 ②使用整數索引(陣列下標)來訪問----->也稱之為隨機訪問。
  5. ListIterator介面時Iterator的一個子介面,這個介面可以讓我們在使用迭代器遍歷時,對元素進行操作,我們可以對元素進行增刪改查操作。
  6. List介面下面有三個實現類分別為:ArrayListLinkedListVector

2.2ArrayList

  • ArrayList是用陣列實現的,它支援動態擴容機制。當前陣列是由預設構造方法生成的空陣列並且第一次新增資料。此時minCapacity等於預設的容量(10)那麼根據下面邏輯可以看到最後陣列的容量會從0擴容成10。而後的陣列擴容才是按照當前容量的1.5倍進行擴容。
    優點:
  1. 根據下標遍歷元素效率高
  2. 根據下標查詢元素效率高
  3. 可以自動擴容
    缺點:
  4. 插入、刪除元素效率低。陣列所謂的刪除元素其實都是覆蓋,沒有真正的刪除
  • Collection介面所擁有的方法,ArrayList都有
  • void add(int index , E element)在此列表中的指定位置插入指定的元素
  • Object clone()返回此ArrayList例項的淺拷貝
  • boolean contains(Object o)如果此列表包含指定的元素,則返回true
  • void ensureCapacity(int minCapacity)如果需要,增加此ArrayList例項的容量,以確保它可以至少儲存最小容量引數指定的元素
  • void forEach(Consumer<? super E> action)對Iterable的每個元素執行給定的操作,直到所有元素都被處理或者動作引發異常。
  • E get(int index)返回此列表中指定位置的元素
  • int indexOf(Object o)返回此列表中指定元素的第一次出現的索引,如果此列表不包含元素,則返回-1
  • boolean isEmpty()如果此列表不包含元素,則返回true
  • int lastIndexOf(Object o)返回此列表中指定元素的最後一次出現的索引,如果此列表不包含元素,則返回-1
  • ListIterator listIterator()返回列表中的列表迭代器
  • E set(int index , E element)用指定的元素替換此列表中指定位置的元素
  • void sort(Comparator<? super E> c)使用提供的Comparator對此列表進行排序比較元素
  • void trimToSize()修改這個ArrayList例項的容量是列表當前的大小

2.3LinkedList

  • LinkedList是用連結串列來實現的,是一個有序集合。
    優點:插入、刪除效率高
    缺點:查詢、修改效率低
  • LinkedList的add()方法是連結串列的尾插法。
  • LinkedList類實現了ListIterator介面,可以在迭代的時候對元素進行操作
public class Test1 {
    public static void main(String[] args) {
        LinkedList<Integer> list = new LinkedList<>();
        list.add(1);
        list.add(2);
        list.add(3);
        list.add(4);
        list.add(5);
        ListIterator<Integer> iterator = list.listIterator();
        Integer next = iterator.next();
        iterator.add(10);//會在迭代器位置前新增元素
        System.out.println(list);
      //輸出結果[1, 10, 2, 3, 4, 5]
    }
}
  • LinkedList常用的方法如下:仍然是Collection有的它也有
  • void addFirst(E e)在該列表開頭插入指定的元素
  • void addLast(E e)將指定的元素追加到此列表的末尾
  • void clear()從列表中刪除所有元素
  • Object clone()返回此LinkedList的淺版本
  • Iterator descendingIterator以相反的順序返回此deque中的元素的迭代器
  • E element()檢索但不刪除此列表的頭(第一個元素)
  • E get(int index)返回此列表中指定位置的元素
  • E getFirst()返回此列表中的第一個元素
  • E getLast()返回此列表中的最後一個元素
  • boolean offer(E e)將指定的元素新增到為此列表的尾部(最後一個元素)
  • boolean offerFirst(E e)在此列表的前面插入指定的元素
  • boolean offerLast(E e)在該列表的末尾插入指定的元素
  • E peek()檢索但不刪除此列表的頭
  • E peekLast()檢索但不刪除此列表的第一個元素
  • E peekLast()檢索但不刪除此列表的最後一個元素
  • E poll()檢索並刪除此列表的頭
  • E pollFirst()檢索並刪除此列表的第一個元素
  • E pollLast()檢索並刪除此列表的最後一個元素
  • E pop()從此列表表示的堆疊中彈出一個元素
  • void push(E e)將元素推送到由此列表表示的堆疊上
  • E set(int index, E element)用指定的元素替換此列表中指定位置的元素

三、Set介面✍

3.1Set介面的特點

  • 首先我們要來了解一下什麼是雜湊表
  • 可以用來快速查詢物件,每個物件都擁有一個自己的雜湊碼。雜湊碼是由String類的hashCode方法產生的。如果a.equals(b)為true,則a與b的hashCode必須相等。
  • Java集合中提供的HashSet類的實現就基於雜湊表的集。
  • 它是無序的插入。
  • 元素不可以重複
  • 無法根據下標進行操作
  • Set介面中的方法和Collection中方法一致的。Set介面取出方式只有一種,迭代器。

3.2HashSet

    |——HashSet:底層資料結構是雜湊表,執行緒是不同步的。無序,高效;

        HashSet集合保證元素唯一性:通過元素的hashCode方法,和equals方法完成的。

        當元素的hashCode值相同時,才繼續判斷元素的equals是否為true。

        如果為true,那麼視為相同元素,不存。如果為false,那麼儲存。

        如果hashCode值不同,那麼不判斷equals,從而提高物件比較的速度。

      |——LinkedHashSet:有序,hashset的子類。

    |——TreeSet:對Set集合中的元素的進行指定順序的排序。不同步。TreeSet底層的資料結構就是二叉樹。
  對於ArrayList集合,判斷元素是否存在,或者刪元素底層依據都是equals方法。

  對於HashSet集合,判斷元素是否存在,或者刪除元素,底層依據的是hashCode方法和equals方法。

3.3TreeSet

  1. 用於對Set集合進行元素的指定順序排序,排序需要依據元素自身具備的比較性。

  2. 如果元素不具備比較性,在執行時會發生ClassCastException異常。

  3. 所以需要元素實現Comparable介面,強制讓元素具備比較性,複寫compareTo方法

  4. 依據compareTo方法的返回值,確定元素在TreeSet資料結構中的位置。

  5. TreeSet方法保證元素唯一性的方式:就是參考比較方法的結果是否為0,如果return 0,視為兩個物件重複,不存。

四、Map介面✍

4.1對映

  • 對映用來存放鍵/值對,如果提供了鍵,就能查詢到值。
  • Java類庫為對映提供了兩個通用的實現:HaspMap和TreeMap。
  • Map集合:

    |——Hashtable:底層是雜湊表資料結構,是執行緒同步的。不可以儲存null鍵,null值。

    |——HashMap:底層是雜湊表資料結構,是執行緒不同步的。可以儲存null鍵,null值。替代了Hashtable.

    |——TreeMap:底層是二叉樹結構,可以對map集合中的鍵進行指定順序的排序。

  • Map集合儲存和Collection有著很大不同:

    Collection一次存一個元素;Map一次存一對元素。

    Collection是單列集合;Map是雙列集合。

    Map中的儲存的一對元素:一個是鍵,一個是值,鍵與值之間有對應(對映)關係。

    特點:要保證map集合中鍵的唯一性。

  • 1,新增。

    put(key,value):當儲存的鍵相同時,新的值會替換老的值,並將老值返回。如果鍵沒有重複,返回null。

    void putAll(Map);

  • 2,刪除。

    void clear():清空

    value remove(key) :刪除指定鍵。

  • 3,判斷。

    boolean isEmpty():

    boolean containsKey(key):是否包含key

    boolean containsValue(value) :是否包含value

  • 4,取出。

    int size():返回長度

    value get(key) :通過指定鍵獲取對應的值。如果返回null,可以判斷該鍵不存在。當然有特殊情況,就是在hashmap集合中,是可以儲存null鍵null值的。

    Collection values():獲取map集合中的所有的值。

  • 5,想要獲取map中的所有元素:

    原理:map中是沒有迭代器的,collection具備迭代器,只要將map集合轉成Set集合,可以使用迭代器了。之所以轉成set,是因為map集合具備著鍵的唯一性,其實set集合就來自於map,set集合底層其實用的就是map的方法。

★ 把map集合轉成set的方法:

  • Set keySet();

  • Set entrySet();//取的是鍵和值的對映關係。

  • Entry就是Map介面中的內部介面;

  • 為什麼要定義在map內部呢?entry是訪問鍵值關係的入口,是map的入口,訪問的是map中的鍵值對。

——————————————————————————————————————————————————————
★取出map集合中所有元素的方式一:keySet()方法。

  • 可以將map集合中的鍵都取出存放到set集合中。對set集合進行迭代。迭代完成,再通過get方法對獲取到的鍵進行值的獲取。
       Set keySet = map.keySet();

      Iterator it = keySet.iterator();

      while(it.hasNext()) {

        Object key = it.next();

        Object value = map.get(key);

        System.out.println(key+":"+value);

      }
——————————————————————————————————————————————————————

★取出map集合中所有元素的方式二:entrySet()方法。

      Set entrySet = map.entrySet();

      Iterator it = entrySet.iterator();

      while(it.hasNext()) {

        Map.Entry me = (Map.Entry)it.next();

        System.out.println(me.getKey()+"::::"+me.getValue());

      }

——————————————————————————————————————————————————————

★使用集合的技巧:

  • 看到Array就是陣列結構,有角標,查詢速度很快。

  • 看到link就是連結串列結構:增刪速度快,而且有特有方法。addFirst; addLast; removeFirst(); removeLast(); getFirst();getLast();

  • 看到hash就是雜湊表,就要想要雜湊值,就要想到唯一性,就要想到存入到該結構的中的元素必須覆蓋hashCode,equals方法。

  • 看到tree就是二叉樹,就要想到排序,就想要用到比較。

★比較的兩種方式:

  • 一個是Comparable:覆蓋compareTo方法;

  • 一個是Comparator:覆蓋compare方法。

  • LinkedHashSet,LinkedHashMap:這兩個集合可以保證雜湊表有存入順序和取出順序一致,保證雜湊表有序。

★集合什麼時候用?

  • 當儲存的是一個元素時,就用Collection。當儲存物件之間存在著對映關係時,就使用Map集合。

  保證唯一,就用Set。不保證唯一,就用List。

五、排序與查詢小補充

  • Collections類提供的用於排序和查詢的類方法如下:?
  • public static sort(List list):該方法可以將list的元素按升序排序
  • int binarySearch(List list,T key,CompareTo c):二分查詢
  • String類實現了Comparable介面,規定字串按字典序比較大小,如果連結串列中存放的物件不是String物件,那麼建立物件的類必須實現Comparable介面,即用該介面中的int compareTo(T b)方法來規定物件的大小關係,(原因是sort()方法在排序時需要呼叫名字時compareTo的方法,來比較的物件大小)。
  • 就時如果物件不是String型別的,我們在用sort時需要自己編寫compareTo()方法。

六、結尾

  • 對於Java集合內容就總結這麼多(後面比較水了!),若想深入學習等待後續更新,基礎部分掌握這些足矣。如果有不足之處,希望大家多多包涵,多多支援。如果有不懂的地方可以直接私信問我,歡迎來訪!
  • 我將會繼續更新關於Java的學習知識,感興趣的小夥伴可以關注一下。
  • 文章寫得比較走心,用了很長時間,絕對不是copy過來的!
  • 尊重每一位學習知識的人,同時也尊重每一位分享知識的人。
  • ?你的點贊與關注,是我努力前行的無限動力。?

相關文章