【剑指offer】面试题五:从尾到头打印链表

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

链表结点定义如下:

typedef struct Node
{
    int m_nKey;
    struct Node *m_pNext; 
}ListNode, *pListNode;

我们可能会想到把链表结点的指针反转过来,改变链表的方向,然后就可以从头到尾输出了。但是这种方法改变了链表原来的结构。

但是,一般的如打印这样的只读操作,我们一般不会去改变其原来的结构。

 

而如果我们要想解决这个问题,就必须要遍历链表。遍历的顺序是从头到尾,可输出的顺序却是从尾到头。

也就是说第一个遍历到的元素最后一个输出;而最后一个遍历到的元素第一个输出。显然我们可以利用栈“先进后出”的规律来帮助我们解决这个问题。

具体步骤:

1、从头到尾遍历元素,同时将其放进栈中;

2、弹出栈顶元素,直到栈为空。

在借助栈的情况下,我们不难写出如下代码:

 1 // printListReversely.cpp
 2 #include "stdio.h"
 3 #include "stdlib.h"
 4 #include <stack>
 5 #define N 20
 6 
 7 typedef struct Node
 8 {
 9     int m_nKey;
10     struct Node *m_pNext;
11 }ListNode;
12 
13 void printList(ListNode *pHead)
14 {
15     while(pHead != NULL)
16     {
17         printf("%3d", pHead->m_nKey);
18         pHead = pHead->m_pNext;
19     }
20     printf("\n");
21 }
22 
23 void createList(ListNode **pHead, int len)
24 {
25     if(len < 0)
26         return;
27 
28     ListNode *pTail = NULL;
29 
30     while(len > 0)
31     {
32         ListNode *pNew = (ListNode*)malloc(sizeof(ListNode));
33         pNew->m_pNext = NULL;
34         pNew->m_nKey = rand()%100;
35 
36         if(*pHead == NULL)
37         {
38             *pHead = pNew;
39             pTail = pNew;
40         }
41         else
42         {
43             pTail->m_pNext = pNew;
44             pTail = pNew;
45         }
46         len--;
47     }
48 }
49 
50 void printListReversely(ListNode *pHead)
51 {
52     std::stack<ListNode*> st;
53 
54     while(pHead != NULL)
55     {
56         st.push(pHead);
57         pHead = pHead->m_pNext;
58     }
59 
60     while(!st.empty())
61     {
62         printf("%3d", (st.top())->m_nKey);
63         st.pop();
64     }
65     printf("\n");
66 }
67 
68 
69 int main(int argc, char const *argv[])
70 {
71     ListNode *pHead = NULL;
72 
73     createList(&pHead, N);
74     printf("Before:\n");
75     printList(pHead);
76 
77     printf("After:\n");
78     printListReversely(pHead);
79 
80     return 0;
81 }
View Code
注:C中并没有Stack的库,如果坚持用C,可自行实现stack的push、pop操作,这里我们用的是.cpp文件。

编译与执行:

1 g++ -o printListReversely printListReversely.cpp
2 ./printListReversely

 

本文完。

posted @ 2015-06-13 00:18  Stephen_Hsu  阅读(380)  评论(0编辑  收藏  举报