2019-11-02 12:07:41 +08:00
|
|
|
|
# 6. 从尾到头打印链表
|
|
|
|
|
|
|
|
|
|
[NowCoder](https://www.nowcoder.com/practice/d0267f7f55b3412ba93bd35cfa8e8035?tpId=13&tqId=11156&tPage=1&rp=1&ru=/ta/coding-interviews&qru=/ta/coding-interviews/question-ranking&from=cyc_github)
|
|
|
|
|
|
|
|
|
|
## 题目描述
|
|
|
|
|
|
|
|
|
|
从尾到头反过来打印出每个结点的值。
|
|
|
|
|
|
2019-11-02 14:39:13 +08:00
|
|
|
|
<div align="center"> <img src="https://cs-notes-1256109796.cos.ap-guangzhou.myqcloud.com/f5792051-d9b2-4ca4-a234-a4a2de3d5a57.png" width="280px"> </div><br>
|
2019-11-02 12:07:41 +08:00
|
|
|
|
|
|
|
|
|
## 解题思路
|
|
|
|
|
|
|
|
|
|
### 使用递归
|
|
|
|
|
|
|
|
|
|
要逆序打印链表 1->2->3(3,2,1),可以先逆序打印链表 2->3(3,2),最后再打印第一个节点 1。而链表 2->3 可以看成一个新的链表,要逆序打印该链表可以继续使用求解函数,也就是在求解函数中调用自己,这就是递归函数。
|
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
public ArrayList<Integer> printListFromTailToHead(ListNode listNode) {
|
|
|
|
|
ArrayList<Integer> ret = new ArrayList<>();
|
|
|
|
|
if (listNode != null) {
|
|
|
|
|
ret.addAll(printListFromTailToHead(listNode.next));
|
|
|
|
|
ret.add(listNode.val);
|
|
|
|
|
}
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### 使用头插法
|
|
|
|
|
|
|
|
|
|
使用头插法可以得到一个逆序的链表。
|
|
|
|
|
|
|
|
|
|
头结点和第一个节点的区别:
|
|
|
|
|
|
|
|
|
|
- 头结点是在头插法中使用的一个额外节点,这个节点不存储值;
|
|
|
|
|
- 第一个节点就是链表的第一个真正存储值的节点。
|
|
|
|
|
|
2019-11-02 14:39:13 +08:00
|
|
|
|
<div align="center"> <img src="https://cs-notes-1256109796.cos.ap-guangzhou.myqcloud.com/0dae7e93-cfd1-4bd3-97e8-325b032b716f.gif" width="370px"> </div><br>
|
2019-11-02 12:07:41 +08:00
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
public ArrayList<Integer> printListFromTailToHead(ListNode listNode) {
|
|
|
|
|
// 头插法构建逆序链表
|
|
|
|
|
ListNode head = new ListNode(-1);
|
|
|
|
|
while (listNode != null) {
|
|
|
|
|
ListNode memo = listNode.next;
|
|
|
|
|
listNode.next = head.next;
|
|
|
|
|
head.next = listNode;
|
|
|
|
|
listNode = memo;
|
|
|
|
|
}
|
|
|
|
|
// 构建 ArrayList
|
|
|
|
|
ArrayList<Integer> ret = new ArrayList<>();
|
|
|
|
|
head = head.next;
|
|
|
|
|
while (head != null) {
|
|
|
|
|
ret.add(head.val);
|
|
|
|
|
head = head.next;
|
|
|
|
|
}
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### 使用栈
|
|
|
|
|
|
|
|
|
|
栈具有后进先出的特点,在遍历链表时将值按顺序放入栈中,最后出栈的顺序即为逆序。
|
|
|
|
|
|
2019-11-02 14:39:13 +08:00
|
|
|
|
<div align="center"> <img src="https://cs-notes-1256109796.cos.ap-guangzhou.myqcloud.com/9d1deeba-4ae1-41dc-98f4-47d85b9831bc.gif" width="300px"> </div><br>
|
2019-11-02 12:07:41 +08:00
|
|
|
|
|
|
|
|
|
```java
|
|
|
|
|
public ArrayList<Integer> printListFromTailToHead(ListNode listNode) {
|
|
|
|
|
Stack<Integer> stack = new Stack<>();
|
|
|
|
|
while (listNode != null) {
|
|
|
|
|
stack.add(listNode.val);
|
|
|
|
|
listNode = listNode.next;
|
|
|
|
|
}
|
|
|
|
|
ArrayList<Integer> ret = new ArrayList<>();
|
|
|
|
|
while (!stack.isEmpty())
|
|
|
|
|
ret.add(stack.pop());
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
```
|
2019-11-02 14:39:13 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2019-11-02 17:33:10 +08:00
|
|
|
|
<div align="center"><img width="320px" src="https://cs-notes-1256109796.cos.ap-guangzhou.myqcloud.com/githubio/公众号二维码-2.png"></img></div>
|