雙向鏈表的Java實現(xiàn)

雙向鏈表

雙向鏈表也叫雙鏈表铃在,是鏈表的一種阵具,它的每個數(shù)據(jù)結(jié)點中都有兩個指針碍遍,分別指向直接后繼和直接前驅(qū)。
所以阳液,從雙向鏈表中的任意一個結(jié)點開始怕敬,都可以很方便地訪問它的前驅(qū)結(jié)點和后繼結(jié)點。

雙鏈表的重要性質(zhì)就是可以根據(jù)任意節(jié)點快速獲取其前后節(jié)點帘皿,
所以提供了getPrevNode()和getNextNode()方法东跪;

還有就是,雙鏈表遍歷節(jié)點時鹰溜,無需實現(xiàn)任何接口虽填,直接利用節(jié)點間的關系即可完成遍歷。

public class DoubleLinkedList<T> {

    private int size;//鏈表大小
    //由于是雙向曹动,頭尾任意選擇一端即可
    private Node<T> head;//鏈表的頭節(jié)點
    private Node<T> tail;//鏈表的尾節(jié)點

    /**
     * 內(nèi)部類:節(jié)點
     * @param <T>
     */
    public static class Node<T>{
        private Node prev;
        private Node next;
        private T data;

        public Node(T data){
            this.data = data;
        }

        private Node(){}
    }

    /**
     * 添加到鏈尾
     * @param data
     */
    public void add(T data){
        add(size, data);
    }

    /**
     * 添加到任意index處
     * @param index
     * @param data
     */
    public void add(int index, T data){
        Node<T> node = new Node<>(data);
        if(isEmpty()){//鏈表為空
            head = node;
            tail = node;
        }else {
            if(index > size - 1){//索引超出當前鏈表大小卤唉,則添加到鏈尾
                Node<T> temp = tail;
                tail = node;
                temp.next = tail;
                tail.prev = temp;
            }else {//原index位置處有值,索引位置大于index的元素向鏈尾移動(實際并不是移動仁期,只是看上去)
                Node<T> origin = getNode(index);
                Node<T> prev = origin.prev;
                prev.next = node;
                node.prev = prev;
                node.next = origin;
                origin.prev = node;
            }
        }

        size++;
    }

    /**
     * 更新index位置處元素的值
     * @param index
     * @param data
     */
    public void set(int index, T data){
        if(index > size - 1 || index < 0){
            throw new IndexOutOfBoundsException("Index: " + index + ", Size: " + size);
        }
        getNode(index).data = data;
    }

    /**
     * 刪除index位置處的元素
     * @param index
     */
    public void delete(int index){
        if(index > size - 1 || index < 0){
            throw new IndexOutOfBoundsException("Index: " + index + ", Size: " + size);
        }
        if(index == 0){//刪除鏈頭
            head = head.next;
            head.prev = null;
        }else if(index == size -1){//刪除鏈尾
            tail = tail.prev;
            tail.next = null;
        }else {//普通節(jié)點
            Node<T> node = getNode(index);
            Node prev = node.prev;
            Node next = node.next;
            prev.next = next;
            next.prev = prev;
            node.prev = null;
            node.next = null;
        }
        size--;
    }

    /**
     * 獲取index位置處的元素的值
     * @param index
     * @return
     */
    public T getValue(int index){
        return getNode(index) == null ? null : getNode(index).data;
    }

    public T getValue(Node<T> node){
        return node == null ? null : node.data;
    }

    /**
     * 獲取節(jié)點node的上一個節(jié)點
     * @param node
     * @return
     */
    public Node<T> getPrevNode(Node<T> node){
        return node == null ? null : node.prev;
    }

    /**
     * 獲取節(jié)點node的下一個節(jié)點
     * @param node
     * @return
     */
    public Node<T> getNextNode(Node<T> node){
        return node == null ? null : node.next;
    }

    public Node<T> getHeadNode(){
        return head;
    }

    public Node<T> getTailNode(){
        return tail;
    }

    /**
     * 獲取index位置處的元素
     * @param index
     * @return
     */
    public Node<T> getNode(int index){

        if (isEmpty() && (index > size - 1)) {
            throw new IndexOutOfBoundsException("Index: " + index + ", Size: " + size);
        }
        Node<T> result = head;
        int n = 0;
        while (n < index) {//注意這里是 n < index桑驱, 而不是n <= index
            result = result.next;
            n++;
        }
        return result;
    }

    /**
     * 獲取值為data的元素在鏈表中的位置(第一次出現(xiàn)的位置,可能含有多個)
     * @param data
     * @return
     */
    public int indexOf(T data){
        if(isEmpty() || data == null){
            return -1;
        }

        int n = 0;
        Node<T> node = head;
        while (n < size){
            if(data.equals(node.data)){
                return n;
            }
            n++;
        }

        return -1;
    }

    /**
     * 判斷是否有值為data的元素
     * @param data
     * @return
     */
    public boolean containValue(T data){
        return indexOf(data) != -1;
    }

    /**
     * 獲取鏈表的大小
     * @return
     */
    public int size(){
        return size;
    }

    /**
     * 判斷鏈表是否為空
     * @return
     */
    public boolean isEmpty(){
        return size == 0;
    }


}
最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末跛蛋,一起剝皮案震驚了整個濱河市熬的,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌赊级,老刑警劉巖押框,帶你破解...
    沈念sama閱讀 218,858評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異理逊,居然都是意外死亡橡伞,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評論 3 395
  • 文/潘曉璐 我一進店門晋被,熙熙樓的掌柜王于貴愁眉苦臉地迎上來兑徘,“玉大人,你說我怎么就攤上這事羡洛」夷裕” “怎么了?”我有些...
    開封第一講書人閱讀 165,282評論 0 356
  • 文/不壞的土叔 我叫張陵欲侮,是天一觀的道長崭闲。 經(jīng)常有香客問我,道長威蕉,這世上最難降的妖魔是什么刁俭? 我笑而不...
    開封第一講書人閱讀 58,842評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮韧涨,結(jié)果婚禮上牍戚,老公的妹妹穿的比我還像新娘沙兰。我一直安慰自己,他們只是感情好翘魄,可當我...
    茶點故事閱讀 67,857評論 6 392
  • 文/花漫 我一把揭開白布鼎天。 她就那樣靜靜地躺著,像睡著了一般暑竟。 火紅的嫁衣襯著肌膚如雪斋射。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,679評論 1 305
  • 那天但荤,我揣著相機與錄音罗岖,去河邊找鬼。 笑死腹躁,一個胖子當著我的面吹牛桑包,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播纺非,決...
    沈念sama閱讀 40,406評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼哑了,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了烧颖?” 一聲冷哼從身側(cè)響起弱左,我...
    開封第一講書人閱讀 39,311評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎炕淮,沒想到半個月后拆火,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,767評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡涂圆,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年们镜,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片润歉。...
    茶點故事閱讀 40,090評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡模狭,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出卡辰,到底是詐尸還是另有隱情胞皱,我是刑警寧澤邪意,帶...
    沈念sama閱讀 35,785評論 5 346
  • 正文 年R本政府宣布九妈,位于F島的核電站,受9級特大地震影響雾鬼,放射性物質(zhì)發(fā)生泄漏萌朱。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,420評論 3 331
  • 文/蒙蒙 一策菜、第九天 我趴在偏房一處隱蔽的房頂上張望晶疼。 院中可真熱鬧酒贬,春花似錦、人聲如沸翠霍。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽寒匙。三九已至零如,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間锄弱,已是汗流浹背考蕾。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評論 1 271
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留会宪,地道東北人肖卧。 一個月前我還...
    沈念sama閱讀 48,298評論 3 372
  • 正文 我出身青樓,卻偏偏與公主長得像掸鹅,于是被迫代替她去往敵國和親塞帐。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,033評論 2 355

推薦閱讀更多精彩內(nèi)容