教育行業(yè)A股IPO第一股(股票代碼 003032)

全國(guó)咨詢(xún)/投訴熱線:400-618-4000

java數(shù)據(jù)結(jié)構(gòu)-雙鏈表的數(shù)據(jù)操作及特性

更新時(shí)間:2018年12月07日11時(shí)35分 來(lái)源:傳智播客 瀏覽次數(shù):

# 數(shù)據(jù)結(jié)構(gòu)-雙鏈表的數(shù)據(jù)操作及特性

java

## 概述

? 上一章中,我們已經(jīng)入門(mén)了單鏈表的添加數(shù)據(jù),刪除數(shù)據(jù)及更新數(shù)據(jù),但是在刪除的時(shí)候,或者在更新的時(shí)候,它的效率不高因?yàn)椋瑳](méi)一個(gè)節(jié)點(diǎn)只記錄了它的下一個(gè)節(jié)點(diǎn),這樣一來(lái),遍歷節(jié)點(diǎn)就變得很慢,添加時(shí),添加到指定的節(jié)點(diǎn)就變得效率低下。那么怎么解決呢?這一章,我們來(lái)看看雙鏈表是如何操作的,以及如何遍歷和快速的添加節(jié)點(diǎn)的。

## 雙鏈表介紹

? 雙向鏈表也叫[雙鏈表](https://baike.baidu.com/item/%E5%8F%8C%E9%93%BE%E8%A1%A8),是鏈表的一種,它的每個(gè)數(shù)據(jù)結(jié)點(diǎn)中都有兩個(gè)[指針](https://baike.baidu.com/item/%E6%8C%87%E9%92%88),分別指向直接后繼和直接前驅(qū)。所以,從雙向鏈表中的任意一個(gè)結(jié)點(diǎn)開(kāi)始,都可以很方便地訪問(wèn)它的前驅(qū)結(jié)點(diǎn)和后繼結(jié)點(diǎn)。

![](image\image1.png)

## 定義節(jié)點(diǎn)及鏈表對(duì)象

```java

/**

* 雙鏈表

* @title DoubleLink.java

*

description

*

company: www.itheima.com

* @author 三國(guó)的包子

* @version 1.0

*

*/

public class DoubleLink {

//節(jié)點(diǎn)對(duì)象

private class Node{

Node prev;//記錄當(dāng)前節(jié)點(diǎn)的上一個(gè)節(jié)點(diǎn)的內(nèi)存地址

Node next;//記錄當(dāng)前節(jié)點(diǎn)的下一個(gè)節(jié)點(diǎn)的內(nèi)存地址

Object data;//當(dāng)前節(jié)點(diǎn)的數(shù)據(jù)

}

private Node head;//頭節(jié)點(diǎn)

private Node rear;//尾節(jié)點(diǎn)

}

```

#### 從尾部添加節(jié)點(diǎn)到鏈表

##### 分析

從尾部添加節(jié)點(diǎn),首先在鏈表中,有一個(gè)head 指向頭節(jié)點(diǎn) 有一個(gè)rear指向尾部節(jié)點(diǎn),并且如果只有一個(gè)節(jié)點(diǎn),這個(gè)節(jié)點(diǎn)既是頭節(jié)點(diǎn)也是尾節(jié)點(diǎn),如果有多個(gè)節(jié)點(diǎn),那么根據(jù)節(jié)點(diǎn)的增加,尾部節(jié)點(diǎn)(rear)指向的節(jié)點(diǎn)不同。

添加的過(guò)程:

? 1.創(chuàng)建數(shù)據(jù)節(jié)點(diǎn)

? 2.將數(shù)據(jù)放入節(jié)點(diǎn)中

? 3.判斷尾部節(jié)點(diǎn)是否為空,如果為空,說(shuō)明鏈表還是空的,此時(shí) 新增節(jié)點(diǎn)即為頭節(jié)點(diǎn)和尾節(jié)點(diǎn)

? 4.如果尾部節(jié)點(diǎn)不為空,鏈表存在,需要將新增的節(jié)點(diǎn)加入到列表中(即原尾部節(jié)點(diǎn)的后面),并移動(dòng)rear 執(zhí)行新增的節(jié)點(diǎn)。

##### 編寫(xiě)添加的代碼

```java

/**

* 從尾部添加節(jié)點(diǎn)

* @param data

*/

public void addFromRear(Object data){

// 1. 創(chuàng)建新的節(jié)點(diǎn)

Node node = new Node();

// 2. 把數(shù)據(jù)放入節(jié)點(diǎn)中

node.data = data;

// 3. 判斷尾部節(jié)點(diǎn)是否為空 如果為空說(shuō)明鏈表還是空的

if (rear == null) {

rear = node;

head = node;

} else {

// 4. 判斷如果尾部節(jié)點(diǎn)不為空,說(shuō)明 鏈表是存在的

//將新增的節(jié)點(diǎn)的內(nèi)存地址賦給 原尾節(jié)點(diǎn)的的next 屬性

rear.next = node;

//將原尾節(jié)點(diǎn)的地址 賦給 新增節(jié)點(diǎn)的 prev 屬性

node.prev = rear;

//最后 將新增的節(jié)點(diǎn) 賦給尾節(jié)點(diǎn)引用

rear = node;

}

}

```

##### 代碼實(shí)現(xiàn)的過(guò)程如下

![](image\image2.png)

## 重寫(xiě)toString

### 分析

? 添加完成節(jié)點(diǎn)之后,需要遍歷節(jié)點(diǎn)打印 查看鏈表的內(nèi)容。所以這里我們重寫(xiě)toString方法。

重寫(xiě)toString ,需要在鏈表中從頭節(jié)點(diǎn)開(kāi)始遍歷,遍歷到尾部節(jié)點(diǎn)之后將每一個(gè)節(jié)點(diǎn)的數(shù)據(jù)連接起來(lái)。這里我們實(shí)現(xiàn)一個(gè)字符串連接。

### 實(shí)現(xiàn)

?

```java

//[a,b,c]

@Override

public String toString() {

StringBuilder sbBuilder = new StringBuilder("[");

// 遍歷鏈表中所有的數(shù)據(jù)

Node node = head;// 從頭節(jié)點(diǎn)開(kāi)始遍歷數(shù)據(jù)

while (node != null) {

//如果node還沒(méi)遍歷到尾部節(jié)點(diǎn)

if (node != rear) {

//就有逗號(hào)

sbBuilder.append(node.data + ", ");

} else {

sbBuilder.append(node.data);

}

// 條件的改變:將下一個(gè)節(jié)點(diǎn)賦值給當(dāng)前node 引用。直到node.next=null 說(shuō)明已經(jīng)到尾部節(jié)點(diǎn)

node = node.next;

}

sbBuilder.append("]");

return sbBuilder.toString();

}

```

## 測(cè)試

```java

package com.itheima.link;

public class TestDoubleLink {

public static void main(String[] args) {

DoubleLink doubleLink = new DoubleLink();

doubleLink.addFromRear("abc1");

doubleLink.addFromRear("abc2");

doubleLink.addFromRear("abc3");

System.out.println(doubleLink);

}

}

```

測(cè)試效果:

```java

[abc1, abc2, abc3]

```

## 總結(jié)

? 通過(guò)添加節(jié)點(diǎn),我們發(fā)現(xiàn),雙向鏈表是 有一個(gè)前驅(qū) 和 后繼節(jié)點(diǎn)指針。這樣就可以從任意的節(jié)點(diǎn)處添加節(jié)點(diǎn)了。只需要改變prev next 即可。下一章節(jié),我們來(lái)討論如何從中間位置添加節(jié)點(diǎn)以及從頭部添加節(jié)點(diǎn)。這樣大家就可以很好對(duì)比單鏈表了。今天先入門(mén)到這里。

0 分享到:
和我們?cè)诰€交談!