每日算法題 | 劍指offer 鏈表專題 (4) 從尾到頭打印鏈表

重磅乾貨,第一時間送達c++

每日算法題 | 劍指offer 鏈表專題 (4) 從尾到頭打印鏈表

題目

從尾到頭打印鏈表程序員

題目要求

輸入一個鏈表的頭結點,從尾到頭反過來打印出每一個結點的值面試

解題思路

要想從尾到頭遍歷鏈表,首先須要作的是倒轉鏈表,再進行遍歷。
該題思路:建立一個空列表,用來存儲鏈表中的值,最後將列表逆序輸出
下面咱們來舉個例子:
每日算法題 | 劍指offer 鏈表專題 (4) 從尾到頭打印鏈表正則表達式

使用棧的狀況:

題目要求,從尾到頭遍歷單鏈表。也就是第一個遍歷到的節點要最後一個輸出,最後一個遍歷到的節點第一個輸出。這就是典型的「後進先出」,由此可藉助棧實現這種順序。
每通過一個結點的時候,把該結點放到一個棧中。
當遍歷完整個鏈表後,再從棧頂開始逐個輸出結點的值,此時輸出的結點的順序已經反轉過來了。算法

使用遞歸:

遞歸在本質上就是一個棧結構。數組

要實現反過來輸出鏈表,咱們每訪問到一個結點的時候,先遞歸輸出它後面的結點,再輸出該結點自身,這樣鏈表的輸出結果就反過來了。微信

但有個問題:
當鏈表很是長的時候,就會致使函數調用的層級很深,從而有可能致使函數調用棧溢出。數據結構

代碼實現

Python :

# -*- coding:utf-8 -*-
# class ListNode:
#     def __init__(self, x):
#         self.val = x
#         self.next = None

class Solution:
    # 返回從尾部到頭部的列表值序列,例如[1,2,3]
    # 該題思路:建立一個空列表,用來存儲鏈表中的值,最後將列表逆序輸出
    def printListFromTailToHead(self, listNode):
        # write code here
        if not listNode:
            return []

        result = []
        while listNode.next is not None:
            # extend() 函數用於在列表末尾一次性追加另外一個序列中的多個值(用新列表擴展原來的列表)。
            result.extend([listNode.val])
            listNode = listNode.next
        # 退出循環恰好在末尾節點,將末尾節點也添加進去
        result.extend([listNode.val])

        return result[::-1]

C++

c++的vector能夠用insert()函數來向指定位置插入元素,arr.begin()表示在頭部,arr.end()表示在尾部。例如:機器學習

//在頭部插入10
arr.insert(arr.begin(),10);
//在尾部插入8
arr.insert(arr.end(),8);

具體實現:

/**
*  struct ListNode {
*        int val;
*        struct ListNode *next;
*        ListNode(int x) :
*              val(x), next(NULL) {
*        }
*  };
*/
class Solution {
public:
    vector<int> printListFromTailToHead(ListNode* head) {
        vector<int> arr;
        while(head){
            arr.insert(arr.begin(),head->val);
            head = head->next;
        }
        return arr;
    }
};

JAVA

public class Solution {
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;
}
}

後記

注:面試季來了,無論是做爲面試者仍是之後做爲面試官,瞭解算法這門程序員之間的溝通方式都是很是必要的。找過工做的朋友應該都據說過《劍指offer》,雖然書中只有六十多道題目,可是道道都是經典。ide

若是是單純的面試需求,劍指offer的優先級確定是在Leetcode以前,總的說它有三個優勢:

  • 1.極可能在面試中出現原題
  • 2.約66題,題量少,可是涵蓋的內容較全
  • 3.能培養一個良好的刷題習慣

它的缺點是:

  • 1.只有66題,刷着容易過擬合
  • 2.動態規劃的題比較少,所以須要在Leetcode上專項訓練。
    算法題主要分紅數據結構和具體算法部分,簡單歸類以下。基本每道題都很精彩,因此這裏就不一一洗寫了,題解能夠看看個人代碼倉庫或者討論區的內容。

數據結構類題目

  • LinkedList
  • 003-從尾到頭打印鏈表
  • 014-鏈表中倒數第k個結點
  • 015-反轉鏈表
  • 016-合併兩個或k個有序鏈表
  • 025-複雜鏈表的複製
  • 036-兩個鏈表的第一個公共結點
  • 055-鏈表中環的入口結點
  • 056-刪除鏈表中重複的結點
  • Tree
  • 004-重建二叉樹
  • 017-樹的子結構
  • 018-二叉樹的鏡像
  • 022-從上往下打印二叉樹
  • 023-二叉搜索樹的後序遍歷序列
  • 024-二叉樹中和爲某一值的路徑
  • 026-二叉搜索樹與雙向鏈表
  • 038-二叉樹的深度
  • 039-平衡二叉樹
  • 057-二叉樹的下一個結點
  • 058-對稱的二叉樹
  • 059-按之字形順序打印二叉樹
  • 060-把二叉樹打印成多行
  • 061-序列化二叉樹
  • 062-二叉搜索樹的第k個結點
  • Stack & Queue
  • 005-用兩個棧實現隊列
  • 020-包含min函數的棧
  • 021-棧的壓入、彈出序列
  • 044-翻轉單詞順序列(棧)
  • 064-滑動窗口的最大值(雙端隊列)
  • Heap
  • 029-最小的K個數
  • Hash Table
  • 034-第一個只出現一次的字符
  • 065-矩陣中的路徑(BFS)
  • 066-機器人的運動範圍(DFS)
  • 具體算法類題目
  • 斐波那契數列
  • 007-斐波拉契數列
  • 008-跳臺階
  • 009-變態跳臺階
  • 010-矩形覆蓋
  • 搜索算法
  • 001-二維數組查找
  • 006-旋轉數組的最小數字(二分查找)
  • 037-數字在排序數組中出現的次數(二分查找)
  • 全排列
  • 027-字符串的排列
  • 動態規劃
  • 030-連續子數組的最大和
  • 052-正則表達式匹配(我用的暴力)
  • 回溯
  • 065-矩陣中的路徑(BFS)
  • 066-機器人的運動範圍(DFS)
  • 排序
  • 035-數組中的逆序對(歸併排序)
  • 029-最小的K個數(堆排序)
  • 029-最小的K個數(快速排序)
  • 位運算
  • 011-二進制中1的個數
  • 012-數值的整數次方
  • 040-數組中只出現一次的數字
  • 其餘算法
  • 002-替換空格
  • 013-調整數組順序使奇數位於偶數前面
  • 028-數組中出現次數超過一半的數字
  • 031-整數中1出現的次數(從1到n整數中1出現的次數)
  • 032-把數組排成最小的數
  • 033-醜數
  • 041-和爲S的連續正數序列(滑動窗口思想)
  • 042-和爲S的兩個數字(雙指針思想)
  • 043-左旋轉字符串(矩陣翻轉)
  • 046-孩子們的遊戲-圓圈中最後剩下的數(約瑟夫環)
  • 051-構建乘積數組

劍指offer刷題交流羣

掃碼添加微信,必定要備註研究方向+地點+學校+暱稱(如機器學習+上海+上交+湯姆)

每日算法題 | 劍指offer 鏈表專題 (4) 從尾到頭打印鏈表
▲長按加羣

每日算法題 | 劍指offer 鏈表專題 (4) 從尾到頭打印鏈表

相關文章
相關標籤/搜索