面试大总结之一:Java搞定面试中的链表题目

链表是面试中常出现的一类题目,本文用Java实现了面试中常见的链表相关题目。本文主要参考整合重写了《轻松搞定面试中的链表题目》和 《算法大全(1)单链表》两篇大作。两篇大神的实现分别是C和C#,因为我更喜欢用Java面试,所以用Java重写了所有实现,并附上自己的一些思考注释。算法大全(1)单链表 尚未有一些问题尚未整合进来,很快我会更新本文。接下来还会出关于二叉树的总结和栈和队列的大总结。因为这些都属于面试中的送分题!必须毫无偏差地拿下。至于像动态规划这些比较“高端”的算法,就只能靠日积月累,而不能像这样临时突击了。

第二篇: 《面试大总结之二: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;
			}
		}
	}
	
}

谢谢@buding12321 指出的bug~ 已经修复了


    原文作者:chiiis
    原文地址: https://blog.csdn.net/fightforyourdream/article/details/16353519
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞