有没有更好的方法使用C#进行shell排序?
// array of integers to hold values
private int[] a = new int[100];
// number of elements in array
private int count;
// Shell Sort Algorithm
public void sortArray()
{
int i, j, increment, temp;
increment = 3;
while( increment > 0 )
{
for( i=0; i < count; i++ )
{
j = i;
temp = a[i];
while( (j >= increment) && (a[j-increment] > temp) )
{
a[j] = a[j - increment];
j = j - increment;
}
a[j] = temp;
}
if( increment/2 != 0 )
{
increment = increment/2;
}
else if( increment == 1 )
{
increment = 0;
}
else
{
increment = 1;
}
}
}
顺便说一句,我很想知道,因为我在不同的语言中有一些不同的“优雅”类型的例子(比如C#和F#中的冒泡排序),我正在比较它们。在现实生活中,我可能会在C#中大部分时间使用以下内容:
Array.Sort( object[] )
我不在乎这些是“学术性的”和非实用的模式。如果你愿意,你可以将我降格为遗忘:)
KA
答案 0 :(得分:2)
您可以非常轻松地进行改进:
ShellSort
而不是shellSort
count
,而不是x
使用条件运算符,例如
// This replaces your last 12 lines
int halfIncrement = increment / 2;
increment = halfIncrement != 0 ? halfIncrement : 1 - increment;
使代码通用 - 为什么要限制自己使用整数?
IList<T>
IComparer<T>
使排序顺序任意,提供使用默认值的重载。这实际上与排序无关 - 我还没有验证你的代码是否真的是合法的shell排序......
答案 1 :(得分:1)
在我的测试中,这比使用相同System.Func比较器的数组排序快75%到90%。我用它来排序自定义结构。您可以轻松修改它以对类进行排序。
public class DualQuickSort<T> where T : struct
{
private readonly System.Func<T, T, int> comparer;
public DualQuickSort(System.Func<T, T, int> comparer)
{
this.comparer = comparer;
}
public DualQuickSort(IComparer<T> comparer)
: this(comparer.Compare)
{
}
public void Sort(T[] a)
{
Sort(a, 0, a.Length);
}
public void Sort(T[] a, int fromIndex, int toIndex)
{
RangeCheck(a.Length, fromIndex, toIndex);
DualPivotQuicksort(a, fromIndex, toIndex - 1, 3);
}
private static void RangeCheck(int length, int fromIndex, int toIndex)
{
if (fromIndex > toIndex)
{
throw new ArgumentException("fromIndex > toIndex");
}
if (fromIndex < 0)
{
throw new IndexOutOfRangeException(fromIndex + " is less than 0");
}
if (toIndex > length)
{
throw new IndexOutOfRangeException(toIndex + " is greater than " + fromIndex);
}
}
private static void Swap(T[] a, int i, int j)
{
var temp = a[i];
a[i] = a[j];
a[j] = temp;
}
private void DualPivotQuicksort(T[] a, int left, int right, int div)
{
var len = right - left;
if (len < 27)
{ // insertion sort for tiny array
for (var i = left + 1; i <= right; i++)
{
for (var j = i; j > left && comparer(a[j] , a[j - 1])==-1; j--)
{
Swap(a, j, j - 1);
}
}
return;
}
var third = len / div;
// "medians"
var m1 = left + third;
var m2 = right - third;
if (m1 <= left)
{
m1 = left + 1;
}
if (m2 >= right)
{
m2 = right - 1;
}
if (comparer(a[m1] , a[m2])==-1)
{
Swap(a, m1, left);
Swap(a, m2, right);
}
else
{
Swap(a, m1, right);
Swap(a, m2, left);
}
// pivots
var pivot1 = a[left];
var pivot2 = a[right];
// pointers
var less = left + 1;
var great = right - 1;
// sorting
for (var k = less; k <= great; k++)
{
if (comparer(a[k] , pivot1)==-1)
{
Swap(a, k, less++);
}
else if (comparer(a[k], pivot2) == 1)
{
while (k < great && comparer(a[great] , pivot2)==1)
{
great--;
}
Swap(a, k, great--);
if (comparer(a[k], pivot1) == -1)
{
Swap(a, k, less++);
}
}
}
// Swaps
var dist = great - less;
if (dist < 13)
{
div++;
}
Swap(a, less - 1, left);
Swap(a, great + 1, right);
// subarrays
DualPivotQuicksort(a, left, less - 2, div);
DualPivotQuicksort(a, great + 2, right, div);
// equal elements
if (dist > len - 13 && comparer(pivot1,pivot2)!=0)
{
for (int k = less; k <= great; k++)
{
if (comparer(a[k] , pivot1)==0)
{
Swap(a, k, less++);
}
else if (comparer(a[k], pivot2) == 0)
{
Swap(a, k, great--);
if (comparer(a[k], pivot1) == 0)
{
Swap(a, k, less++);
}
}
}
}
// subarray
if (comparer(pivot1 , pivot2)==-1)
{
DualPivotQuicksort(a, less, great, div);
}
}
}
答案 2 :(得分:0)
请参考以下代码。它会自动计算距离,并且将考虑最多生成三个列表来计算距离。
using System;
namespace ShellSortExample
{
class Program
{
static void Main(string[] args)
{
int[] num = { 70, 30, 40, 10, 80, 20, 90, 110, 75, 60, 45, 199, 35, 2, 31, 22, 1, 11, 211, 100, 23, 55, 50 };
WriteList("Unsorted List", num);
int distance = 0;
// Generate the distance as per the length of the list
//It will consider maximum 3 list should generate
for (int i = 1; i < num.Length; i++)
{
var div = num.Length / i;
// Maximum 3 list should be generate
if (div <= 3)
{
distance = i;
break;
}
}
// Do the grouping and Apply insertion sort
for (int i = distance; i > 0; i--)
{
int tmpListVal = 0;
for (int j = i; j < num.Length;)
{
int tmpVal = num[j];
//Applying insertion sort with each shell/group element
for (int k = j - i; k >= 0; k -= i)
{
if (k <= num.Length - 1 && tmpVal < num[k])
{
num[k + i] = num[k];
num[k] = tmpVal;
}
}
if (j + i >= num.Length)
{
j = tmpListVal + 1;
tmpListVal = j;
if (j == i)
{
break;
}
}
else
{
j += i;
}
}
}
WriteList("After applied shell sort",num);
}
static void WriteList(string msg, int[] list)
{
Console.WriteLine($"{msg} : ");
for (int i = 0; i < list.Length; i++)
{
Console.Write($" {list[i]}");
}
Console.Write("\r\n");
}
}
}