剑指offer-3、从尾到头打印链表

题目描述

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

如输入{1,2,3}的链表如下图:

返回一个数组为[3,2,1]

0 <= 链表长度 <= 10000

示例1

输入:

复制代码
{1,2,3}

返回值:

csharp 复制代码
[3,2,1]

示例2

输入:

复制代码
{67,0,24,58}

返回值:

csharp 复制代码
[58,24,0,67]

思路及解答

⾸先我们需要想⽤哪些解法可以解,⼤概有如下:

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

借助栈实现

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

java 复制代码
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;
    }
}

递归调⽤

前⾯我们能想到栈,那么我们何必⾃⼰实现呢?其实⽅法的调⽤过程,就是⼀个天然的栈调⽤的过程呀,只需要判断当前节点是不是为空,为空则不输出,不为空则递归下⼀个节点,对下⼀个节点处理之后,把结果使⽤ArrayList.addAll() 加到结果中,再把⾃身加到结果中即可:

java 复制代码
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;
    }
}

头插法

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

java 复制代码
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;
    }
}
相关推荐
程序猿小D2 分钟前
Java项目:基于SSM框架实现的校园活动资讯网管理系统【ssm+B/S架构+源码+数据库+毕业论文+远程部署】
java·数据库·mysql·spring·毕业设计·ssm框架·校园活动
阿华的代码王国7 分钟前
【Android】PopupWindow实现长按菜单
android·xml·java·前端·后端
找不到、了11 分钟前
关于MyBatis 的懒加载(Lazy Loading)机制
java·mybatis
啃火龙果的兔子40 分钟前
快速搭建Java服务指南
java·开发语言
未来之窗软件服务42 分钟前
智慧收银系统开发进销存库存统计,便利店、水果店、建材与家居行业的库存汇总管理—仙盟创梦IDE
java·开发语言·ide·进销存·仙盟创梦ide·东方仙盟·收银台
CodeCraft Studio1 小时前
国产化PDF处理控件Spire.PDF教程:Java 提取 PDF 图片,高质量提取与图片过滤技巧
java·python·pdf·国产化·文档处理·spire·pdf图片提取
枫叶落雨2222 小时前
Hutool 的 WordTree(敏感词检测)
java·开发语言
用户2018792831673 小时前
Java序列化之幽灵船“Serial号”与永生契约
android·java
用户2018792831673 小时前
“对象永生”的奇幻故事
android·java
周某某~3 小时前
Rabbit MQ的消息模式-Java原生代码
java·分布式·rabbitmq