LeetCode:第19题 删除链表的倒数第N个节点 C语言实现

 

问题说明:

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

示例:

给定一个链表: 1->2->3->4->5, 和 n = 2.

当删除了倒数第二个节点后,链表变为 1->2->3->5.
说明:

给定的 n 保证是有效的。

 

进阶:

你能尝试使用一趟扫描实现吗?

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/remove-nth-node-from-end-of-list
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

解法1:安安

//为了方便处理极端情况,自己创建了带头结点的链表进行处理,可以正确处理样例中的多种情况
//但是,发现leetcode上面的链表是不带头结点的,气死人了

 

解法2:leetcode直接提交 安安

据自己理解,该题leetcode应该处理的是不带头结点的

按照不同情况进行处理

 

 

解法3:leetcode直接提交 

官方题解创建了一个哑结点,挂在了第一个结点的前面,那不就是带头结点的情况嘛

 

解法4(进阶):快慢指针

 

思路:安安

两次遍历,第一次遍历找到链表的长度,第二次进行定位找到所需结点的前一个结点,进行删除

思路:进阶(一次遍历)

(leetcode)我们可以使用两个指针而不是一个指针。第一个指针从列表的开头向前移动 n+1n+1 步,而第二个指针将从列表的开头出发。现在,这两个指针被 nn 个结点分开。我们通过同时移动两个指针向前来保持这个恒定的间隔,直到第一个指针到达最后一个结点。此时第二个指针将指向从最后一个结点数起的第 nn 个结点。我们重新链接第二个指针所引用的结点的 next 指针指向该结点的下下个结点。

 


/*
019.删除链表的倒数第N个结点
anan
2020.1.18
*/ 

//这个是自己写的带头结点的

#include<stdio.h>
#include<malloc.h>
 
struct ListNode {
	int val;
	struct ListNode *next;
};


struct ListNode* createLinkList(int a[], int length);
void dispLinkList(struct ListNode *head);

void dispLinkList(struct ListNode *head){
	struct ListNode *p = head->next;
	while(p){
		printf("%d ", p->val);
		p = p->next;
	}
	printf("\n");
}

struct ListNode* createLinkList(int a[], int length){   //带头节点 
	struct ListNode *s;
    struct ListNode *h=NULL;   //头节点 
    struct ListNode *r;  //指向新链表的最后一个节点
	
	h =  (struct ListNode *)malloc(sizeof(struct ListNode));
	h->next = NULL;
	r = h;
	
	for(int i = 0; i < length; i++){
        s = (struct ListNode *)malloc(sizeof(struct ListNode));
        s->val = a[i];
        //printf("s->val:", s->val);
        s->next = NULL;

		r->next = s;
		r = s;		
 	}
	
	return h;
}

//为了方便处理极端情况,自己创建了带头结点的链表进行处理,可以正确处理样例中的多种情况
//但是,发现leetcode上面的链表是不带头结点的,气死人了

struct ListNode* removeNthFromEnd(struct ListNode* head, int n){
	struct ListNode *p;
	int link_length = 0;
	struct ListNode *pre;
	
	p = head->next;  //确定链表长度
	while(p){     
		link_length++;
		p = p->next;
	}
	//printf("链表长度为:%d\n", link_length);
	
	pre = head;
	p = pre->next;
	for(int i = 1; i < link_length-n+1; i++){
		pre = p;
		p = p->next;
		//printf("%d, %d\n", i, p->val);
	}

	pre->next = p->next;
	free(p);
	return head;
}




int main(){

	int a[] = {1,2,3,4,5};
	int length_a = 5;
	int n = 4;

//	int a[] = {1};
//	int length_a = 1;
//	int n = 1;

//	int a[] = {1,2};
//	int length_a = 2;
//	int n = 1;

//	int a[] = {1,2};
//	int length_a = 2;
//	int n = 2;
		
	struct ListNode* l1;
	struct ListNode* l2;
	
	l1 = createLinkList(a, length_a);
	printf("链表a为:");
	dispLinkList(l1);
	
	l2 = removeNthFromEnd(l1, n);
	printf("去掉了倒数第%d个节点的链表b为:", n);	
	dispLinkList(l2);
}

//这个是leetcode进行提交的  不创建哑结点


/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
struct ListNode* removeNthFromEnd(struct ListNode* head, int n){
	struct ListNode *p = head;
	int link_length = 0;
	struct ListNode *pre = head;
	
	while(p){    //确定链表长度 
		link_length++;
		p = p->next;
	}
	//printf("链表长度为:%d\n", link_length);
	
	
	p = pre;
	for(int i = 1; i < link_length-n+1; i++){
		pre = p;
		p = p->next;
		//printf("%d, %d\n", i, p->val);
	}
	if(p == pre){ //有两种情况
        if(p->next){   //说明删除了第一个节点,后面还有其他节点
            head = p->next;
            return head;
        }else{      //说明这个链表只有一个节点,删除完就没有了
            return NULL;
        }		
	}else{
		pre->next = p->next;
		free(p);		
		return head;   		
	}
}
//这个是直接在leetcode上提交的  创建了哑结点

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
struct ListNode* removeNthFromEnd(struct ListNode* head, int n){
    struct ListNode *dummy = (struct ListNode *)malloc(sizeof(struct ListNode));
    dummy->next = head; //设置一个哑结点指向链表的第一个结点(该题不带头结点)

	struct ListNode *p;   //进行链表遍历
	int link_length = 0;
	struct ListNode *pre;
	
    //确定链表长度 
    p = head;
	while(p){   
		link_length++;
		p = p->next;
	}
	//printf("链表长度为:%d\n", link_length);
	
    //定位所需结点,进行删除
	pre = dummy;
	p = pre->next;
	for(int i = 1; i < link_length-n+1; i++){
		pre = p;
		p = p->next;
		//printf("%d, %d\n", i, p->val);
	}

	pre->next = p->next;
	free(p);
	//return head;
    return dummy->next;
}
//快慢指针

/ *
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
struct ListNode* removeNthFromEnd(struct ListNode* head, int n){
    struct ListNode *dummy = (struct ListNode *)malloc(sizeof(struct ListNode));
    dummy->next = head; //设置一个哑结点指向链表的第一个结点(该题不带头结点)

	struct ListNode *p = dummy;   //进行链表遍历
	struct ListNode *q = dummy;

	for(int i = 0; i < n; i++){
		p = p->next;
	}

    while(p->next){
        p = p->next;
        q = q->next;
    }

	q->next = q->next->next;

    return dummy->next;
}

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

安安csdn

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值