链表删除结点

题目描述

输入一个正整数repeat(0 < repeat < 10),做repeat次下列运算:
输入一个正整数n(0 < n < 10)和一组( n 个 )整数,建立一个单向链表
再输入一个整数x,将链表中最后一个与x相等的整数删除。

输入格式

见输入样例

输出格式

见输出样例

输入样例

3
5
1 2 4 3 7
4
5
2 1 5 7 5
5
3
1 2 4
100

输出样例

size=4:1 2 3 7
size=4:2 1 5 7
size=3:1 2 4

思路

创建一条单向链表,因为题目要找出并删除最后一个与x相同的数,所以反转链表,删除第一个与x相同的数,这样就不用做标记,删除后再反转一次,最后输出。

#include<stdio.h>
#include<stdlib.h>
int flag;
struct Node{
    int num;
    struct Node *next;
};

struct Node *Create(int n)   //创建链表
{
    struct Node *current,*prev;
    struct Node *head = NULL;
    int cnt = 0;

    while (cnt != n)
    {
        current = (struct Node *)malloc(sizeof(struct Node));

        scanf("%d",&current->num);

        if (head == NULL)
            head = current;
        else
            prev->next = current;
        prev = current;
        cnt++;
    }
    prev->next = NULL;
    return head;
}

void Print(struct Node *head)  //打印链表
{
    struct Node * current;
    current = head;

    printf("%d",current->num);
    current = current->next;
    while (current != NULL)
    {
        printf(" %d",current->num);
        current = current->next;
    }
    printf("\n");
}

struct Node *Reverse(struct Node *head)  //反转链表
{
    struct Node *current,*prev,*tmp;

    current = head;
    prev= current->next;
    while (prev != NULL)
    {
        tmp = prev->next;
        prev->next = current;
        current = prev;
        prev = tmp;
    }
    head->next = NULL;
    head = current;
    return head;
}

struct Node *Delete(struct Node *head,int num)  //删除特定结点
{
    struct Node *current;
    struct Node *prev;

    current = head;

    while (current->num != num && current->next != NULL)
    {
        prev = current;
        current = current->next;
    }

    if (current->num == num)
    {
        if (head == current)
        {
            head = current->next;
        }
        else
        {
            prev->next = current->next;
        }
    }
    else
    {
        flag = 0;
    }

    return head;
}

int main()
{
    int N,repeat,x;
    struct Node *head;

    scanf("%d",&repeat);

    while (repeat--)
    {
        flag = 1;
        scanf("%d",&N);
        head = Create(N);
        scanf("%d",&x);
        head = Reverse(head);
        head = Delete(head,x);
        head = Reverse(head);
        if (flag)
           printf("size=%d:",N-1);
        else
           printf("size=%d:",N);
        Print(head);
    }

    return 0;
}

后来在和同学讨论的时候发现自己忘了释放内存,之后就补了Free函数

void Free(struct Node *head)
{
    struct Node *current;

    current = head;

    while (current != NULL)
    {
        free(current);
        current = current->next;
    }
}

但是在添加Free函数调用它的时候主函数中只repeat了一次,没有再次进入while(repeat--)中,回看代码看了很久都没发现错误,除Free函数各模块代码都没有错,因为之前没有Free函数时都可以正常的输出。之后调试了一下程序,发现程序一直停止在

Free(head);

这个语句。问了下学姐,提醒我打出current地址,尝试打印了一下Free函数里面current变量的地址值

地址五个一次循环,难怪一直没有第二次进入while循环。但是在Free函数里面也没发现错误。最后求助了学长,学长说在Free函数里面,free掉current后又用到了current的值导致出错。然后我就尝试着修改了一下

while (current != NULL)
    {
        tmp = current;
        free(current);
        current = tmp->next;
    }

运行之后还是老样子,想了很久没发现逻辑上的问题,学长学姐提点了一下,tmp和current指向同一个内存区,然后又free掉了,所以tmp最后也指向了不明内存区,跟第一次犯的错误根源上一样。现在也慢慢发现对于指针的内涵还是不是很懂。最后改成如此

void Free(struct Node *head)
{
    struct Node *current,*tmp;

    current = head;

    while (current != NULL)
    {
        tmp = current->next;
        free(current);
        current = tmp;
    }
} 

附上大神本题另一种解法:

#include <cstdio>
using namespace std;
struct Node{
    int val;
    Node *next;
}   tab[10086];
Node *p[10086];

int  n,repeat;
int  i,j,k,x;

int main()
{

    scanf("%d",&repeat);

    for(j=0; j<repeat; j++)
    {
        scanf("%d",&n);

        for (i=0; i<=20; i++) p[i]=&tab[i];
        //--1 read
        for (i=1; i<=n; i++)
        {
            scanf("%d",&p[i]->val);
            p[i-1]->next=p[i];
        }
        p[n]->next=NULL;

        //--2 search x
        scanf("%d",&x);
        Node *tmp=NULL;
        for (Node *tai=p[0]->next; tai!=NULL; tai=tai->next)
            if (tai->val==x) tmp=tai;

        //--3 delete tmp
        for (Node *tai=p[0]; tai!=NULL; tai=tai->next)
        if (tai->next!=NULL)
        if (tai->next==tmp)
        {
            tai->next=tai->next->next;
            n-=1;
        }

        //--4 Print
        printf("size=%d:",n);
        for (Node *tai=p[0]->next; tai!=NULL; tai=tai->next)
        {
            printf("%d",tai->val);
            n-=1;
            if(n>0) printf(" ");
        }

        if (j<repeat-1) printf("\n");
    }
    return 0;
}

总结

虽然在线提交的时候,数据点测试通过,但是没有考虑指针,通过简单的一个Free函数,折腾了好久,也错了很多次,但也因此更了解指针
指针的理解与应用仍需加强啊

PTA 链表删除结点的题目测试的更多相关文章

  1. pta 奇数值结点链表&&单链表结点删除

    本题要求实现两个函数,分别将读入的数据存储为单链表.将链表中奇数值的结点重新组成一个新的链表.链表结点定义如下: struct ListNode { int data; ListNode *next; ...

  2. leetCode:237 删除链表的结点

    删除链表的结点 编写一个函数,在给定单链表一个结点(非尾结点)的情况下,删除该结点. 假设该链表为1 -> 2 -> 3 -> 4 并且给定你链表中第三个值为3的节点,在调用你的函数 ...

  3. 237 Delete Node in a Linked List 删除链表的结点

    编写一个函数,在给定单链表一个结点(非尾结点)的情况下,删除该结点. 假设该链表为1 -> 2 -> 3 -> 4 并且给定你链表中第三个值为3的节点,在调用你的函数后,该链表应变为 ...

  4. 数据结构:链表 >> 链表按结点中第j个数据属性排序(冒泡排序法)

    创建结点类,链表类,测试类 import java.lang.Object; //结点node=数据date+指针pointer public class Node { Object iprop; p ...

  5. A Magic Lamp---hdu3183(链表删除| RMQ)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3183 给你一个长度<1000的数a,和m<len(a); 让把数a删除m个数字之后剩下的数 ...

  6. 删除Oracle Online Redo 测试

    删除Oracle Online Redo 测试 SQL> select * from v$log; ​   GROUP#   THREAD# SEQUENCE#     BYTES BLOCKS ...

  7. xml解析-jaxp删除结点

    jaxp删除结点 / 删除sex结点 * 1.创建解析器工厂 * 2.根据解析器工厂创建解析器 * 3.解析xml返回document * * 4.得到sex结点 * 5.得到sex的父节点 getP ...

  8. C语言:将字符串中的字符逆序输出,但不改变字符串中的内容。-在main函数中将多次调用fun函数,每调用一次,输出链表尾部结点中的数据,并释放该结点,使链表缩短。

    //将字符串中的字符逆序输出,但不改变字符串中的内容. #include <stdio.h> /************found************/ void fun (char ...

  9. LeetCode 876. Middle of the Linked List(获得链表中心结点)

    题意:获得链表中心结点.当有两个中心结点时,返回第二个. 分析:快慢指针. /** * Definition for singly-linked list. * struct ListNode { * ...

随机推荐

  1. Java应用程序项目的打包与发行(run.bat形式)

    参考: http://www.iteye.com/topic/57312 背景: 以前一直都是在eclipse上面创建应用程序,每次要要运行的时候都要打开eclipse, 直到有个同事叫我帮忙写一个应 ...

  2. Theano2.1.10-基础知识之循环

    来自:http://deeplearning.net/software/theano/tutorial/loop.html loop 一.Scan 一个递归的通常的形式,可以用来作为循环语句. 约间和 ...

  3. JavaScript 位运算总结&拾遗

    最近补充了一些位运算的知识,深感位运算的博大精深,此文作为这个系列的总结篇,在此回顾下所学的位运算知识和应用,同时也补充下前文中没有提到的一些位运算知识. 把一个数变为大于等于该数的最小的2的幂 一个 ...

  4. 3到6年的.NETer应该掌握哪些知识?

    我们组的开发人力一直比较紧张,今年春节后,高层终于给了几个headcount,我们可以开始招人了.从三月初我们就开始找简历,渠道有拉钩,内推,我司自己的招聘网站和智联等.简历筛了很多,也打了很多电话, ...

  5. C#中数组Array、ArrayList、泛型List<T>的比较

    在C#中数组Array,ArrayList,泛型List都能够存储一组对象,但是在开发中根本不知道用哪个性能最高,下面我们慢慢分析分析. 一.数组Array 数组是一个存储相同类型元素的固定大小的顺序 ...

  6. 基于DDD的.NET开发框架 - ABP缓存Caching实现

    返回ABP系列 ABP是“ASP.NET Boilerplate Project (ASP.NET样板项目)”的简称. ASP.NET Boilerplate是一个用最佳实践和流行技术开发现代WEB应 ...

  7. 如何把自己打造成技术圈的 papi 酱

    最近半年,一个叫papi酱的平胸女子连续在微博.朋友圈.创业圈刷屏,当之无愧成了中文互联网的第一大网红.呃,你以为我会巴拉巴拉说一堆网工创业的事?NO,今天想借papi酱的话题跟大家一起聊聊程序员如何 ...

  8. SQL Server Data Tools – Business Intelligence for Visual Studio 2012安装时提示“The CPU architecture....”的解决方法

    SQL Server Data Tools – Business Intelligence for Visual Studio 2012,一个很强大的工具,下载地址:http://www.micros ...

  9. 如何批量删除虚拟机及其关联的存储(Windows Azure)

    可以通过运行附件中PowerShell脚本文件RemoveVMandDisk.ps1批量删除VM和Disk,详细代码如下: param($serviceName) echo "Startin ...

  10. Java序列化中的SerialVersionUid

    版权声明:本文为博主fbysss原创文章,转载请注明出处 作者:fbysssmsn:jameslastchina@hotmail.com  blog:blog.csdn.NET/fbysss声明:本文 ...