• 【Java】 剑指offer(5) 从尾到头打印链表


    本文参考自《剑指offer》一书,代码采用Java语言。

     更多:《剑指Offer》Java实现合集

    题目

      输入一个链表的头结点,从尾到头反过来打印出每个结点的值。结点定义如下:

    public class ListNode {
        int val;
        ListNode next = null;
    
        ListNode(int val) {
            this.val = val;
        }
    }

    思路

      结点遍历顺序只能从头到尾,但是输出的顺序却为从尾到头,是典型的“后进先出”问题,这就要联想到使用栈,从而也可以联想到使用递归。

    测试用例

      1.功能测试(单个结点链表,多个结点链表)

      2.特殊输入测试(链表为空)

    完整Java代码

    import java.util.Stack;
    
    /**
     * 
     * @Description 从尾到头打印链表
     *
     * @author yongh
     * @date 2018年9月10日 下午7:07:23
     */
    
    //题目:输入一个链表的头结点,从尾到头反过来打印出每个结点的值。
    
    public class PrintListInReversedOrder {
    	class ListNode{
    		int key;
    		ListNode next;
    		public ListNode(int key) {
    			this.key=key;
    			this.next=null;
    		}
    	}
    	
    	// 采用栈
    	public void printListReversingly_Iteratively(ListNode node) {
    		Stack<ListNode> stack = new Stack<ListNode>();
    		while (node!= null) {
    			stack.push(node);
    			node=node.next;
    		}
    		while(!stack.empty()) {
    			System.out.println(stack.pop().key);
    		}
    	}
    	
    	//采用递归
    	public void printListReversingly_Recursively(ListNode node) {
    		if(node!=null) {
    			printListReversingly_Recursively(node.next);
    			System.out.println(node.key);
    		}else
    			return;
    		
    	}
    	
    	
    	// ==================================测试代码==================================
    	/**
    	 * 链表为空
    	 */
    	public void test1() {
    		ListNode aListNode = null;
    		System.out.println("采用栈:");
    		printListReversingly_Iteratively(aListNode);
    		System.out.println("采用递归:");
    		printListReversingly_Recursively(aListNode);
    	}
    	
    	/**
    	 * 多个结点链表
    	 */
    	public void test2() {
    		ListNode ListNode1 = new ListNode(1);
    		ListNode ListNode2 = new ListNode(2);
    		ListNode ListNode3 = new ListNode(3);
    		ListNode ListNode4 = new ListNode(4);
    		ListNode ListNode5 = new ListNode(5);
    		ListNode1.next=ListNode2;
    		ListNode2.next=ListNode3;
    		ListNode3.next=ListNode4;
    		ListNode4.next=ListNode5;
    		System.out.println("采用栈:");
    		printListReversingly_Iteratively(ListNode1);
    		System.out.println("采用递归:");
    		printListReversingly_Recursively(ListNode1);
    	}
    	
    	/**
    	 * 单个结点链表
    	 */
    	public void test3() {
    		ListNode ListNode1 = new ListNode(1);
    		System.out.println("采用栈:");
    		printListReversingly_Iteratively(ListNode1);
    		System.out.println("采用递归:");
    		printListReversingly_Recursively(ListNode1);
    	}
    	
    	public static void main(String[] args) {
    		PrintListInReversedOrder demo = new PrintListInReversedOrder();
    		System.out.println("test1:");
    		demo.test1();
    		System.out.println("test2:");
    		demo.test2();
    		System.out.println("test3:");
    		demo.test3();
    	}
    }
    

      

    test1:
    采用栈:
    采用递归:
    test2:
    采用栈:
    5
    4
    3
    2
    1
    采用递归:
    5
    4
    3
    2
    1
    test3:
    采用栈:
    1
    采用递归:
    1
    PrintListInReversedOrder

    递归部分代码也可以像下面这样写,注意体会不同的递归写法:

    	// 采用递归
    	public void printListReversingly_Recursively(ListNode node) {
    		// if(node!=null) {
    		// printListReversingly_Recursively(node.next);
    		// System.out.println(node.key);
    		// }else
    		// return;
    		if (node != null) {
    			if (node.next != null) {
    				printListReversingly_Recursively(node.next);
    			}
    			System.out.println(node.key);
    		}
    	}
    

       

    ====================================================================

     在牛客网中提交的代码如下(参考自:grass_stars 的代码):

    public class Solution {
    	ArrayList<Integer> arrayList = new ArrayList<Integer>();
    	public ArrayList<Integer> printListFromTailToHead(ListNode listNode) {
    		if (listNode != null) {
    			this.printListFromTailToHead(listNode.next);
    			arrayList.add(listNode.val);
    		}
    		return arrayList;
    	}
    }
    

      上面代码采用的递归,非常简洁,很值得学习。

    收获

      1.对于“后进先出”问题,要快速想到”栈“,也同时想到递归。

      2.采用递归时,返回的函数值不一定要有赋值操作,只要实现了遍历的作用就可以了,上面牛客网的代码可以多多学习。

     更多:《剑指Offer》Java实现合集

  • 相关阅读:
    LeetCode 面试题56-I
    LeetCode T2
    统计中的AUC和ROC曲线
    【转载】RNN
    One layer SoftMax Classifier, "Handwriting recognition"
    【转载】深度学习中softmax交叉熵损失函数的理解
    【转载】softmax的性质及其实现
    logistics多分类
    logistics二分类
    多元线性回归
  • 原文地址:https://www.cnblogs.com/yongh/p/9622919.html
Copyright © 2020-2023  润新知