您好,我正在尝试对各种排序算法进行微基准测试,并且在使用jmh和快速排序进行基准测试时遇到了奇怪的问题。也许我的实现有问题。如果有人能帮助我看看问题出在哪里,我会很感兴趣。首先,我使用ubuntu 14.04和jdk 7以及jmh 0.9.1。以下是我尝试进行基准测试的方式:
@OutputTimeUnit(TimeUnit.MILLISECONDS)
@BenchmarkMode(Mode.AverageTime)
@Warmup(iterations = 3, time = 1)
@Measurement(iterations = 3, time = 1)
@State(Scope.Thread)
public class SortingBenchmark {
private int length = 100000;
private Distribution distribution = Distribution.RANDOM;
private int[] array;
int i = 1;
@Setup(Level.Iteration)
public void setUp() {
array = distribution.create(length);
}
@Benchmark
public int timeQuickSort() {
int[] sorted = Sorter.quickSort(array);
return sorted[i];
}
@Benchmark
public int timeJDKSort() {
Arrays.sort(array);
return array[i];
}
public static void main(String[] args) throws RunnerException {
Options opt = new OptionsBuilder().include(".*" + SortingBenchmark.class.getSimpleName() + ".*").forks(1)
.build();
new Runner(opt).run();
}
}
还有其他算法,但它们或多或少都可以。现在快速排序由于某种原因非常慢。时间的数量级更慢!而且,我需要为其分配更多栈空间,以便在没有StackOverflowException的情况下运行。看起来由于某种原因快速排序只是做了很多递归调用。有趣的是,当我在我的主类中简单地运行算法时 - 它运行良好(使用相同的随机分布和100000个元素)。不需要增加堆栈大小,并且简单的nanotime基准测试显示的时间非常接近其他算法。并且在基准测试JDK sort中,在使用jmh进行测试时非常快,并且在使用朴素的nanotime基准测试时与其他算法更加一致。我这里做错了什么或者漏掉了什么吗? 下面是我的快速排序算法:
public static int[] quickSort(int[] data) {
Sorter.quickSort(data, 0, data.length - 1);
return data;
}
private static void quickSort(int[] data, int sublistFirstIndex, int sublistLastIndex) {
if (sublistFirstIndex < sublistLastIndex) {
// move smaller elements before pivot and larger after
int pivotIndex = partition(data, sublistFirstIndex, sublistLastIndex);
// apply recursively to sub lists
Sorter.quickSort(data, sublistFirstIndex, pivotIndex - 1);
Sorter.quickSort(data, pivotIndex + 1, sublistLastIndex);
}
}
private static int partition(int[] data, int sublistFirstIndex, int sublistLastIndex) {
int pivotElement = data[sublistLastIndex];
int pivotIndex = sublistFirstIndex - 1;
for (int i = sublistFirstIndex; i < sublistLastIndex; i++) {
if (data[i] <= pivotElement) {
pivotIndex++;
ArrayUtils.swap(data, pivotIndex, i);
}
}
ArrayUtils.swap(data, pivotIndex + 1, sublistLastIndex);
return pivotIndex + 1; // return index of pivot element
}
现在我明白了,由于我的枢轴选择方式,如果我在已排序的数据上运行算法,它会非常慢(O(n^2))。但即使我尝试在我的主方法中运行已排序的数据,它仍然比在随机数据上使用jmh版本要快得多。我相信我错过了某些东西。您可以在此处找到其他算法的完整项目:https://github.com/ignl/SortingAlgos/