面试大总结:Java搞定面试中的链表题目总结
package LinkedListSummary; import java.util.HashMap; import java.util.Stack; /** * http://blog.csdn.net/luckyxiaoqiang/article/details/7393134 轻松搞定面试中的链表题目 * http://www.cnblogs.com/jax/archive/2009/12/11/1621504.html 算法大全(1)单链表 * * 目录: * 1. 求单链表中结点的个数: getListLength * 2. 将单链表反转: reverseList(遍历),reverseListRec(递归) * 3. 查找单链表中的倒数第K个结点(k > 0): reGetKthNode * 4. 查找单链表的中间结点: getMiddleNode * 5. 从尾到头打印单链表: reversePrintListStack,reversePrintListRec(递归) * 6. 已知两个单链表pHead1 和pHead2 各自有序,把它们合并成一个链表依然有序: mergeSortedList, mergeSortedListRec * 7. 判断一个单链表中是否有环: hasCycle * 8. 判断两个单链表是否相交: isIntersect * 9. 求两个单链表相交的第一个节点: getFirstCommonNode * 10. 已知一个单链表中存在环,求进入环中的第一个节点: getFirstNodeInCycle, getFirstNodeInCycleHashMap * 11. 给出一单链表头指针pHead和一节点指针pToBeDeleted,O(1)时间复杂度删除节点pToBeDeleted: delete * */ public class Demo { public static void main(String[] args) { Node n1 = new Node(1); Node n2 = new Node(2); Node n3 = new Node(3); Node n4 = new Node(4); Node n5 = new Node(5); n1.next = n2; n2.next = n3; n3.next = n4; n4.next = n5; printList(n1); // System.out.println(getListLength(n1)); // Node head = reverseList(n1); // Node head = reverseListRec(n1); // printList(head); Node x = reGetKthNode(n1, 2); System.out.println(x.val); reGetKthNodeRec(n1, 2); // x = getMiddleNode(head); // System.out.println(x.val); // System.out.println("reversePrintListStack:"); // reversePrintListStack(head); // System.out.println("reversePrintListRec:"); // reversePrintListRec(head); } // public static void main(String[] args) { // Node n1 = new Node(1); // Node n2 = new Node(3); // Node n3 = new Node(5); // n1.next = n2; // n2.next = n3; // // Node m1 = new Node(1); // Node m2 = new Node(4); // Node m3 = new Node(6); // m1.next = m2; // m2.next = m3; // // // Node ret = mergeSortedList(n1, m1); // printList(ret); // } private static class Node { int val; Node next; public Node(int val) { this.val = val; } } public static void printList(Node head) { while (head != null) { System.out.print(head.val + " "); head = head.next; } System.out.println(); } // 求单链表中结点的个数 // 注意检查链表是否为空。时间复杂度为O(n) public static int getListLength(Node head) { // 注意头结点为空情况 if (head == null) { return 0; } int len = 0; Node cur = head; while (cur != null) { len++; cur = cur.next; } return len; } // 翻转链表(遍历) // 从头到尾遍历原链表,每遍历一个结点, // 将其摘下放在新链表的最前端。 // 注意链表为空和只有一个结点的情况。时间复杂度为O(n) public static Node reverseList(Node head) { // 如果链表为空或只有一个节点,无需反转,直接返回原链表表头 if (head == null || head.next == null) { return head; } Node reHead = null; // 反转后新链表指针 Node cur = head; while (cur != null) { Node preCur = cur; // 用preCur保存住对要处理节点的引用 cur = cur.next; // cur更新到下一个节点 preCur.next = reHead; // 更新要处理节点的next引用 reHead = preCur; // reHead指向要处理节点的前一个节点 } return reHead; } // 翻转递归(递归) // 递归的精髓在于你就默认reverseListRec已经成功帮你解决了子问题了!但别去想如何解决的 // 现在只要处理当前node和子问题之间的关系。最后就能圆满解决整个问题。 /* head 1 -> 2 -> 3 -> 4 head 1-------------- | 4 -> 3 -> 2 // Node reHead = reverseListRec(head.next); reHead head.next 4 -> 3 -> 2 -> 1 // head.next.next = head; reHead 4 -> 3 -> 2 -> 1 -> null // head.next = null; reHead */ public static Node reverseListRec(Node head){ if(head == null || head.next == null){ return head; } Node reHead = reverseListRec(head.next); head.next.next = head; // 把head接在reHead串的最后一个后面 head.next = null; // 防止循环链表 return reHead; } /** * 查找单链表中的倒数第K个结点(k > 0) * 最普遍的方法是,先统计单链表中结点的个数,然后再找到第(n-k)个结点。注意链表为空,k为0,k为1,k大于链表中节点个数时的情况 * 。时间复杂度为O(n)。代码略。 这里主要讲一下另一个思路,这种思路在其他题目中也会有应用。 * 主要思路就是使用两个指针,先让前面的指针走到正向第k个结点 * ,这样前后两个指针的距离差是k-1,之后前后两个指针一起向前走,前面的指针走到最后一个结点时,后面指针所指结点就是倒数第k个结点 */ public static Node reGetKthNode(Node head, int k) { // 这里k的计数是从1开始,若k为0或链表为空返回null if (k == 0 || head == null) { return null; } Node q = head; // q在p前面 p--q Node p = head; // p在q后面 // 让q领先p距离k while (k > 1 && q != null) { q = q.next; k--; } // 当节点数小于k,返回null if (k > 1 || q == null) { return null; } // 前后两个指针一起走,直到前面的指针指向最后一个节点 while (q.next != null) { p = p.next; q = q.next; } // 当前面的指针指向最后一个节点时,后面的指针指向倒数k个节点 return p; } /** * 递归打印出倒数第k位的值 * @param head * @param dist */ static int level = 0; public static void reGetKthNodeRec(Node head, int k) { if(head == null){ return; } if(k == 1){ return; } reGetKthNodeRec(head.next, k); level++; if(level == k) { System.out.println(head.val); } } // 查找单链表的中间结点 /** * 此题可应用于上一题类似的思想。也是设置两个指针,只不过这里是,两个指针同时向前走,前面的指针每次走两步,后面的指针每次走一步, * 前面的指针走到最后一个结点时,后面的指针所指结点就是中间结点,即第(n/2+1)个结点。注意链表为空,链表结点个数为1和2的情况。时间复杂度O(n */ public static Node getMiddleNode(Node head) { if (head == null || head.next == null) { return head; } Node q = head; // p---q Node p = head; // 前面指针每次走两步,直到指向最后一个结点,后面指针每次走一步 while (q.next != null) { q = q.next; p = p.next; if (q.next != null) { q = q.next; } } return p; } /** * 从尾到头打印单链表 * 对于这种颠倒顺序的问题,我们应该就会想到栈,后进先出。所以,这一题要么自己使用栈,要么让系统使用栈,也就是递归。注意链表为空的情况 * 。时间复杂度为O(n) */ public static void reversePrintListStack(Node head) { Stack<Node> s = new Stack<Node>(); Node cur = head; while (cur != null) { s.push(cur); cur = cur.next; } while (!s.empty()) { cur = s.pop(); System.out.print(cur.val + " "); } System.out.println(); } /** * 从尾到头打印链表,使用递归(优雅!) */ public static void reversePrintListRec(Node head) { if (head == null) { return; } else { reversePrintListRec(head.next); System.out.print(head.val + " "); } } /** * 已知两个单链表pHead1 和pHead2 各自有序,把它们合并成一个链表依然有序 * 这个类似归并排序。尤其注意两个链表都为空,和其中一个为空时的情况。只需要O(1)的空间。时间复杂度为O(max(len1, len2)) */ public static Node mergeSortedList(Node head1, Node head2) { // 其中一个链表为空的情况,直接返回另一个链表头,O(1) if (head1 == null) { return head2; } if (head2 == null) { return head1; } Node mergeHead = null; // 先确定下来mergeHead是在哪里 if (head1.val < head2.val) { mergeHead = head1; head1 = head1.next; // 跳过已经合并了的元素 mergeHead.next = null; // 断开mergeHead和后面的联系 } else { mergeHead = head2; head2 = head2.next; mergeHead.next = null; } Node mergeCur = mergeHead; while (head1 != null && head2 != null) { if (head1.val < head2.val) { mergeCur.next = head1; // 把找到较小的元素合并到merge中 head1 = head1.next; // 跳过已经合并了的元素 mergeCur = mergeCur.next; // 找到下一个准备合并的元素 mergeCur.next = null; // 断开mergeCur和后面的联系 } else { mergeCur.next = head2; head2 = head2.next; mergeCur = mergeCur.next; mergeCur.next = null; } } // 合并剩余的元素 if (head1 != null) { mergeCur.next = head1; } else if (head2 != null) { mergeCur.next = head2; } return mergeHead; } /** * 递归合并两链表(优雅!) */ public static Node mergeSortedListRec(Node head1, Node head2) { if (head1 == null) { return head2; } if (head2 == null) { return head1; } Node mergeHead = null; if (head1.val < head2.val) { mergeHead = head1; // 连接已解决的子问题 mergeHead.next = mergeSortedListRec(head1.next, head2); } else { mergeHead = head2; mergeHead.next = mergeSortedListRec(head1, head2.next); } return mergeHead; } /** * 判断一个单链表中是否有环 * 这里也是用到两个指针。如果一个链表中有环,也就是说用一个指针去遍历,是永远走不到头的。因此,我们可以用两个指针去遍历,一个指针一次走两步 * ,一个指针一次走一步,如果有环,两个指针肯定会在环中相遇。时间复杂度为O(n) */ public static boolean hasCycle(Node head) { Node fast = head; // 快指针每次前进两步 Node slow = head; // 慢指针每次前进一步 while (fast != null && fast.next != null) { fast = fast.next.next; slow = slow.next; if (fast == slow) { // 相遇,存在环 return true; } } return false; } // 判断两个单链表是否相交 /** * 如果两个链表相交于某一节点,那么在这个相交节点之后的所有节点都是两个链表所共有的。 也就是说,如果两个链表相交,那么最后一个节点肯定是共有的。 * 先遍历第一个链表,记住最后一个节点,然后遍历第二个链表, 到最后一个节点时和第一个链表的最后一个节点做比较,如果相同,则相交, * 否则不相交。时间复杂度为O(len1+len2),因为只需要一个额外指针保存最后一个节点地址, 空间复杂度为O(1) */ public static boolean isIntersect(Node head1, Node head2) { if (head1 == null || head2 == null) { return false; } Node tail1 = head1; // 找到链表1的最后一个节点 while (tail1.next != null) { tail1 = tail1.next; } Node tail2 = head2; // 找到链表2的最后一个节点 while (tail2.next != null) { tail2 = tail2.next; } return tail1 == tail2; } /** * 求两个单链表相交的第一个节点 对第一个链表遍历,计算长度len1,同时保存最后一个节点的地址。 * 对第二个链表遍历,计算长度len2,同时检查最后一个节点是否和第一个链表的最后一个节点相同,若不相同,不相交,结束。 * 两个链表均从头节点开始,假设len1大于len2 * ,那么将第一个链表先遍历len1-len2个节点,此时两个链表当前节点到第一个相交节点的距离就相等了,然后一起向后遍历,直到两个节点的地址相同。 * 时间复杂度,O(len1+len2) * * ---- len2 * |__________ * | * --------- len1 * |---|<- len1-len2 */ public static Node getFirstCommonNode(Node head1, Node head2) { if (head1 == null || head2 == null) { return null; } int len1 = 1; Node tail1 = head1; while (tail1.next != null) { tail1 = tail1.next; len1++; } int len2 = 1; Node tail2 = head2; while (tail2.next != null) { tail2 = tail2.next; len2++; } // 不相交直接返回NULL if (tail1 != tail2) { return null; } Node n1 = head1; Node n2 = head2; // 略过较长链表多余的部分 if (len1 > len2) { int k = len1 - len2; while (k != 0) { n1 = n1.next; k--; } } else { int k = len2 - len1; while (k != 0) { n2 = n2.next; k--; } } // 一起向后遍历,直到找到交点 while (n1 != n2) { n1 = n1.next; n2 = n2.next; } return n1; } /** * 求进入环中的第一个节点 用快慢指针做(本题用了Crack the Coding Interview的解法,因为更简洁易懂!) */ public static Node getFirstNodeInCycle(Node head) { Node slow = head; Node fast = head; // 1) 找到快慢指针相遇点 while (fast != null && fast.next != null) { slow = slow.next; fast = fast.next.next; if (slow == fast) { // Collision break; } } // 错误检查,这是没有环的情况 if (fast == null || fast.next == null) { return null; } // 2)现在,相遇点离环的开始处的距离等于链表头到环开始处的距离, // 这样,我们把慢指针放在链表头,快指针保持在相遇点,然后 // 同速度前进,再次相遇点就是环的开始处! slow = head; while (slow != fast) { slow = slow.next; fast = fast.next; } // 再次相遇点就是环的开始处 return fast; } /** * 求进入环中的第一个节点 用HashMap做 一个无环的链表,它每个结点的地址都是不一样的。 * 但如果有环,指针沿着链表移动,那这个指针最终会指向一个已经出现过的地址 以地址为哈希表的键值,每出现一个地址,就将该键值对应的实值置为true。 * 那么当某个键值对应的实值已经为true时,说明这个地址之前已经出现过了, 直接返回它就OK了 */ public static Node getFirstNodeInCycleHashMap(Node head) { HashMap<Node, Boolean> map = new HashMap<Node, Boolean>(); while (head != null) { if (map.get(head) == true) { return head; // 这个地址之前已经出现过了,就是环的开始处 } else { map.put(head, true); head = head.next; } } return head; } /** * 给出一单链表头指针head和一节点指针toBeDeleted,O(1)时间复杂度删除节点tBeDeleted * 对于删除节点,我们普通的思路就是让该节点的前一个节点指向该节点的下一个节点 * ,这种情况需要遍历找到该节点的前一个节点,时间复杂度为O(n)。对于链表, * 链表中的每个节点结构都是一样的,所以我们可以把该节点的下一个节点的数据复制到该节点 * ,然后删除下一个节点即可。要注意最后一个节点的情况,这个时候只能用常见的方法来操作,先找到前一个节点,但总体的平均时间复杂度还是O(1) */ public void delete(Node head, Node toDelete){ if(toDelete == null){ return; } if(toDelete.next != null){ // 要删除的是一个中间节点 toDelete.val = toDelete.next.val; // 将下一个节点的数据复制到本节点! toDelete.next = toDelete.next.next; } else{ // 要删除的是最后一个节点! if(head == toDelete){ // 链表中只有一个节点的情况 head = null; }else{ Node node = head; while(node.next != toDelete){ // 找到倒数第二个节点 node = node.next; } node.next = null; } } } }
面试大总结:Java搞定面试中的链表题目总结的更多相关文章
- (转)面试大总结之一:Java搞定面试中的链表题目
面试大总结之一:Java搞定面试中的链表题目 分类: Algorithm Interview2013-11-16 05:53 11628人阅读 评论(40) 收藏 举报 链表是面试中常出现的一类题目, ...
- 面试大总结之二:Java搞定面试中的二叉树题目
package BinaryTreeSummary; import java.util.ArrayList; import java.util.Iterator; import java.util.L ...
- Java原来还可以这么学:如何搞定面试中必考的集合类
原创声明 本文作者:黄小斜 转载请务必在文章开头注明出处和作者. 系列文章介绍 本文是<五分钟学Java>系列文章的一篇 本系列文章主要围绕Java程序员必须掌握的核心技能,结合我个人三年 ...
- 【搞定面试官】谈谈你对JDK中Executor的理解?
## 前言 随着当今处理器计算能力愈发强大,可用的核心数量越来越多,各个应用对其实现更高吞吐量的需求的不断增长,多线程 API 变得非常流行.在此背景下,Java自JDK1.5 提供了自己的多线程框架 ...
- 【搞定面试官】- Synchronized如何实现同步?锁优化?(1)
前言 说起Java面试中最高频的知识点非多线程莫属.每每提起多线程都绕不过一个Java关键字--synchronized.我们都知道该关键字可以保证在同一时刻,只有一个线程可以执行某个方法或者某个代码 ...
- 告别set和get,两大利器轻松搞定model转换
场景一:一般我们遇到需要新建model,常规做法就是创建一个类,老老实实的定义好model中的所有属性,一般来说属性对应的set方法和get方法都是少不了的,有时候还需要toString甚至equal ...
- 大前端时代搞定PC/Mac端开发,我有绝招
如果你是一位前端开发工程师,对"跨平台"一词应该不会感到陌生.像常见的前端框架:比如React.Vue.Angular,它们可以做网页端,也可以做移动端,但很少能做到跨PC.Mac ...
- 如何用python搞定验证码中的噪点
背景:朋友在为"关山口男子职业技术学校"写一款校园应用,于是找MoonXue写一个学生选课系统的登录接口.为了搞定这个接口,不得不先搞定这个系统的验证码. 验证码大概是这个样子 看 ...
- 三步轻松搞定delphi中CXGRID手动添加复表头(多行表头,报表头)
网上有代码动态生成cxgrid多行表头的源码,地址为:http://mycreature.blog.163.com/blog/static/556317200772524226400/ 如果要手动设计 ...
随机推荐
- SecureCRT for Linux突破30天使用限制
当然还有一种方法,就是当你试用点i agree到时候,在~/.vandyke/Config 会生成一个文件SecureCRT_eval.lic,删除以后就可以恢复30天试用
- [UNIX环境高级编程](第三版)中apue.h的问题
编译 gcc -g myls.c 时,报错 ‘找不到头文件 apue.h’ apue.h是作者自己写的一个文件,系统不自带.其中包含了常用的头文件,以及出错处理函数的定义. 需要到 http://ww ...
- 停车场管理软件附带源代码 J2EE服务端+android客户端
该源码是停车场管理软件附带源代码 J2EE服务端+android客户端,也是一套停车场管理车辆进出的管理软,喜欢的朋友可以看看吧. 应用的后台管理主要功能介绍:1 机构管理 ,机构有从属管理< ...
- 找回mysql数据库密码
前提条件:你需要有数据库服务器的权限 1:修改my.ini配置文件 Mysqld:其中的d代表什么? Deamon后台运行的服务程序,增加一行跳过权限验证 2:停止mysql服务运行 3:启动mysq ...
- apache重写字段详细说明
用Apache虚拟主机的朋友很多,apache提供的.htaccess模块可以为每个虚拟主机设定rewrite规则,这对网站SEO优化相当有用,同时也改善了用户体验.国内的虚拟机一般不提供.htacc ...
- jQuery Easyui DataGrid应用
冻结列 $('#tbList').datagrid({ pagination: true, frozenColumns: [[ { field: 'BId',checkbox:'true',width ...
- Hibernate Cascade & Inverse
Cascade - 修改实体表 Inverse - 修改中间表 http://www.cnblogs.com/amboyna/archive/2008/02/18/1072260.html 1.到底在 ...
- 使用 JSONP 实现跨域通信
简介 Asynchronous JavaScript and XML (Ajax) 是驱动新一代 Web 站点(流行术语为 Web 2.0 站点)的关键技术.Ajax 允许在不干扰 Web 应用程序的 ...
- 2014年辛星完全解读Javascript第八节 json
json是JavaScript Object Notation的简写,它是一种轻量级的数据交换格式,而且表达上很容易靠字面去理解.json是用于存储和传输数据的格式,通常用于向服务器端传递数据. ** ...
- ThinkPHP的缓存 F方法
一般使用文件方式的缓存就能够满足要求,而thinkphp还提供了一个专门用于文件方式的快速缓存方法f方法. 由于采用的是php返回方式,所以其效率较s方法较高. f方法具有如下特点: 1.简单数据缓存 ...