删除特定元素链接列表

问题描述 投票:2回答:2

你能帮我理解一下为什么这个函数不能删除链接列表中的特定元素?我到底做错了什么?

typedef struct str_node{
    int data;
    struct str_node *next;
}node;

...
node *head;
head = malloc(sizeof(node));
...

void delete_spec(node *a){
    int num;
    node *tmp;

    tmp = a;
    printf("Insert number to eliminate: ");
    scanf("%d",&num);
    while(tmp!=NULL){
        if(tmp->data == num){
            tmp = tmp->next->next;
        }
        tmp = tmp->next;
    }
}
c struct linked-list singly-linked-list function-definition
2个回答
1
投票

首先,不清楚这里的分配是做什么的

node *head;
head = malloc(sizeof(node));

指针头最初应设置为NULL。

node *head = NULL;

和新的节点应通过在列表中插入新值的函数在列表中插入新的节点,从列表中删除节点的函数不应发出任何提示。

从列表中删除节点的函数不应发出任何提示。而是由函数的调用者要求用户指定将从列表中删除的值,然后调用传递指定值的函数。所以该函数应该有两个参数:指向头部节点的指针和应从列表中删除的整数值。

该函数的定义方式如下

void delete_spec( node **head, int data )
{
    while ( *head != NULL )
    {
        if ( ( *head )->data == data )
        {
            node *tmp = *head;
            *head = ( *head )->next;
            free( tmp );
        }
        else
        {
            head = &( *head )->next;
        }
    }
}     

下面是一个演示程序。

#include <stdio.h>
#include <stdlib.h>

typedef struct str_node
{
    int data;
    struct str_node *next;
} node;

int push_front( node **head, int data )
{
    node *new_node = malloc( sizeof( node ) );
    int success = new_node != NULL;

    if ( success )
    {
        new_node->data = data;
        new_node->next = *head;
        *head = new_node;
    }

    return success;
}

void delete_spec( node **head, int data )
{
    while ( *head != NULL )
    {
        if ( ( *head )->data == data )
        {
            node *tmp = *head;
            *head = ( *head )->next;
            free( tmp );
        }
        else
        {
            head = &( *head )->next;
        }
    }
}

void display( node *head )
{
    for ( ; head != NULL; head = head->next )
    {
        printf( "%d -> ", head->data );
    }

    puts( "null" );
}

int main(void) 
{
    node *head = NULL;
    int a[] = { 1, 3, 5, 7, 1, 2, 3, 1 };
    const size_t N = sizeof( a ) / sizeof( *a );

    for ( size_t i = 0; i < N; i++ )
    {
        push_front( &head, a[i] );
    }

    display( head );

    delete_spec( &head, 1 );

    display( head );

    return 0;
}

它的输出是

1 -> 3 -> 2 -> 1 -> 7 -> 5 -> 3 -> 1 -> null
3 -> 2 -> 7 -> 5 -> 3 -> null

0
投票

delete_spec 无论如何都不会修改输入列表。同时:它不会释放任何内存。

为了实际删除一个节点,你必须:1.释放它的内存.2.修改列表,使 "下一个 "指针得到更新.为了更新列表,你必须向删除函数提供头的地址,这样它就可以同时修改头。

就像这样。

void delete_spec(node **a){
    int num;
    node *tmp;

    if (a == NULL || *a == NULL) return;

    tmp = *a;

    printf("Insert number to eliminate: ");
    scanf("%d",&num);

    if (tmp->data == num)
    {
        *a = (*a)->next;
        free(tmp);
        return;
    }


    while(tmp->next!=NULL){
        if(tmp->next->data == num){
            node* tmp2 = tmp->next;
            tmp->next = tmp->next->next;
            free(tmp2);
        }
        tmp = tmp->next;
    }
}
© www.soinside.com 2019 - 2024. All rights reserved.