链表中环的入口结点
解答:本题使用hash法比较简单,也可以使用双指针法,但是比较难。索性我就捡了一个相对容易理解的解法。用的是hash法。
应该就是创建一个新的空的单链表,然后进行遍历每个节点,如果这个节点没有就存进去,如果有了,就是链表节点重复了,这个重复的节点就是入口。
public ListNode EntryNodeOfLoop(ListNode pHead) {
// 使用set来记录出现的结点
HashSet<ListNode> set = new HashSet<>();
while(pHead != null){
// 当set中包含结点,说明第一次出现重复的结点,即环的入口结点
if(set.contains(pHead)){
return pHead;
}
// set中加入未重复的结点
set.add(pHead);
pHead = pHead.next;
}
return null;
}
链表中倒数最后k个结点
解析:
我们无法逆序遍历链表,就很难得到链表的倒数第k个元素,那我们可以试试反过来考虑,如果当前我们处于倒数第k的位置上,即距离链表尾的距离是k,那我们假设双指针指向这两个位置,二者同步向前移动,当前面个指针到了链表头的时候,两个指针之间的距离还是k。虽然我们没有办法让指针逆向移动,但是我们刚刚这个思路却可以正向实施:
step 1:准备一个快指针,从链表头开始,在链表上先走k步。
step 2:准备慢指针指向原始链表头,代表当前元素,则慢指针与快指针之间的距离一直都是k。
step 3:快慢指针同步移动,当快指针到达链表尾部的时候,慢指针正好到了倒数k个元素的位置。
画图解析: