从头到尾打印链表

从尾到头打印链表

http://www.nowcoder.com/questionTerminal/d0267f7f55b3412ba93bd35cfa8e8035

题目的主要信息:
  • 输入一个链表的头节点,按链表从尾到头的顺序返回每个节点的值
  • 返回值保存在数组中
举一反三:

学习完本题的思路你可以解决如下题目:

JZ24. 反转链表

方法一:递归(推荐使用)

知识点:递归

递归是一个过程或函数在其定义或说明中有直接或间接调用自身的一种方法,它通常把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解。因此递归过程,最重要的就是查看能不能讲原本的问题分解为更小的子问题,这是使用递归的关键。

思路:

我们都知道链表无法逆序访问,那肯定无法直接遍历链表得到从尾到头的逆序结果。但是我们都知道递归是到达底层后才会往上回溯,因此我们可以考虑递归遍历链表,因此三段式如下:

  • 终止条件: 递归进入链表尾,即节点为空节点时结束递归。
  • 返回值: 每次返回子问题之后的全部输出。
  • 本级任务: 每级子任务递归地进入下一级,等下一级的子问题输出数组返回时,将自己的节点值添加在数组末尾。

具体做法:

  • step 1:从表头开始往后递归进入每一个节点。
  • step 2:遇到尾节点后开始返回,每次返回依次添加一个值进入输出数组。
  • step 3:直到递归返回表头。

Java实现代码:

import java.util.ArrayList;
public class Solution {
    //递归函数
    public void recursion(ListNode head, ArrayList<Integer> res){ 
        if(head != null){
            //先往链表深处遍历
            recursion(head.next, res); 
            //再填充到数组就是逆序
            res.add(head.val); 
        }
    }
    public ArrayList<Integer> printListFromTailToHead(ListNode listNode) {
        ArrayList<Integer> res = new ArrayList<Integer>();
        //递归函数解决
        recursion(listNode, res);
        return res;
    }
}

C++实现代码:

class Solution {
public:
    //递归函数
    void recursion(ListNode* head, vector<int>& res){ 
        if(head != NULL){
            //先往链表深处遍历
            recursion(head->next, res); 
            //再填充到数组就是逆序
            res.push_back(head->val); 
        }
    }
    vector<int> printListFromTailToHead(ListNode* head) {
        vector<int> res;
        //递归函数打印
        recursion(head, res); 
        return res;
    }
};

Python代码实现:

import sys
#设置递归深度
sys.setrecursionlimit(100000) 
class Solution:
    def recursion(self, head: ListNode, res: List[int]):
        if head:
            #先往链表深处遍历
            self.recursion(head.next, res) 
            #再填充到数组就是逆序
            res.append(head.val) 
    def printListFromTailToHead(self , listNode: ListNode) -> List[int]:
        res = []
        #递归函数打印
        self.recursion(listNode, res) 
        return res

复杂度分析:

  • 时间复杂度:O(n)O(n),其中nn为链表长度,递归遍历一次链表
  • 空间复杂度:O(n)O(n),递归栈的最大空间为链表长度
方法二:栈(扩展思路)

知识点:栈

栈是一种仅支持在表尾进行插入和删除操作的线性表,这一端被称为栈顶,另一端被称为栈底。元素入栈指的是把新元素放到栈顶元素的上面,使之成为新的栈顶元素;元素出栈指的是从一个栈删除元素又称作出栈或退栈,它是把栈顶元素删除掉,使其相邻的元素成为新的栈顶元素。

思路:

递归的思想也可以用栈实现,因为栈是先进后出的,符合逆序的特点,递归本质上就是用栈实现的。

具体做法:

  • step 1:我们可以顺序遍历链表,将链表的值push到栈中。
  • step 2:然后再依次弹出栈中的元素,加入到数组中,即可实现链表逆序。

图示:

图片说明

Java实现代码:

import java.util.*;
public class Solution {
    public ArrayList<Integer> printListFromTailToHead(ListNode listNode) {
        ArrayList<Integer> res = new ArrayList<Integer>();
        Stack<Integer> s = new Stack<Integer>();
        //正序输出链表到栈中
        while(listNode != null){ 
            s.push(listNode.val);
            listNode = listNode.next;
        }
        //输出栈中元素到数组中
        while(!s.isEmpty()) 
            res.add(s.pop());
        return res;
    }
}

C++实现代码:

class Solution {
public:
    vector<int> printListFromTailToHead(ListNode* head) {
        vector<int> res;
        stack<int> s;
        //正序输出链表到栈中
        while(head != NULL){ 
            s.push(head->val);
            head = head->next;
        }
        //输出栈中元素到数组中
        while(!s.empty()){ 
            res.push_back(s.top());
            s.pop();
        }
        return res;
    }
};

Python实现代码:

class Solution:
    def printListFromTailToHead(self , listNode: ListNode) -> List[int]:
        res = []
        s = []
        #正序输出链表到栈中
        while listNode: 
            s.append(listNode.val)
            listNode = listNode.next
        #输出栈中元素到数组中
        while len(s): 
            res.append(s[-1])
            s.pop()
        return res

复杂度分析:

  • 时间复杂度:O(n)O(n),遍历链表是一个O(n)O(n),弹空一个栈需要O(n)O(n)
  • 空间复杂度:O(n)O(n),栈空间最大长度是链表的长度nn
全部评论
第三个 LinkNode *temp = cur;
4 回复 分享
发布于 2020-09-21 00:48
方法三 第四行:应该是ListNode* temp = cur; 吧,过不了编译=_=
1 回复 分享
发布于 2021-03-05 10:55
递归爆栈了呀,哥
2 回复 分享
发布于 2021-10-21 21:14
递归不会内存溢出的吗
1 回复 分享
发布于 2021-08-18 14:48
class Solution: def printListFromTailToHead(self , listNode: ListNode) -> List[int]: arrary_list = [] while listNode: arrary_list.append(listNode.val) listNode = listNode.next return arrary_list[::-1]
点赞 回复 分享
发布于 2024-06-02 09:32 陕西
C++递归不会爆栈,给的链也不算太长,爆了的话可能是你递归方法的if用成了while
点赞 回复 分享
发布于 2023-09-13 13:54 重庆
为什么递归可以不用return啊
点赞 回复 分享
发布于 2023-02-23 22:38 上海
直接inser(0, cur.val)啊,反转也浪费内存
点赞 回复 分享
发布于 2022-08-10 23:37
反转还得再反转回来吧
点赞 回复 分享
发布于 2021-11-09 22:50
递归的方法为什么不用翻转数组呢?
点赞 回复 分享
发布于 2021-08-09 17:44
ListNode* temp = cur; 是的,应该是指针
点赞 回复 分享
发布于 2021-06-12 10:43
vector<int> ret的声明是不是应该放在外面啊</int>
点赞 回复 分享
发布于 2020-08-14 19:22
为什么递归版本不用反转呢?
点赞 回复 分享
发布于 2020-08-13 11:54

相关推荐

不愿透露姓名的神秘牛友
07-15 17:24
点赞 评论 收藏
分享
评论
171
26
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务