删除链表的倒数第k个结点

删除链表的倒数第k个结点

问题重述:

给你一个链表,删除链表的倒数第 k 个结点,并且返回链表的头结点。

示例 1:

删除链表的倒数第k个结点

输入:head = [1,2,3,4,5], k = 2
输出:[1,2,3,5]

示例 2:

输入:head = [1], k = 1
输出:[]

示例 3:

输入:head = [1,2], k = 1
输出:[1]

提示:

  • 链表中结点的数目为 sz
  • 1 <= sz <= 30
  • 0 <= Node.val <= 100
  • 1 <= k <= sz

问题分析:

本题要求删除倒数第k个结点,重要的是获取删除结点的前一个结点,将该节点的下一个结点指向要删除的节点的下一个结点,常规思路是遍历链表,获得链表长度,然后根据链表长度找到要删除的结点,这种方法需要遍历两次链表。使用双指针的方法可以做到一次遍历就能够找到结点。使用双指针,一个指针从头开始一个指针从第k个开始,然后两个指针一起向后遍历,当快指针到达链表尾部的时候,慢指针对应的刚好是倒数第k+1个结点

解法:

链表遍历、双指针(一次遍历)

解题:

代码:
package test.linklist;
/**
 * 
 * @author FOLDN
 * 从长度为n的链表中,删除倒数第k个结点
 *
 */
public class DeleteKthNode {
	public static void main(String[] args) {
		
	}
	public static Node deleteKthNode(Node head,int k) {
		/*
		if(k < 1 || head == null) {
			return head;
		}
		// 方法一,直接遍历链表
		// 记录链表的头,因为最后要返回链表的头
		Node linkedList = head;
		while(linkedList != null) {
			// 第一次遍历链表,每经过一个结点,k-1,遍历到最后k = k-n
			k = k-1;
			linkedList = linkedList.next;
		}
		// 遍历完链表后,k有三种情况,分别是等于0,大于0,小于0
		// k大于0,说明需要删除的结点不存在于链表中,不需要处理,直接返回链表即可
		// k等于0,说明要删除的是链表的头节点
		if(k == 0) {
			// 删除头节点然后返回链表
			head = head.next;
		}
		// k小于0,需要再次进行遍历确定要删除的结点
		if(k < 0) {
			// 第二次遍历需要注意的是此时的linkedList对应的是最后的结点,所以需要重新赋值
			linkedList = head;
			while(k != 0) {
				k = k+1;
				linkedList = linkedList.next;
			}
			// 删除结点
			linkedList.next = linkedList.next.next;
		}
		return head;
		*/
		// 方法二,使用双指针,只需要使用一次循环
		if(k < 1 || head == null) {
			return head;
		}
		Node fast = head;
		Node slow = head;
		for(int i = 0;i < k; i++) {
			fast = fast.next;
		}
		// 这一步是为了当删除的结点是头结点的时候,可以正常返回
		if(fast == null) {
			return head.next;
		}
		while(fast.next != null) {
			fast = fast.next;
			slow = slow.next;
		}
		slow.next = slow.next.next;
		return head;
	}

}
代码解析:

链表遍历相当简单,就是先遍历一般确定长度,然后通过长度确定要删除的节点的前一个结点,然后第二次遍历找到要要删除的结点的前一个结点,对链表进行更改

本题我们使用快慢双指针对链表进行遍历,一遍遍历即可确定要删除的结点。首先让快指针fast先向前移动k个位置,此时让fast指针和slow指针一起向后遍历,当fast指针遍历完最后一个节点(即此时的fast结点对应的是链表最后一个结点对应的next,为null)的时候,slow结点对应的结点就是要删除的结点。为了简便删除对应结点,我们可以一开始创建一个哑结点,将该节点的next设为head,这样我们可以处理直接处理当要删除的结点为头节点的问题,并且找到的结点是要删除的节点的下一个结点

总结:

快慢双指针一般用于倒数的元素的处理或者检测会回文序列

上一篇:在项目中使用Spring Cloud Alibaba Sentinel组件


下一篇:写出我心(十四)