C#

时间:2017-12-26 17:58:37

标签: c# sorting mergesort

我正在尝试按照Wikipedia中的描述在C#中实现自上而下的合并排序算法。以下实现是我提出的,然而,它似乎没有正确地对数组进行排序。我已经通过调试器几次但无法弄清楚代码出错的地方。任何帮助将不胜感激。

using System;

class MergeSort
{
    static void Main()
    {
        int[] A = { 4, 3, 2, 1 };
        int lastIndex = A.Length - 1;
        int[] B = new int[A.Length];
        Array.Copy(A, B, A.Length);
        MergeSortNumbers(B, 0, lastIndex, A);
        foreach (var item in A)
        {
            Console.WriteLine("{0} ", item);
        }
    }

    private static void MergeSortNumbers(int[] B, int iStart, int iEnd, int[] A)
    {
        int iMiddle = (iStart + iEnd) / 2;
        if ((iEnd - iStart) < 2)
        {
            // Merge(B, iStart, iMiddle, iEnd, A);
            return;
        }
        MergeSortNumbers(A, iStart, iMiddle - 1, B);
        MergeSortNumbers(A, iMiddle, iEnd, B);
        Merge(B, iStart, iMiddle, iEnd, A);
    }

    private static void Merge(int[] A, int iStart, int iMiddle, int iEnd, int[] B)
    {
        int i = iStart;
        int j = iMiddle;

        for (int k = iStart; k < iEnd; k++)
        {
            if (i < iMiddle && (j >= iEnd || A[i] <= A[j]))
            {
                B[k] = A[i];
                i = i + 1;
            }
            else
            {
                B[k] = A[j];
                j = j + 1;
            }
        }
    }
}

1 个答案:

答案 0 :(得分:2)

有两个问题。你应该知道iEnd是独家的。这意味着iEnd它自己不会在索引中考虑。这是因为Merge方法中的j >= iEnd条件和MergeSortNumbers方法中的(iEnd - iStart) < 2

iMiddle也是左侧专有的。因为Merge方法中的i < iMiddle条件。

所以基本上没有递减1.(如果结束是包含的话,你只减1)

static void Main()
{
    int[] A = { 4, 3, 2, 1 };
    int[] B = new int[A.Length];
    Array.Copy(A, B, A.Length);
    MergeSortNumbers(B, 0, A.Length, A); // Do not decrement A.Length
    foreach (var item in A)
    {
        Console.WriteLine("{0} ", item);
    }
}

private static void MergeSortNumbers(int[] B, int iStart, int iEnd, int[] A)
{
    if ((iEnd - iStart) < 2) return;

    int iMiddle = (iStart + iEnd) / 2;
    MergeSortNumbers(A, iStart, iMiddle, B); // Do Not decrement iMiddle
    MergeSortNumbers(A, iMiddle, iEnd, B);
    Merge(B, iStart, iMiddle, iEnd, A);
}

private static void Merge(int[] A, int iStart, int iMiddle, int iEnd, int[] B)
{
    int i = iStart;
    int j = iMiddle;

    for (int k = iStart; k < iEnd; k++)
    {
        if (i < iMiddle && (j >= iEnd || A[i] <= A[j]))
        {
            B[k] = A[i++];
        }
        else
        {
            B[k] = A[j++];
        }
    }
}