【问题标题】:function which arranges the elements of an array in ascending order按升序排列数组元素的函数
【发布时间】:2021-04-22 03:02:12
【问题描述】:

我尝试创建一个按升序排列数组元素的函数。但结果并不好。我想我忘记了一件事……谢谢你的帮助。

#include <stdio.h>

void ft_sort_int_tab(int *tab, int size) {
    int i;
    int j;
    int temp;
    size -= 1;

    i = 0;
    while (i < size) {
        if (tab[i] > tab[i + 1]) {
            temp = tab[i];
            tab[i] = tab[i + 1];
            tab[i + 1] = temp;
        }
        i++;
    }
}

int main(void) {
    int tab[9] = {9, 5, 2, 3, 8, 4, 16, 20, 24};
    ft_sort_int_tab(tab, 9);
    for (int i = 0; i < 9; i++) {
        printf("%d ", tab[i]);
    }
}

结果:5 2 3 8 4 9 16 20 24

【问题讨论】:

  • 添加调试打印和/或使用调试器单步调试您的代码。 Stack Overflow 无法为您调试代码。
  • 当你使用调试器时,程序做错的第一件事是什么?
  • 这里if (tab[i] &gt; tab[i + 1])if i == size-1,你有一个未定义的行为
  • @Damien size 在循环开始之前预减了1。因此,对于在函数中传递的原始大小值,运行到 &lt; size 实际上是运行到 &lt; size-1
  • 您的排序不好,因为您忘记了外循环。这基本上是一种冒泡排序,但只有一次通过。它会让数字“更接近”他们所属的地方,甚至可能在他们最终的家中,但你的代码只会通过一次。

标签: arrays c sorting bubble-sort function-definition


【解决方案1】:

如果需要,您的代码会交换这两个元素,然后移动到数组中的下一个位置。在您的示例开始时,9 和 5 被交换,因此 5 最终位于第一个位置,但 5 永远不会与其他元素进行比较。

那里有很多排序算法,您应该阅读它们。与您的代码最相似的一种称为“冒泡排序”。要到达那里,请修改您的代码,使其多次遍历数组,直到数组完全排序。例如,如果您进行第二次遍历,则 5 和 2 将被交换,您将更接近排序结果。

冒泡排序不是最快的排序方式,但很容易理解。其他方法包括快速排序、合并排序和堆排序。 Wikipedia 可以解释其中的每一个,如果您需要更好的性能,它们值得研究。

【讨论】:

    【解决方案2】:

    您已经编写了冒泡排序的内部扫描循环,但您显然错过了算法的要点,因为这只是算法的一半。

    如果相邻元素相对于彼此无序(不是整个序列),则冒泡排序的每一次遍历都会交换相邻元素。当通过完成时,您可以保证极值(最大或最小,取决于您选择的比较器)在扫描分区的末端找到了它的位置。那时,不应再次访问它。下一次扫描运行到但不包括该元素。每次扫描都会使越来越多的元素更接近其正确的家,并且至少有一个(该分区长度的最后一个)在其永久家中)。作为一种优化,您可以跟踪当前扫描间隔是否实际交换了任何值。

    (可选)如果扫描完成时没有交换,您可以完全弹出;序列已排序。这个属性赋予了冒泡排序它唯一的可取之处。在最佳情况下,当使用交换检测时输入序列已经排序时,它是 O(n)。但它在一般情况下也是 O(n^2) 最坏的情况,因此一般来说它是一个糟糕的排序选择。

    不管

    #include <stdio.h>
    
    void ft_sort_int_tab(int *tab, int size) 
    {
        while (size-- > 0) // note descending ceiling
        {
            int swapped = 0;
    
            for (int i=0; i<size; ++i) // partition sweep 
            {
                if (tab[i] > tab[i + 1]) 
                {
                    int temp = tab[i];
                    tab[i] = tab[i + 1];
                    tab[i + 1] = temp;
                    swapped = 1;
                }
            }
    
            if (!swapped) // early exit on no-swap sweep
                break;
        }
    }
    
    int main(void) 
    {
        int tab[9] = {9, 5, 2, 3, 8, 4, 16, 20, 24};
        ft_sort_int_tab(tab, 9);
        
        for (int i = 0; i < 9; i++) 
            printf("%d ", tab[i]);
        fputc('\n', stdout);
    }
    

    输出

    2 3 4 5 8 9 16 20 24
    

    【讨论】:

      【解决方案3】:

      您似乎正在尝试实现冒泡排序算法。

      您的函数仅使用一个循环将最大元素移动到其目标位置。但是您需要对数组的所有其他元素重复该过程。

      该函数可以如下面的演示程序所示。

      #include <stdio.h>
      
      void ft_sort_int_tab( int *tab, size_t size )
      {
          for ( size_t last = size; !( size < 2 ); size = last )
          {
              for ( size_t i = last = 1; i < size; ++i )
              {
                  if ( tab[i] < tab[i-1] )
                  {
                      int tmp = tab[i];
                      tab[i] = tab[i - 1];
                      tab[i - 1] = tmp;
                      
                      last = i;
                  }
              }
          }
      }
          
      int main(void) 
      {
          int tab[] = { 9, 5, 2, 3, 8, 4, 16, 20, 24 };
          const size_t N = sizeof( tab ) / sizeof( *tab );
          
          for ( size_t i = 0; i < N; i++ )
          {
              printf( "%d ", tab[i] );
          }
          putchar( '\n' );
          
          ft_sort_int_tab( tab, N );
          
          for ( size_t i = 0; i < N; i++ )
          {
              printf( "%d ", tab[i] );
          }
          putchar( '\n' );
      
          return 0;
      }
      

      程序输出是

      9 5 2 3 8 4 16 20 24 
      2 3 4 5 8 9 16 20 24 
      

      请注意,通常数组中元素的数量应由size_t 类型的对象指定。它是运算符sizeof 返回的值的类型。 int 类型的对象可能不够大,无法存储数组中可能存在的元素数量。

      一种更通用的方法是向函数添加一个参数,该参数将指定对数组进行排序的标准。例如,通过这种方法,您可以使用相同的一个函数按升序或降序对数组进行排序。

      这是一个演示程序。

      #include <stdio.h>
      
      int ascending( int x, int y )
      {
          return x < y;
      }
      
      int descending( int x, int y )
      {
          return y < x;
      }
      
      void ft_sort_int_tab( int *tab, size_t size, int cmp( int, int ) )
      {
          for ( size_t last = size; !( size < 2 ); size = last )
          {
              for ( size_t i = last = 1; i < size; ++i )
              {
                  if ( cmp( tab[i], tab[i-1] ) )
                  {
                      int tmp = tab[i];
                      tab[i] = tab[i - 1];
                      tab[i - 1] = tmp;
                      
                      last = i;
                  }
              }
          }
      }
          
      int main(void) 
      {
          int tab[] = { 9, 5, 2, 3, 8, 4, 16, 20, 24 };
          const size_t N = sizeof( tab ) / sizeof( *tab );
          
          for ( size_t i = 0; i < N; i++ )
          {
              printf( "%d ", tab[i] );
          }
          putchar( '\n' );
          
          ft_sort_int_tab( tab, N, ascending );
          
          for ( size_t i = 0; i < N; i++ )
          {
              printf( "%d ", tab[i] );
          }
          putchar( '\n' );
      
          ft_sort_int_tab( tab, N, descending );
          
          for ( size_t i = 0; i < N; i++ )
          {
              printf( "%d ", tab[i] );
          }
          putchar( '\n' );
      
          return 0;
      }
      

      程序输出是

      9 5 2 3 8 4 16 20 24 
      2 3 4 5 8 9 16 20 24 
      24 20 16 9 8 5 4 3 2 
      

      【讨论】:

        猜你喜欢
        • 2018-08-01
        • 2016-02-01
        • 2019-04-14
        • 1970-01-01
        • 2022-07-30
        • 1970-01-01
        • 2018-08-11
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多