mirror of
https://github.com/gnustep/libs-base.git
synced 2025-04-26 02:01:03 +00:00
interface that can be used for all sorting tasks in the library. The actual sort algorithms to use are now plugable. Timsort is the new default sorting algorithm, the existing algorithms, shellsort and quicksort, can still be selected using a configure switch. Also implement the new NSComparator (blocks) based sorting and insertion index searching methods for NSMutableArray and NSArray. git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@35573 72102866-910b-0410-8b05-ffd578937521
125 lines
3 KiB
Objective-C
125 lines
3 KiB
Objective-C
/* Implementation of ShellSort for GNUStep
|
|
Copyright (C) 1995-2012 Free Software Foundation, Inc.
|
|
|
|
Written by: Richard Frith-Macdonald <rfm@gnu.org>
|
|
|
|
Modified by: Niels Grewe <niels.grewe@halbordnung.de>
|
|
Date: September 2012
|
|
|
|
This file is part of the GNUstep Base Library.
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with this library; if not, write to the Free
|
|
Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
Boston, MA 02111 USA.
|
|
*/
|
|
|
|
|
|
|
|
#import "Foundation/NSSortDescriptor.h"
|
|
#import "Foundation/NSArray.h"
|
|
#import "Foundation/NSObjCRuntime.h"
|
|
#import "GSSorting.h"
|
|
|
|
#ifndef GS_DISABLE_SHELLSORT
|
|
void
|
|
_GSShellSort(id *objects,
|
|
NSRange sortRange,
|
|
id comparisonEntity,
|
|
GSComparisonType type,
|
|
void *context)
|
|
{
|
|
/* Shell sort algorithm taken from SortingInAction - a NeXT example */
|
|
#define STRIDE_FACTOR 3 // good value for stride factor is not well-understood
|
|
// 3 is a fairly good choice (Sedgewick)
|
|
NSUInteger c;
|
|
NSUInteger d;
|
|
NSUInteger stride = 1;
|
|
BOOL found;
|
|
NSUInteger count = NSMaxRange(sortRange);
|
|
#ifdef GSWARN
|
|
BOOL badComparison = NO;
|
|
#endif
|
|
|
|
while (stride <= count)
|
|
{
|
|
stride = stride * STRIDE_FACTOR + 1;
|
|
}
|
|
|
|
while (stride > (STRIDE_FACTOR - 1))
|
|
{
|
|
// loop to sort for each value of stride
|
|
stride = stride / STRIDE_FACTOR;
|
|
for (c = (sortRange.location + stride); c < count; c++)
|
|
{
|
|
found = NO;
|
|
if (stride > c)
|
|
{
|
|
break;
|
|
}
|
|
d = c - stride;
|
|
while (!found) /* move to left until correct place */
|
|
{
|
|
id a = objects[d + stride];
|
|
id b = objects[d];
|
|
NSComparisonResult r;
|
|
r = GSCompareUsingDescriptorOrComparator(a, b, comparisonEntity, context);
|
|
if (r < 0)
|
|
{
|
|
#ifdef GSWARN
|
|
if (r != NSOrderedAscending)
|
|
{
|
|
badComparison = YES;
|
|
}
|
|
#endif
|
|
objects[d+stride] = b;
|
|
objects[d] = a;
|
|
if (stride > d)
|
|
{
|
|
break;
|
|
}
|
|
d -= stride; // jump by stride factor
|
|
}
|
|
else
|
|
{
|
|
#ifdef GSWARN
|
|
if (r != NSOrderedDescending && r != NSOrderedSame)
|
|
{
|
|
badComparison = YES;
|
|
}
|
|
#endif
|
|
found = YES;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#ifdef GSWARN
|
|
if (badComparison == YES)
|
|
{
|
|
NSWarnFLog(@"Detected bad return value from comparison");
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
@interface GSShellSortPlaceHolder : NSObject
|
|
|
|
@end
|
|
|
|
@implementation GSShellSortPlaceHolder
|
|
+ (void)load
|
|
{
|
|
_GSSortUnstable = _GSShellSort;
|
|
}
|
|
@end
|
|
#endif
|