菜单
本页目录

题目描述

输入一个链表的头节点,按链表从尾到头的顺序返回每个节点的值(用数组返回)。比如下面的链表:

image-20211207084126054

返回的数组为 [3,2,1]

思路 & 解答

首先我们需要想用哪些解法可以解,大概有如下:

  • 使用栈
  • 使用递归调用
  • 头插法

借助栈实现

先把元素里面的元素从头到尾遍历取出放在栈里面,然后再把栈的元素去出来放在ArraList里面。主要利用了栈的先进后出的规则,这样就可以实现倒序的功能。Java 代码实现如下:

/**
*    public class ListNode {
*        int val;
*        ListNode next = null;
*
*        ListNode(int val) {
*            this.val = val;
*        }
*    }
*
*/
import java.util.ArrayList;
import java.util.Stack;
public class Solution {
    public ArrayList<Integer> printListFromTailToHead(ListNode listNode) {
        Stack<Integer> stack = new Stack<>();
        while (listNode != null) {
            stack.push(listNode.val);
            listNode = listNode.next;
        }
        ArrayList<Integer> results = new ArrayList<>();
        while (!stack.isEmpty()) {
            results.add(stack.pop()); 
        }
        return results;
    }
}

C++ 代码实现如下:

#include <iostream>
#include <vector>
#include <stack>

using namespace std;

struct ListNode {
    int val;
    struct ListNode *next;

    ListNode(int x) :
            val(x), next(NULL) {
    }
};

class Solution {
public:
    vector<int> printListFromTailToHead(ListNode *head) {
        stack<int> myStack;
        while (head != NULL) {
            myStack.push(head->val);
            head = head->next;
        }
        vector<int> results;
        while (!myStack.empty()) {
            int top = myStack.top();
            results.push_back(top);
            myStack.pop();
        }
        return results;
    }
};

递归调用

前面我们能想到栈,那么我们何必自己实现呢?其实方法的调用过程,就是一个天然的栈调用的过程呀,只需要判断当前节点是不是为空,为空则不输出,不为空则递归下一个节点,对下一个节点处理之后,把结果使用ArrayList.addAll()加到结果中,再把自身加到结果中,emmmmm有道理,说码就码:

/**
*    public class ListNode {
*        int val;
*        ListNode next = null;
*
*        ListNode(int val) {
*            this.val = val;
*        }
*    }
*
*/
import java.util.ArrayList;
import java.util.Stack;
public class Solution {
    public ArrayList<Integer> printListFromTailToHead(ListNode listNode) {
        ArrayList<Integer> results = new ArrayList<>();
        if(listNode!=null){
            // 对后面的元素进行处理
            results.addAll(printListFromTailToHead(listNode.next));
            // 最后添加自身
            results.add(listNode.val);
        }
        return results;
    }
}

C++ 实现如下:

#include <iostream>
#include <vector>
#include <stack>

using namespace std;

struct ListNode {
    int val;
    struct ListNode *next;

    ListNode(int x) :
            val(x), next(NULL) {
    }
};

class Solution {
public:
    vector<int> printListFromTailToHead(ListNode *head) {
        vector<int> results;
        if (head != NULL) {
            vector<int> nextResult = printListFromTailToHead(head->next);
            for(int r:nextResult){
                // 对后面的元素进行处理
                results.push_back(r);
            }
            // 最后添加自身
            results.push_back(head->val);
        }
        return results;
    }
};

头插法

我们遍历每一个节点,然后把它插入到头部,这样一直遍历到尾的时候,就相当于将整个链表都反转一遍了,然后再从头到尾遍历放到ArryList即可。

import java.util.ArrayList;
import java.util.Stack;
public class Solution {
    public  ArrayList<Integer> printListFromTailToHead(ListNode listNode) {
        ListNode head = new ListNode(-1);
        while(listNode!=null){
            // 先把当前node的next保存起来
            ListNode temp = listNode.next;
            // 把当前节点的next指针指向head的下一个节点
            listNode.next = head.next;
            // 把head的next指向当前节点
            head.next = listNode;
            // 将遍历的指针指向了遍历的下一个元素
            listNode = temp;
        }
        ArrayList<Integer> results = new ArrayList<>();
        head = head.next;
        // 遍历输出
        while(head!=null){
            results.add(head.val);
            head = head.next;
        }
        return results;
    }
}

C++ 代码如下:

#include <iostream>
#include <vector>
#include <stack>

using namespace std;

struct ListNode {
    int val;
    struct ListNode *next;

    ListNode(int x) :
            val(x), next(NULL) {
    }
};

class Solution {
public:
    vector<int> printListFromTailToHead(ListNode *head) {
        ListNode *pre = nullptr;
        ListNode *current = head;
        ListNode *temp = current;
        while (current) {
            temp = current->next;
            current->next = pre;
            pre = current;
            current = temp;
        }
        vector<int> results;
        while (pre) {
            results.push_back(pre->val);
            pre = pre->next;
        }
        return results;
    }
};