从已排序的链接列表中一次删除“2个重复的”元素

时间:2016-01-09 07:53:28

标签: c data-structures linked-list

void del_duplicated(struct node *first) {
    struct node *current = listfirst, *prev = listfirst, *tmp;
    if (current == NULL)
        printf("Nothing to delete !\n");
    while (current != NULL) {
        /* Keeping traverse the list to find the duplicated elements. */
        prev = current;
        current = current->next;

        /* for the first and second duplicated node such as 1 1 2 3 5 */
        if (prev->data == listfirst->data && current->data == listfirst->data) {
            listfirst = current->next;
            printf("LIST FIRST NOW AT %d", listfirst->data);
        }
        /* The rest requirement such as 1 2 4 4 5 convert to 1 2 5 */
        else if ((prev->next)->data == (current->next)->data) {
            (prev->next) = (current->next)->next;  /*this is point 2 to 5*/
            tmp = current;                 /*delete the node*/
            free(tmp);
            tmp = current->next;
            free(tmp);
        }
    }
}

我有一个链表问题,要求我从列表中删除“2重复节点”。 这就像1 1 2 3 5转换为2 3 5一样。 1 2 4 4 5将转换为1 2 5

但我的程序崩溃是因为指针指向一个奇怪的地方,我不知道为什么。 (.exe已停止工作) 我认为我的移动指针的逻辑是正常的,但...... error like this 我的逻辑附在我的源代码的注释中。

我是台湾计算机科学编程和学习的新生。 请原谅我糟糕的英语。

1小时后

修改

我在我的两个法官陈述中添加了BREAK,并且运作良好。 但我不知道为什么。

void del_duplicated(struct node *first) {
    struct node *current = listfirst, *prev = listfirst, *tmp, *tmp2;
    if (current == NULL)
        printf("Nothing to delete !\n");
    while (current != NULL) {
        /* Keeping traverse the list to find the duplicated elements. */
        prev = current;
        current = current->next;
        //printf("prev %d,current %d\n", prev->data, current->data);
        //system("pause");
        /* for the first and second duplicated node such as 1 1 2 3 5 */
        if (prev->data == listfirst->data && current->data == listfirst->data) {
            listfirst = current->next;
            system("pause");
            break;
        }
        /* The rest requirement such as 1 2 4 4 5 convert to 1 2 5 */
        else if (current->data == (current->next)->data) {
            (prev->next) = (current->next)->next;  /* this is point 2 to 5 */
            tmp = current->next;
            tmp2 = current;                 /* delete the node */
            current = (current->next)->next;
            free(tmp);
            free(tmp2);
            break;
        }
    }
}

1 个答案:

答案 0 :(得分:0)

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

typedef struct node NODE;
struct node{
    int         data;
    NODE        *next;
    NODE        *prev;
};


NODE * find_node( NODE *first, int value){
    while( first ){
        if( first->data == value )
            return first;
        first = first->next;
    }
    return NULL;
}

NODE* delete_node( NODE  **pnd ){

    NODE *tmp, *nd;

    if(!pnd || !*pnd)
        return  NULL;

    nd  =      *pnd;
    tmp =      nd->next;



    if(nd->next)
        nd->next->prev =    nd->prev;

    if(nd->prev)
        nd->prev->next =    nd->next;

   free( nd );

   *pnd=NULL;

   return tmp;
}


void del_duplicated( NODE **first ) {
    NODE    *tmp,
            *dup,
            *current=*first;
    int val;
    while( current ){

        dup =   find_node(tmp,current->data);   // find in next nodes
        tmp =   current->next;
        val=    current->data;
        while(1){
            dup=find_node(tmp,val);
            if(!dup) break;

            while(dup){                         // delete all occurences
                if(dup == tmp)
                    tmp = dup->next;
                delete_node(&dup);
                dup=find_node(tmp,val);
            }
            delete_node(&current);              // delete current one aswell
       }     
       current=tmp;                                 

    }
}


NODE *new_node(NODE *parent,int value){

    NODE *nd =  malloc( sizeof( NODE ) );
    nd->prev =  parent;

    if( parent ){
        nd->next =      parent->next;
        if(parent->next)
            parent->next->prev =    nd;

        parent->next =  nd;
    }else
        nd->next =      NULL;

    nd->data =          value;

    return nd;

}



void free_nodes(NODE *root){
    NODE    *tmp;

    if(root && root->prev){
        root->prev->next =  NULL;
    }
    while(root){
        tmp =               root;
        root =              root->next;
        free( tmp );
    }
}

void print_nodes( NODE *root, char *text){

    if(text)
        printf("%s:\n", text);

    while( root ){
        printf("%d ", root->data );
        root =  root->next;
    }
    printf("\n");
}

int main( void ){
    NODE        *root;
    root =      new_node(NULL,10);

    new_node( root, 25);
    new_node( root, 25);
    new_node( root, 25);
    new_node( root, 20);
    new_node( root, 15);
    new_node( root, 16);
    new_node( root, 16);
    new_node( root, 5);

    print_nodes( root ,"\nWith duplicated items");
    del_duplicated(&root);
    print_nodes( root ,"\nDuplicated items removed");

    free_nodes(root);
    return 0;
}