簡體   English   中英

用於從已排序數組中刪除元素的函數

[英]Function to remove an element from a sorted array

從排序數組中刪除元素時遇到一些問題(必須刪除元素的所有實例)。 當我運行程序時,我遇到了分段錯誤。 我不知道為什么會發生這種情況,因為函數remElem(int *,int *,int)其中1st arg是一個數組,2nd arg是一個數組的長度(當元素被刪除時會改變)而3rd arg是一個元素要刪除 - 在使用switch語句擴展程序之前正常工作,以及其他函數,如shuffle(int *,int)(用於控制數組的元素)和insElem(int *,int *,int)(用於插入元素)我不知道出了什么問題,請幫忙。我會提供注釋的代碼。評論中還提供了一些其他的小問題。提前謝謝:)

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

#define MAX_LEN 1000


void initArray(int*, int);
void printArray(int*, int);
void swap(int*, int*);
void insSort(int*, int);
void insElem(int*, int*, int);
void remElem(int*, int*, int);
void shuffle(int*, int);


int main() {
    int array[MAX_LEN];
    int len, elem, comm;

    srandom(time(NULL));

    printf("Number of elements? >> ");
    scanf("%d", &len);

    initArray(array, len);

    do {
        printf("Command? (6 for help) >> ");
        scanf("%d", &comm);
        switch (comm) {
            case 1:
                printArray(array, len);
                break;
            case 2:
                shuffle(array, len);
                break;
            case 3:
                insSort(array, len);
                break;
            case 4:
                printf("Insert element? >> ");
                scanf("%d", &elem);
                insElem(array, &len, elem);
                break;
            case 5:
                printf("Remove element? >> ");
                scanf("%d", &elem);
                remElem(array, &len, elem);
                break;
            case 6:
                printf("Help:\n");
                printf("1 - Print\n");
                printf("2 - Shuffle\n");
                printf("3 - Sort\n");
                printf("4 - Insert element\n");
                printf("5 - Remove element\n");
                printf("6 - Help (this screen)\n");
                printf("0 - Quit\n");
                break;
            case 0:
                break;
            default :
                printf("Wrong input! Repeat!\n");
        }
    } while (comm);

    return EXIT_SUCCESS;
}


/*Initializes array with n random numbers from 0 to 9 (including)*/
void initArray(int a[], int n) {
    int i;

    for (i = 0; i < n; i++)
        a[i] = random() % 10;
}


/*Prints n elements of an array*/
void printArray(int a[], int n) {
    int i;

    for (i = 0; i < n; i++)
        printf("%d ", a[i]);
    putchar('\n');
}


/*Swaps the values of two variables*/
void swap(int *i, int *j) {
    *i = *i + *j;
    *j = *i - *j;
    *i = *i - *j;
    /*I saved up some memory yaaaaaaay :3 */
    /*I spent some processing time nooooo :| */
    /*Which is better... with or without tmp variable???*/
    /*I suppose it depends on application... clearly, this
      method doesnt work with structures, for example*/
}


/*Sorts the elements of an array using insertion sort algorythm*/
void insSort(int a[], int n) {
    int i, j;

    for (i = 1; i < n; i ++)
        for (j = i; j > 0 && a[j] < a[j-1]; j--)
            swap(&a[j], &a[j-1]);
}


/*Inserts an element into a sorted array*/
/*Wassn meant to be working with unsorted arrays
  in that case unpreddictable results*/
void insElem(int a[], int *n, int e) {
    int i, j;

    (*n)++;
    for (i = 0; a[i] < e && a[i+1] < e; i++);

    for (j = *n; j > i + 1; j--)
        a[j] = a[j-1];

    a[i+1] = e;
}


/*Removes an element from a sorted array*/
/*Wassn meant to be working with unsorted arrays
  in that case unpreddictable results*/
void remElem(int a[], int *n, int e) {
    int i, j;

    for (i = 0; i < *n; i++) {
        while (a[i] == e) {
            for (j = i; j < *n; j++)
                a[j] = a[j+1];
        (*n)--;
        }
    }
}


/*Shuffles the elements of an array*/
/*I just did this on the fly...
  are there any known algorythms for doing this*/
void shuffle(int a[], int n) {
    int i;

    for (i = 0; i < n; i++)
        swap(&a[rand()%n], &a[rand()%n]);
}

以下可能導致越界訪問,導致未定義的行為:

        for (j = i; j < *n; j++)
            a[j] = a[j+1];

考慮當j == (*n)-1時, a[j+1]會發生什么。

對於swap()函數,您看起來很聰明的實現存在整數溢出和未定義行為的風險。 只需使用臨時變量,讓編譯器擔心效率。

在這個循環中:

for (j = i; j < *n; j++)
  a[j] = a[j+1];

j = *n - 1 ,您將訪問數組邊界之外的內存。 這會調用未定義的行為,結果可能是任何內容,因此我相信它會解釋您的崩潰。 至於程序之前沒有崩潰 - 這可以做任何事情也可能沒有負面影響。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM