面试题15:链表中倒数第K个节点
2016-06-11 11:03
381 查看
题目:输入一个链表,输出该链表中倒数第K个节点。为了符合大多数人的习惯,本题从1开始计数,即链表的尾节点是倒数第一个节点。
链表的定义如下:struct ListNode{ int value; ListNode* next; ListNode(int data=int()):value(data),next(NULL) {} };
分析
有三种方法:递归法,自定义栈法和前后指针法。递归法
代码如下:ListNode* _FindKthToTail(ListNode* head,int& k) { if(head==NULL) { return NULL; } else { ListNode* tmp=_FindKthToTail(head->next,k); if(k==0) return tmp; } if(k==1) { k=0; return head; } else if(k>1) k--; } ListNode* FindKthToTail(ListNode* head,int k)throw(ListIsTooShort) { ListNode* tmp=_FindKthToTail(head,k); if(k==0) return tmp; else throw ListIsTooShort(); }
自定义栈法
ListNode* FindKthToTail(ListNode* head,int k)throw(ListIsTooShort) { if(head==NULL || k<=0) return NULL; stack<ListNode*> s; while(head!=NULL) { s.push(head); head=head->next; } while(--k!=0 && !s.empty()) { s.pop(); } if(k==0) return s.top(); else throw ListIsTooShort(); }
前后指针法
ListNode* FindKthToTail(ListNode* head,int k)throw(ListIsTooShort) { if(head==NULL || k==0) return NULL; ListNode* front=head,*last=head; while(k-->0 && front!=NULL) { front=front->next; } if(k>0) { throw ListIsTooShort(); } else { while(front!=NULL) { front=front->next; last=last->next; } } return last; }
这三种方法的优缺点如下:
时间复杂度 | 空间复杂度 | 是否最优 | |
---|---|---|---|
递归法 | O(n) | O(n) | 否 |
自定义栈法 | O(n) | O(n) | 否 |
前后指针法 | O(n) | O(1) | 是 |
以上
如果你有任何想法或是可以改进的地方,欢迎和我交流!
完整代码及测试用例在github上:点我前往
相关文章推荐
- 面试题14:调整数组顺序使奇数位于偶数前面
- 面试题13:在O(1)时间 内删除链表节点
- 面试题12:打印1到最大的n位数
- 面试题11:数值的整数次方
- 记录面试需要的一些基本资料1
- 面试总结之查找算法
- Android面试题(一)
- sevice层和dao层的简单理解
- 面试题:Iterator遍历的添加删除
- [置顶] 阻碍新手程序员提升的8件小事
- 黑马程序员——Java基础---包、内部类、匿名内部类
- 假如程序员生活在童话里…
- 程序员编程的 7 + 1 条小贴士
- 一个 fork 的面试题
- java多线程常见面试题
- 《程序员修炼之道--从小工到专家》阅读笔记02
- 剑指offer面试题4 替换空格
- 面试:哈希:最长不同字符子串
- 程序员接私活经验谈
- 2016面试——智力逻辑