About qsort() in C, difference between ** buf and buf[][]

后端 未结 2 1140
没有蜡笔的小新
没有蜡笔的小新 2021-01-16 13:59

When I use qsort() in the C on my Mac, these code works well, It can sort every lines in one file well.

int compare(const void *p, const void *q) {
    retur         


        
相关标签:
2条回答
  • 2021-01-16 14:04

    When you attempt to qsort buf declared as char **buf, you are using the wrong comparison function. As I noted in the comment, char buf[x][y] is an x array of character arrays of y chars, e.g. char (*)[y] when passed as a parameter. When you declare buf as char **buf;, you declare a pointer to a pointer to type char. In either case, you have a pointer to string, and you must dereference each value passed to qsort one additional level of indirection., e.g.

    int cmpstrings (const void *a, const void *b) {
        return strcmp (*(char * const *)a, *(char * const *)b);
    }
    

    A short example using char **buf; would be:

    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    
    /* qsort string comparison - for pointer to pointer to char */
    int cmpstrings (const void *a, const void *b) {
        return strcmp (*(char * const *)a, *(char * const *)b);
    }
    
    int main (void) {
    
        char *ap[] = { "This is a tale",
                        "of captian Jack Sparrow",
                        "a pirate so brave",
                        "on the seven seas." },
            **buf = NULL;
        int n = sizeof ap/sizeof *ap;    
    
        if (!(buf = malloc (n * sizeof *buf))) { /* allocate pointers */
            fprintf (stderr, "error: virtual memory exhausted.\n");
            return 1;
        }
    
        for (int i = 0; i < n; i++)
            buf[i] = strdup (ap[i]);    /* allocate/copy strings */
    
        qsort (buf, n, sizeof *buf, cmpstrings);
    
        for (int i = 0; i < n; i++) {   /* print and free */
            printf ("buf[%d] : %s\n", i, buf[i]);
            free (buf[i]);
        }
        free (buf);
    
        return 0;
    }
    

    Example

    $ ./bin/qsortptp
    buf[0] : This is a tale
    buf[1] : a pirate so brave
    buf[2] : of captian Jack Sparrow
    buf[3] : on the seven seas.
    
    0 讨论(0)
  • 2021-01-16 14:13

    qsort, memcpy and similar functions assumes that the pointer passed points at an array. The very definition of an array is x number of items allocated contiguously, in adjacent memory cells. For example char array [x][y];.

    There is wide-spread confusion about the use of pointer-to-pointers. There's an old trick you can use to declare a look-up table, where each item points to an array of variable length, which goes like this:

    char** ptr = malloc(x * sizeof(char*));
    for(int i=0; i<x; i++)
    {
      ptr[i] = malloc(y * sizeof(char));
    }
    

    This allows you to access the look-up table with array-like syntax, ptr[i][j]. But that does not mean that this is an array, there is no relation whatsoever beween a pointer-to-pointer and a 2D array! This is rather x number of segments, where each segment allocated at any place on the heap.

    The only reason why you would ever use the above pointer-to-pointer trick is when you must have variable length for each item in the look-up table. If you don't need that, then the above method is plain bad and always incorrect - it is much slower than a real array, both in terms of allocation overhead, heap fragmentation and poor data cache use. And as you discovered, it can't even be used as an array... because it isn't one.

    Now there are unfortunately numerous incompetent would-be gurus all over the world incorrectly teaching the above as the way to allocate a multi-dimensional array dynamically. Which is complete nonsense. It is not a multi-dimensional array and can't be used as one. See this for an explanation of how you actually should allocate multi-dimensional arrays dynamically.

    0 讨论(0)
提交回复
热议问题