欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

如何實現(xiàn)Java中一個簡單的LinkedList

 更新時間:2017年02月16日 09:08:17   作者:byhieg  
LinkedList與ArrayList都是List接口的具體實現(xiàn)類。下面將介紹如何實現(xiàn)一個簡單的LinkedList,具有很好的參考價值,下面跟著小編一起來看下吧

LinkedList與ArrayList都是List接口的具體實現(xiàn)類。LinkedList與ArrayList在功能上也是大體一致,但是因為兩者具體的實現(xiàn)方式不一致,所以在進行一些相同操作的時候,其效率也是有差別的。

對于抽象的數(shù)據(jù)結(jié)構(gòu)——線性表而言,線性表分為兩種,一種是順序存儲結(jié)構(gòu)的順序表,另一種是通過指針來描述其邏輯位置的鏈表。

針對于具體的Java實現(xiàn):

  1. 順序存儲的順序表是用數(shù)組來實現(xiàn)的,以數(shù)組為基礎(chǔ)進行封裝各種操作而形成的List為ArrayList
  2. 鏈表是用指針來描述其邏輯位置,在Java中以雙向鏈表為基礎(chǔ)進行封裝各種操作而形成的List為LinkedList

針對插入與刪除操作,ArrayList每插入一個元素,首先需要判斷數(shù)組的空間夠不夠,不夠要進行擴容,在有足夠的空間的基礎(chǔ)上,在指定的index位置上插入元素,但是該index及以后的元素都要后移。雖然刪除操作不需要判斷空間夠不夠,但同樣需要該index及以后的元素向前移動,這些移動的操作會增加時間的復雜度。但是對于LinkedList就不一樣,因為使用指針來指示其邏輯的位置,所以插入與刪除的操作的時間復雜度都是 ** O(1) **

雖然對于ArrayList而言,插入與刪除的時間復雜度很高,但是對于查找指定位置的元素這種操作而言,就非常的快,因為可以通過數(shù)組直接得到該下標對應(yīng)的元素。反而,LinkedList而言,無法直接返回指定位置的元素,需要一個個查詢,其時間的復雜度就是 ** O(n) **

如何實現(xiàn)Java的ArrayList經(jīng)典實體類一樣,實現(xiàn)的目的主要在于練手以及掌握官方實現(xiàn)的原理和一些技巧,因此很多需要與其他類配合的方法和功能,就先不在這里實現(xiàn)如iterator等

所以,實現(xiàn)的LinkedList的方法如下:

add方法

get方法

indexOf方法

remove方法

與實現(xiàn)ArrayList的名字一樣,為SimpleLinkedList。源碼地址,歡迎star,fork

構(gòu)建一個雙向鏈表

構(gòu)建的代碼如下:

 private static class Node<E>{
 E item;
 Node<E> next;
 Node<E> prev;
 public Node(E item, Node<E> next, Node<E> prev) {
 this.item = item;
 this.next = next;
 this.prev = prev;
 }
 } 

常規(guī)的雙向鏈表的構(gòu)建方法,一個數(shù)字域存放數(shù)組,一個前指針指向一個Node類型的元素,一個后指針指向一個Node類型的元素。

對于LinkedList的實現(xiàn)而言,還需要以下三個成員變量

 private int size;
 private Node<E> first;
 private Node<E> last;

Add方法

這里實現(xiàn)的add方法是簡單的add(E e)以及add(int index,E e)兩個方法,addAll()將其他集合轉(zhuǎn)換LinkedList的方法,暫時放到以后去實現(xiàn)。

add方法兩個重載方法,其分別對應(yīng)不同的添加方式。先說add(E e)方法的實現(xiàn)。

 public boolean add(E element) {
 addAtLast(element);
 return true;
 }

不指定位置添加元素,則默認添加到了鏈表的最后。addAtLast的核心代碼如下:

 private void addAtLast(E element) {
 Node<E> l = last;
 Node<E> node = new Node<E>(element, null, l);
 last = node;
 if (l == null) {
 first = node;
 } else {
 l.next = node;
 }
 size++;
 }

首先找到最后一位的Node元素,然后根據(jù)element創(chuàng)建一個新的Node元素,其next指向為null,prev指向為最后一位Node元素。在創(chuàng)建完Node元素之后,last就變成了先創(chuàng)建的Node元素,接下來只需要把新node元素加到鏈表中即可。即讓l對象(原最后一位,現(xiàn)倒數(shù)第二位元素的next指針,指向新node元素)。至此,新node元素的next指向null,prev指向倒數(shù)第二個元素,倒數(shù)第二個元素的next指向新node,就將node成功加入鏈表。

上述的操作也可以看出,其插入的操作非常省時間,比起ArrayList,擴容,移動元素快很多。

add的第二個重載方法 add(int index ,E e),先看代碼實現(xiàn):

 public void add(int index, E element) {
 checkRangeForAdd(index);
 if (index == size) {
 addAtLast(element);
 } else {
 Node<E> l = node(index);
 addBeforeNode(element, l);
 }
 }

首先判斷要插入的index是否在范圍內(nèi),在的話,再執(zhí)行后續(xù)的add操作。如果要插入的index剛好是最后一位,則執(zhí)行上面講的addAtLast,如果不是,則得到index所對應(yīng)的Node元素,執(zhí)行addBeforeNode。

獲取index所對應(yīng)的Node元素,是node方法,代碼如下:

 private Node<E> node(int index) {
 if (index < (size << 1)) {
 Node<E> cursor = first;
 for (int i = 0; i < index; i++) {
 cursor = cursor.next;
 }
 return cursor;
 } else {
 Node<E> cursor = last;
 for (int i = size - 1; i > index; i--) {
 cursor = cursor.prev;
 }
 return cursor;
 }
 }

這里的查找采用二分查找,節(jié)省查找時間,而且也應(yīng)用到了雙向鏈表的特點。首先判斷index在前一半的范圍內(nèi),還是后一半的范圍內(nèi)。如果是前一半,則游標Node初始為first,用游標Node元素的next,不斷指向index所在的元素。如果是后一半,則游標Node初始為last,用游標Node元素的prev,不斷指向index所在的元素。

在指定元素的前面插入新節(jié)點的addBeforeNode的方法如下:

 private void addBeforeNode(E element, Node<E> specifiedNode) {
 Node<E> preNode = specifiedNode.prev;
 Node<E> newNode = new Node<>(element, specifiedNode, preNode);
 if (preNode == null) {
 first = newNode;
 } else {
 preNode.next = newNode;
 }
 specifiedNode.prev = newNode;
 size++;
 }

插入的方式很簡單,新節(jié)點的prev是原index元素的prev,新節(jié)點的next是原index元素。剩下的操作是把該node放到鏈表中,讓原index元素的prev的next為新節(jié)點,但是要判斷preNode是不是空,是的話,表示newNode為第一個元素,就是first。

至此,一個add方法,就實現(xiàn)完了。

get方法

get方法在有了上述node方法之后,就非常的簡單。代碼如下:

 public E get(int index) {
 checkRange(index);
 return node(index).item;
 }

checkRange檢查index是否不在范圍內(nèi)。

 private void checkRange(int index) {
 if (index >= size || index < 0) {
 throw new IndexOutOfBoundsException("指定index超過界限");
 }
 }

indexOf方法

indexOf(Object o)用來得到指定元素的下標。

 public int indexOf(Object element) {
 Node<E> cursor = first;
 int count = 0;
 while (cursor != null) {
 if (element != null) {
 if (element.equals(cursor.item)) {
  return count;
 }
 }else{
 if (cursor.item == null) {
  return count;
 }
 }
 count ++;
 cursor = cursor.next;
 }
 return -1;
 }

與ArrayList一樣,從第一位開始查找,首先先判斷element是不是null,分成兩種情況。

remove方法

remove方法與add方法一樣,同樣有兩個重載的方法,remove(Object o)與remove(int index)

先看簡單的remove(int index)方法,代碼如下:

 public E remove(int index) {
 checkRange(index);
 return deleteLink(index);
 }

deleteLink是將該index所對應(yīng)的節(jié)點的鏈接刪除的方法,其代碼如下:

 private E deleteLink(int index) {
 Node<E> l = node(index);
 E item = l.item;
 Node<E> prevNode = l.prev;
 Node<E> nextNode = l.next;
 if (prevNode == null) {
 first = nextNode;
 }else{
 prevNode.next = nextNode;
 l.next = null;
 }
 if (nextNode == null) {
 last = prevNode;
 }else{
 nextNode.prev = prevNode;
 l.prev = null;
 }
 size--;
 l.item = null;
 return item;
 }

首先獲得該index對應(yīng)的Node元素,得到該Node元素的前一個元素和后一個元素。接下來,只需要將前一個元素和后一個元素直接相連即可,其他只需要額外判斷前一個元素和后一個元素是否為null就行。在判斷前一個元素是否為null的時候,只需要操作前一個元素,在判斷后一個元素是否為null的時候,也只需要操作后一個元素。最后,將要刪除的元素各個引用至為null。

remove另一個重載方法remove(Object o),在實現(xiàn)了indexOf和deleteLink方法之后,就非常簡單。

 public boolean remove(Object o) {
 int index = indexOf(o);
 if (index < 0){
 return false;
 }
 deleteLink(index);
 return true;
 }

獲取該元素對應(yīng)對應(yīng)的下標,然后執(zhí)行deleteLink方法,完成remove操作。

總結(jié)

至此,一個功能簡單的LinkedList就實現(xiàn)完成了,全部的代碼可以看源碼地址,

以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,同時也希望多多支持腳本之家!

相關(guān)文章

  • Java簡單計算器的實現(xiàn)

    Java簡單計算器的實現(xiàn)

    這篇文章主要為大家詳細介紹了Java簡單計算器的實現(xiàn),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-12-12
  • Java的String類中的startsWith方法和endsWith方法示例詳解

    Java的String類中的startsWith方法和endsWith方法示例詳解

    大家應(yīng)該都知道startsWith()方法用于檢測字符串是否以指定的前綴開始,endsWith()方法用于測試字符串是否以指定的后綴結(jié)束,本文就Java的String類中的startsWith方法和endsWith方法給大家詳細講解,感興趣的朋友一起看看吧
    2023-11-11
  • Java HashMap 如何正確遍歷并刪除元素的方法小結(jié)

    Java HashMap 如何正確遍歷并刪除元素的方法小結(jié)

    這篇文章主要介紹了Java HashMap 如何正確遍歷并刪除元素的方法小結(jié),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-05-05
  • java中簡單的截取分割字符串實例

    java中簡單的截取分割字符串實例

    下面小編就為大家?guī)硪黄猨ava中簡單的截取分割字符串實例。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-03-03
  • 如何使用Java 8 中的 Stream 遍歷樹形結(jié)構(gòu)

    如何使用Java 8 中的 Stream 遍歷樹形結(jié)構(gòu)

    這篇文章主要介紹了使用Java 8中的Stream遍歷樹形結(jié)構(gòu),我們可以使用Java8中的Stream流一次性把數(shù)據(jù)查出來,然后通過流式處理,我們一起來看看,代碼實現(xiàn)為了實現(xiàn)簡單,就模擬查看數(shù)據(jù)庫所有數(shù)據(jù)到List里面,需要的朋友可以參考下
    2023-08-08
  • MyBatis詳細執(zhí)行流程的全紀錄

    MyBatis詳細執(zhí)行流程的全紀錄

    這篇文章主要給大家介紹了關(guān)于MyBatis詳細執(zhí)行流程的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-04-04
  • 關(guān)于Java數(shù)組聲明、創(chuàng)建、初始化的相關(guān)介紹

    關(guān)于Java數(shù)組聲明、創(chuàng)建、初始化的相關(guān)介紹

    這篇文章主要是關(guān)于Java數(shù)組聲明、創(chuàng)建、初始化的相關(guān)介紹,并給出其對應(yīng)的代碼,需要的朋友可以參考下
    2015-08-08
  • Spring的@PreAuthorize注解自定義權(quán)限校驗詳解

    Spring的@PreAuthorize注解自定義權(quán)限校驗詳解

    這篇文章主要介紹了Spring的@PreAuthorize注解自定義權(quán)限校驗詳解,由于項目中,需要對外開放接口,要求做請求頭校驗,不做其他權(quán)限控制,所以準備對開放的接口全部放行,不做登錄校驗,需要的朋友可以參考下
    2023-11-11
  • kotlin之閉包案例詳解

    kotlin之閉包案例詳解

    這篇文章主要介紹了kotlin之閉包案例詳解,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下
    2021-09-09
  • 解決org.springframework.context.ApplicationContextException報錯的問題

    解決org.springframework.context.ApplicationContextException報錯的

    這篇文章主要介紹了解決org.springframework.context.ApplicationContextException報錯的問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-06-06

最新評論