Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance. Project price only 1 $
You can buy this project and download/modify it how often you want.
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.flink.runtime.operators.sort;
public final class QuickSort implements IndexedSorter {
private static final IndexedSorter alt = new HeapSort();
public QuickSort() {
}
/**
* Fix the records into sorted order, swapping when the first record is
* greater than the second record.
*
* @param s paged sortable
* @param pN page number of first record
* @param pO page offset of first record
* @param rN page number of second record
* @param rO page offset of second record
*/
private static void fix(IndexedSortable s, int pN, int pO, int rN, int rO) {
if (s.compare(pN, pO, rN, rO) > 0) {
s.swap(pN, pO, rN, rO);
}
}
/**
* Deepest recursion before giving up and doing a heapsort.
* Returns 2 * ceil(log(n)).
*/
protected static int getMaxDepth(int x) {
if (x <= 0) {
throw new IllegalArgumentException("Undefined for " + x);
}
return (32 - Integer.numberOfLeadingZeros(x - 1)) << 2;
}
/**
* Sort the given range of items using quick sort. {@inheritDoc} If the recursion depth falls below
* {@link #getMaxDepth}, then switch to {@link HeapSort}.
*/
public void sort(final IndexedSortable s, int p, int r) {
int recordsPerSegment = s.recordsPerSegment();
int recordSize = s.recordSize();
int maxOffset = recordSize * (recordsPerSegment - 1);
int pN = p / recordsPerSegment;
int pO = (p % recordsPerSegment) * recordSize;
int rN = r / recordsPerSegment;
int rO = (r % recordsPerSegment) * recordSize;
sortInternal(s, recordsPerSegment, recordSize, maxOffset, p, pN, pO, r, rN, rO, getMaxDepth(r - p));
}
public void sort(IndexedSortable s) {
sort(s, 0, s.size());
}
/**
* Sort the given range of items using quick sort. If the recursion depth falls below
* {@link #getMaxDepth}, then switch to {@link HeapSort}.
*
* @param s paged sortable
* @param recordsPerSegment number of records per memory segment
* @param recordSize number of bytes per record
* @param maxOffset offset of a last record in a memory segment
* @param p index of first record in range
* @param pN page number of first record in range
* @param pO page offset of first record in range
* @param r index of last-plus-one'th record in range
* @param rN page number of last-plus-one'th record in range
* @param rO page offset of last-plus-one'th record in range
* @param depth recursion depth
*
* @see #sort(IndexedSortable, int, int)
*/
private static void sortInternal(final IndexedSortable s, int recordsPerSegment, int recordSize, int maxOffset,
int p, int pN, int pO, int r, int rN, int rO, int depth) {
while (true) {
if (r - p < 13) {
// switch to insertion sort
int i = p+1, iN, iO; if (pO == maxOffset) { iN = pN+1; iO = 0; } else { iN = pN; iO = pO+recordSize; }
while (i < r) {
int j = i, jN = iN, jO = iO;
int jd = j-1, jdN, jdO; if (jO == 0) { jdN = jN-1; jdO = maxOffset; } else { jdN = jN; jdO = jO-recordSize; }
while (j > p && s.compare(jdN, jdO, jN, jO) > 0) {
s.swap(jN, jO, jdN, jdO);
j = jd; jN = jdN; jO = jdO;
jd--; if (jdO == 0) { jdN--; jdO = maxOffset; } else { jdO -= recordSize; }
}
i++; if (iO == maxOffset) { iN++; iO = 0; } else { iO += recordSize; }
}
return;
}
if (--depth < 0) {
// switch to heap sort
alt.sort(s, p, r);
return;
}
int rdN, rdO; if (rO == 0) { rdN = rN-1; rdO = maxOffset; } else { rdN = rN; rdO = rO-recordSize; }
int m = (p+r)>>>1, mN = m / recordsPerSegment, mO = (m % recordsPerSegment) * recordSize;
// select, move pivot into first position
fix(s, mN, mO, pN, pO);
fix(s, mN, mO, rdN, rdO);
fix(s, pN, pO, rdN, rdO);
// Divide
int i = p, iN = pN, iO = pO;
int j = r, jN = rN, jO = rO;
int ll = p, llN = pN, llO = pO;
int rr = r, rrN = rN, rrO = rO;
int cr;
while (true) {
i++; if (iO == maxOffset) { iN++; iO = 0; } else { iO += recordSize; }
while (i < j) {
if ((cr = s.compare(iN, iO, pN, pO)) > 0) {
break;
}
if (0 == cr) {
ll++; if (llO == maxOffset) { llN++; llO = 0; } else { llO += recordSize; }
if (ll != i) {
s.swap(llN, llO, iN, iO);
}
}
i++; if (iO == maxOffset) { iN++; iO = 0; } else { iO += recordSize; }
}
j--; if (jO == 0) { jN--; jO = maxOffset; } else { jO -= recordSize; }
while (j > i) {
if ((cr = s.compare(pN, pO, jN, jO)) > 0) {
break;
}
if (0 == cr) {
rr--; if (rrO == 0) { rrN--; rrO = maxOffset; } else { rrO -= recordSize; }
if (rr != j) {
s.swap(rrN, rrO, jN, jO);
}
}
j--; if (jO == 0) { jN--; jO = maxOffset; } else { jO -= recordSize; }
}
if (i < j) {
s.swap(iN, iO, jN, jO);
} else {
break;
}
}
j = i; jN = iN; jO = iO;
// swap pivot- and all eq values- into position
while (ll >= p) {
i--; if (iO == 0) { iN--; iO = maxOffset; } else { iO -= recordSize; }
s.swap(llN, llO, iN, iO);
ll--; if (llO == 0) { llN--; llO = maxOffset; } else { llO -= recordSize; }
}
while (rr < r) {
s.swap(rrN, rrO, jN, jO);
rr++; if (rrO == maxOffset) { rrN++; rrO = 0; } else { rrO += recordSize; }
j++; if (jO == maxOffset) { jN++; jO = 0; } else { jO += recordSize; }
}
// Conquer
// Recurse on smaller interval first to keep stack shallow
assert i != j;
if (i - p < r - j) {
sortInternal(s, recordsPerSegment, recordSize, maxOffset, p, pN, pO, i, iN, iO, depth);
p = j; pN = jN; pO = jO;
} else {
sortInternal(s, recordsPerSegment, recordSize, maxOffset, j, jN, jO, r, rN, rO, depth);
r = i; rN = iN; rO = iO;
}
}
}
}