Commit cf00245bf9

Steve Perkins <stereosteve@gmail.com>
2016-11-03 05:10:33
sort requires compare function.
Provide some handy functions for builtin comparable types.
1 parent e761aa2
Changed files (1)
std/sort.zig
@@ -4,22 +4,22 @@ const math = @import("math.zig");
 
 pub const Cmp = math.Cmp;
 
-pub fn sort(inline T: type, array: []T) {
+pub fn sort(inline T: type, array: []T, inline cmp: fn(a: T, b: T)->Cmp) {
     if (array.len > 0) {
-        quicksort(T, array, 0, array.len - 1);
+        quicksort(T, array, 0, array.len - 1, cmp);
     }
 }
 
-fn quicksort(inline T: type, array: []T, left: usize, right: usize) {
+fn quicksort(inline T: type, array: []T, left: usize, right: usize, inline cmp: fn(a: T, b: T)->Cmp) {
     var i = left;
     var j = right;
     var p = (i + j) / 2;
 
     while (i <= j) {
-        while (array[i] < array[p]) {
+        while (cmp(array[i], array[p]) == Cmp.Less) {
             i += 1;
         }
-        while (array[j] > array[p]) {
+        while (cmp(array[j], array[p]) == Cmp.Greater) {
             j -= 1;
         }
         if (i <= j) {
@@ -31,42 +31,28 @@ fn quicksort(inline T: type, array: []T, left: usize, right: usize) {
         }
     }
 
-    if (left < j) quicksort(T, array, left, j);
-    if (i < right) quicksort(T, array, i, right);
+    if (left < j) quicksort(T, array, left, j, cmp);
+    if (i < right) quicksort(T, array, i, right, cmp);
 }
 
-// ---------------------------------------
-// sortCmp
+pub fn i32asc(a: i32, b: i32) -> Cmp {
+    return if (a > b) Cmp.Greater else if (a < b) Cmp.Less else Cmp.Equal;
+}
 
-pub fn sortCmp(inline T: type, array: []T, inline cmp: fn(a: T, b: T)->Cmp) {
-    if (array.len > 0) {
-        quicksortCmp(T, array, 0, array.len - 1, cmp);
-    }
+pub fn i32desc(a: i32, b: i32) -> Cmp {
+    return reverse(i32asc(a, b));
 }
 
-fn quicksortCmp(inline T: type, array: []T, left: usize, right: usize, inline cmp: fn(a: T, b: T)->Cmp) {
-    var i = left;
-    var j = right;
-    var p = (i + j) / 2;
+pub fn u8asc(a: u8, b: u8) -> Cmp {
+    return if (a > b) Cmp.Greater else if (a < b) Cmp.Less else Cmp.Equal;
+}
 
-    while (i <= j) {
-        while (cmp(array[i], array[p]) == Cmp.Less) {
-            i += 1;
-        }
-        while (cmp(array[j], array[p]) == Cmp.Greater) {
-            j -= 1;
-        }
-        if (i <= j) {
-            const tmp = array[i];
-            array[i] = array[j];
-            array[j] = tmp;
-            i += 1;
-            if (j > 0) j -= 1;
-        }
-    }
+pub fn u8desc(a: u8, b: u8) -> Cmp {
+    return reverse(u8asc(a, b));
+}
 
-    if (left < j) quicksortCmp(T, array, left, j, cmp);
-    if (i < right) quicksortCmp(T, array, i, right, cmp);
+fn reverse(was: Cmp) -> Cmp {
+    return if (was == Cmp.Greater) Cmp.Less else if (was == Cmp.Less) Cmp.Greater else Cmp.Equal;
 }
 
 // ---------------------------------------
@@ -85,7 +71,7 @@ fn testSort() {
     };
 
     for (u8cases) |case| {
-        sort(u8, case[0]);
+        sort(u8, case[0], u8asc);
         assert(str.eql(case[0], case[1]));
     }
 
@@ -99,28 +85,14 @@ fn testSort() {
     };
 
     for (i32cases) |case| {
-        sort(i32, case[0]);
+        sort(i32, case[0], i32asc);
         assert(str.sliceEql(i32, case[0], case[1]));
     }
 }
 
-fn testSortCmp() {
+fn testSortDesc() {
     @setFnTest(this, true);
 
-    const i32cases = [][][]i32 {
-        [][]i32{[]i32{}, []i32{}},
-        [][]i32{[]i32{1}, []i32{1}},
-        [][]i32{[]i32{0, 1}, []i32{0, 1}},
-        [][]i32{[]i32{1, 0}, []i32{0, 1}},
-        [][]i32{[]i32{1, -1, 0}, []i32{-1, 0, 1}},
-        [][]i32{[]i32{2, 1, 3}, []i32{1, 2, 3}},
-    };
-
-    for (i32cases) |case| {
-        sortCmp(i32, case[0], normalCmp);
-        assert(str.sliceEql(i32, case[0], case[1]));
-    }
-
     const revCases = [][][]i32 {
         [][]i32{[]i32{}, []i32{}},
         [][]i32{[]i32{1}, []i32{1}},
@@ -131,16 +103,9 @@ fn testSortCmp() {
     };
 
     for (revCases) |case| {
-        sortCmp(i32, case[0], revCmp);
+        sort(i32, case[0], i32desc);
         assert(str.sliceEql(i32, case[0], case[1]));
     }
 
 }
 
-fn normalCmp(a: i32, b: i32) -> Cmp {
-    return if (a > b) Cmp.Greater else if (a < b) Cmp.Less else Cmp.Equal;
-}
-
-fn revCmp(a: i32, b: i32) -> Cmp {
-    return if (a < b) Cmp.Greater else if (a > b) Cmp.Less else Cmp.Equal;
-}