首先统一链表的数据结构为:

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

题目一:从尾到头打印链表:输入一个链表。从尾到头打印链表每一个节点的值

分析:

难点在于链表仅仅有指向后继的指针,没有指向前驱的指针。

转换思路。结合栈后进先出的特点,能够遍历链表,依次将数据元素存入栈中,然后再依次出栈,即为从尾到头的顺序。

vector<int> printListFromTailToHead(struct ListNode* head)
{
ListNode *p=head;
stack<int> temp;
while(p)
{
temp.push(p->val);
p=p->next;
}
vector<int>result;
while(!temp.empty())
{
result.push_back(temp.top());
temp.pop();
}
return result;
}

题目二:链表中倒数第k个结点:输入一个链表,输出该链表中倒数第k个结点。

分析:

(1)依据上题的启示。事实上这个题也能够借助栈来完毕。先从头到尾依次将结点存入栈。然后取出从栈顶開始的第k个结点就可以。

(2)还有一种方法是使用先后指针来完毕,一个指针先从头開始向前走k-1步,然后还有一个指针从头開始走。当第一个指针指向最后一个     结点时,后一个指针指向倒数第k个结点。

边界条件:要记得考虑k大于链表长度的情况和k=0的情况都返回空。

方法一:

ListNode* FindKthToTail(ListNode* pListHead, unsigned int k)
{
ListNode *p=pListHead;
stack<ListNode*> temp;
int len=0;
while(p)
{
++len;
temp.push(p);
p=p->next;
}
if(len<k||k==0)
return NULL;
while(k!=1)
{
temp.pop();
--k;
}
return temp.top();
}

方法二:

ListNode* FindKthToTail(ListNode* pListHead, unsigned int k)
{
if(pListHead==NULL||k==0)
return NULL;
ListNode *pAhead=pListHead;
ListNode *pBehind=pListHead;
for(int i=0;i<k-1;i++)
{
if(pAhead->next!=NULL)
pAhead=pAhead->next;
else
return NULL;
}
while(pAhead->next!=NULL)
{
pAhead=pAhead->next;
pBehind=pBehind->next;
}
return pBehind;
}

题目三:反转链表(链表逆序):输入一个链表,反转链表后,输出链表的全部元素。

ListNode* ReverseList(ListNode* pHead)
{
if(!pHead)
return pHead;
ListNode *reverse=NULL;
ListNode *pre=NULL;
ListNode *next=NULL;
ListNode *curr=pHead;
while(curr)
{
next=curr->next;
if(!next)
reverse=curr;
curr->next=pre;
pre=curr;
curr=next;
}
return reverse;
}

题目四:合并两个排序的链表:输入两个单调递增的链表。输出两个链表合成后的链表,当然我们须要合成后的链表满足单调不减规则。

ListNode* Merge(ListNode* pHead1, ListNode* pHead2)
{
ListNode *result=new ListNode(0);
ListNode *r=result;
while(pHead1&&pHead2)
{
if(pHead1->val<=pHead2->val)
{
r->next=pHead1;
r=r->next;
pHead1=pHead1->next; }
else
{
r->next=pHead2;
r=r->next;
pHead2=pHead2->next;
}
}
if(pHead1)
r->next=pHead1;
if(pHead2)
r->next=pHead2;
return result->next;
}

题目五:两个链表的第一个公共结点,输入两个链表,找出它们的第一个公共结点。

分析:两个链表都是单向链表。假设他们有公共的结点,那么这两个链表从某一结点開始,他们的next都指向同一个结点,之后全部的点都重合。不可能再出现分叉。所以它们的拓扑形看起来像一个Y形,而不可能是X形。

方法一:首先遍历两个链表得到它们的长度,就能知道哪个链表长,以及长的链表比短的链表多几个结点。在第二次遍历的时候,在较长的链表上先走相差的步数,接着同一时候在两个链表上遍历。找到的第一个同样的结点就是它们的公共结点。

时间复杂度O(m+n)。不须要辅助栈。

方法二:分别将两个链表存入两个辅助栈中,然后比較两个栈顶的结点是否同样。假设同样,则把栈顶弹出,接着比較下一个栈顶,直到找到最后一个同样的结点。

时间复杂度O(m+n),空间复杂度O(m+n)。

方法一:

ListNode* FindFirstCommonNode( ListNode *pHead1, ListNode *pHead2)
{
int len1=0,len2=0;
ListNode *p=pHead1,*q=pHead2;
while(p)
{
len1++;
p=p->next;
}
while(q)
{
len2++;
q=q->next;
} if(len1==0||len2==0)
return NULL;
p=pHead1;q=pHead2;
while(len1>len2)
{
p=p->next;
len1--;
}
while(len1<len2)
{
q=q->next;
len2--;
}
while(p!=q)
{
p=p->next;
q=q->next;
}
return p;
}

方法二:

 ListNode* FindFirstCommonNode( ListNode *pHead1, ListNode *pHead2)
{
ListNode *p=pHead1;
ListNode *q=pHead2;
stack<ListNode *> temp1;
stack<ListNode *> temp2;
while(p)
{
temp1.push(p);
p=p->next;
}
while(q)
{
temp2.push(q);
q=q->next;
}
ListNode *result=NULL;
while(!temp1.empty()&&!temp2.empty()&&temp1.top()==temp2.top())
{
result=temp1.top();
temp1.pop();
temp2.pop();
}
return result;
}

题目六:链表中环的入口结点:一个链表中包括环,请找出该链表的环的入口结点。

分析:有两个能够面试的问题:一个题是推断一个链表中。是否有环。

第二个是环的入口结点。

经典方法就是使用快慢指针。快的一次走两步,慢的一次走一步,假设指针重合,说明链表有环。

在此基础上,能够想到,快的比慢的刚好多走了一个环的长度。并且速度是慢的二倍,说明快的总共走的是两个环的长度。慢的总共走了一个环的长度。

所以保持慢指针如今的位置,让快指针再次从头走起。每次走一步,当这次两个指针重合的时候。它们刚好都在环的入口结点上。

推断是否有环的代码:

bool HasLoop(ListNode* pHead)
{
ListNode *slow=pHead,*fast=pHead;
while(fast&&fast->next)
{
slow=slow->next;
fast=fast->next->next;
if(slow==fast)
return true;
}
return false;
}

找环的入口结点代码

ListNode* EntryNodeOfLoop(ListNode* pHead)
{
ListNode *slow=pHead,*fast=pHead;
while(fast&&fast->next)
{
slow=slow->next;
fast=fast->next->next;
if(slow==fast)
{
fast=pHead;
while(fast!=slow)
{
fast=fast->next;
slow=slow->next;
}
return slow;
}
}
return NULL;
}

转载请注明出处:http://blog.csdn.net/xingyanxiao/article/details/47068509

剑指offer——链表相关问题总结的更多相关文章

  1. 剑指Offer 链表中倒数第k个结点

    题目描述 输入一个链表,输出该链表中倒数第k个结点.     思路: 法1:设置2个指针p,q.p先移动k次,然后pq同时后移,p到链表尾尾的时候,q指向倒数第k个节点. 注意://需要考虑k=0,以 ...

  2. 剑指offer——链表中倒数第k个结点

    输入一个链表,输出该链表中倒数第k个结点. class Solution { public: ListNode* FindKthToTail(ListNode* pListHead, unsigned ...

  3. 剑指Offer——链表中环的入口结点

    题目描述: 一个链表中包含环,请找出该链表的环的入口结点. 分析: 设置两个指针p1,p2, 两个指针都从链表的头部开始走,不过p1每次走一步,p2每次走两步. 直到相遇的时候,p2走的长度是p1的两 ...

  4. 剑指Offer——链表中倒数第k个节点

    Question 输入一个链表,输出该链表中倒数第k个结点. Solution 一种想法就是扫描两边,第一遍求出总的节点个数,第二遍从头开始走n-k个 第二种思想类似于fast-slow指针的方法,f ...

  5. python剑指offer 链表中环的入口节点

    题目: 一个链表中包含环,请找出该链表的环的入口结点. 思路: 先说个定理:两个指针一个fast.一个slow同时从一个链表的头部出发, fast一次走2步,slow一次走一步,如果该链表有环,两个指 ...

  6. acwing 70-72 剑指OFFER 二叉树相关

    地址 https://www.acwing.com/problem/content/66/ https://www.acwing.com/problem/content/67/ https://www ...

  7. 用js刷剑指offer(链表中倒数第k个结点)

    题目描述 输入一个链表,输出该链表中倒数第k个结点. 牛客网链接 思路 设置两个指针,p,q,先让p走k-1步,然后再一起走,直到p为最后一个 时,q即为倒数第k个节点 js代码 // 空间复杂度1 ...

  8. 剑指offer 链表中环的入口位置

    题目描述 一个链表中包含环,请找出该链表的环的入口结点.   思路:这题需要知道a = c,然后head和slow每次走一步,相遇的时候就是第一个入口交点, 注意:for循环或者while循环之后,一 ...

  9. 剑指offer 链表中倒数第K个节点

    利用两个间隔为k的指针来实现倒数第k个 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 ...

随机推荐

  1. POJ2187 旋转卡壳 求最长直径

    给定平面上的一些散点集,求最远两点距离的平方值. 题解: 旋转卡壳求出凸包,然后根据单调性,求出最远两点的最大距离 #pragma GCC optimize(2) #pragma G++ optimi ...

  2. 我们曾经心碎的C#之 第一章.我的第一个C#程序

    第一章.      C#入门 1.1        .NET与C#            001..NET是Microsoft.NET的简称,是基于Windows平台的一种技术            ...

  3. [ CodeVS冲杯之路 ] P1214

    不充钱,你怎么AC? 题目:http://codevs.cn/problem/1214/ 这道题类似于最长区间覆盖,仅仅是将最长区间改成了最多线段,我们贪心即可 先将线段直接右边-1,然后按左边为第一 ...

  4. centos 下文件夹共享

    [root@localhost share]# yum install samba -y[root@localhost share]# cp /etc/samba/smb.conf /etc/samb ...

  5. BCB中选择文件对话框TOpenDialog过滤后缀名使用方法

    BCB中使用TOpenDialog选择对话框时,直接OpenDialog->Execute()弹出的对话框是显示所有文件的,如果我们希望过滤指定的文件后缀名就需要在Execute()前做一些初始 ...

  6. openstack token

    在openstack的token中,包含如下: 'domain', {'id': None, 'name': None}'expires', datetime.datetime(2017, 8, 9, ...

  7. Python模块学习:glob 文件路径查找

    glob模块是最简单的模块之一,内容非常少. 用它可以查找符合特定规则的文件路径名.跟使用windows下的文件搜索差不多. 查找文件只用到三个匹配符:”*”, “?”, “[]”. ”*”匹配0个或 ...

  8. HDU 2544.最短路-最短路(Dijkstra)

    本来不想写,但是脑子不好使,还是写一下备忘_(:з」∠)_ 最短路 Time Limit: 5000/1000 MS (Java/Others)    Memory Limit: 32768/3276 ...

  9. 洛谷 P1048 采药【裸01背包】

    题目描述 辰辰是个天资聪颖的孩子,他的梦想是成为世界上最伟大的医师.为此,他想拜附近最有威望的医师为师.医师为了判断他的资质,给他出了一个难题.医师把他带到一个到处都是草药的山洞里对他说:" ...

  10. HDU 6270 Marriage (2017 CCPC 杭州赛区 G题,生成函数 + 容斥 + 分治NTT)

    题目链接  2017 CCPC Hangzhou Problem G 题意描述很清晰. 考虑每个家庭有且仅有$k$对近亲的方案数: $C(a, k) * C(b, k) * k!$ 那么如果在第$1$ ...