Skip to content

Commit

Permalink
Merge branch 'main' into tri
Browse files Browse the repository at this point in the history
  • Loading branch information
Lampese authored Mar 25, 2024
2 parents 73859ad + 605f4a5 commit f479b0f
Show file tree
Hide file tree
Showing 6 changed files with 100 additions and 88 deletions.
6 changes: 2 additions & 4 deletions array/moon.pkg.json
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
{
"import": [
"moonbitlang/core/assertion"
]
}
"import": ["moonbitlang/core/assertion", "moonbitlang/core/math"]
}
47 changes: 47 additions & 0 deletions array/slice.mbt
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
// Copyright 2024 International Digital Economy Academy
//
// Licensed 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.

priv struct ArraySlice[T] {
array : Array[T]
start : Int
end : Int
}

fn length[T](self : ArraySlice[T]) -> Int {
self.end - self.start
}

fn op_get[T](self : ArraySlice[T], index : Int) -> T {
self.array[self.start + index]
}

fn op_set[T](self : ArraySlice[T], index : Int, value : T) -> Unit {
self.array[self.start + index] = value
}

fn swap[T](self : ArraySlice[T], a : Int, b : Int) -> Unit {
self.array.swap(self.start + a, self.start + b)
}

fn reverse[T](self : ArraySlice[T]) -> Unit {
let mid_len = self.length() / 2
for i = 0; i < mid_len; i = i + 1 {
let j = self.length() - i - 1
self.swap(i, j)
}
}

fn slice[T](self : ArraySlice[T], start : Int, end : Int) -> ArraySlice[T] {
{ array: self.array, start: self.start + start, end: self.start + end }
}
61 changes: 8 additions & 53 deletions array/sort.mbt
Original file line number Diff line number Diff line change
Expand Up @@ -12,40 +12,6 @@
// See the License for the specific language governing permissions and
// limitations under the License.

struct ArraySlice[T] {
array : Array[T]
start : Int
end : Int
}

fn length[T](self : ArraySlice[T]) -> Int {
self.end - self.start
}

fn op_get[T](self : ArraySlice[T], index : Int) -> T {
self.array[self.start + index]
}

fn op_set[T](self : ArraySlice[T], index : Int, value : T) -> Unit {
self.array[self.start + index] = value
}

fn swap[T](self : ArraySlice[T], a : Int, b : Int) -> Unit {
self.array.swap(self.start + a, self.start + b)
}

fn reverse[T](self : ArraySlice[T]) -> Unit {
let mid_len = self.length() / 2
for i = 0; i < mid_len; i = i + 1 {
let j = self.length() - i - 1
self.swap(i, j)
}
}

fn slice[T](self : ArraySlice[T], start : Int, end : Int) -> ArraySlice[T] {
{ array: self.array, start: self.start + start, end: self.start + end }
}

/// Sorts the array
///
/// It's an in-place, unstable sort(it will reorder equal elements). The time complexity is O(n log n) in the worst case.
Expand Down Expand Up @@ -141,7 +107,7 @@ fn quick_sort[T](
}
let (pivot, partitioned) = partition(arr, cmp, pivot_index)
was_partitioned = partitioned
balanced = min(pivot, len - pivot) >= len / 8
balanced = @math.minimum(pivot, len - pivot) >= len / 8
if not(balanced) {
limit -= 1
}
Expand Down Expand Up @@ -183,29 +149,22 @@ fn get_limit(len : Int) -> Int {
limit
}

fn min[T : Compare](a : T, b : T) -> T {
if a < b {
a
} else {
b
}
}

/// Try to sort the array with bubble sort.
///
/// It will only tolerate at most 8 unsorted elements. The time complexity is O(n).
///
/// Returns whether the array is sorted.
fn try_bubble_sort[T](arr : ArraySlice[T], cmp : (T, T) -> Int) -> Bool {
let max_tries = 8
fn try_bubble_sort[T](
arr : ArraySlice[T],
cmp : (T, T) -> Int,
~max_tries : Int = 8
) -> Bool {
let mut tries = 0
for i = 1; i < arr.length(); i = i + 1 {
let mut j = i
let mut sorted = true
while j > 0 && cmp(arr[j - 1], arr[j]) > 0 {
for j = i; j > 0 && cmp(arr[j - 1], arr[j]) > 0; j = j - 1 {
sorted = false
arr.swap(j, j - 1)
j = j - 1
}
if not(sorted) {
tries += 1
Expand All @@ -224,12 +183,8 @@ fn try_bubble_sort[T](arr : ArraySlice[T], cmp : (T, T) -> Int) -> Bool {
/// Returns whether the array is sorted.
fn bubble_sort[T](arr : ArraySlice[T], cmp : (T, T) -> Int) -> Unit {
for i = 1; i < arr.length(); i = i + 1 {
let mut j = i
let mut sorted = true
while j > 0 && cmp(arr[j - 1], arr[j]) > 0 {
sorted = false
for j = i; j > 0 && cmp(arr[j - 1], arr[j]) > 0; j = j - 1 {
arr.swap(j, j - 1)
j = j - 1
}
}
}
Expand Down
39 changes: 13 additions & 26 deletions priority_queue/priority_queue.mbt
Original file line number Diff line number Diff line change
Expand Up @@ -12,22 +12,7 @@
// See the License for the specific language governing permissions and
// limitations under the License.

struct Cont[T] {
content : T
mut sibling : Node[T]
mut child : Node[T]
}

enum Node[T] {
Nil
Cons(Cont[T])
}

struct PriorityQueue[T] {
min : Bool
mut len : Int
mut top : Node[T]
}

/// Creates a new empty priority queue.
///
Expand Down Expand Up @@ -66,11 +51,15 @@ pub fn PriorityQueue::from_array[T : Compare](
arr : Array[T],
~min : Bool = false
) -> PriorityQueue[T] {
for i = 0, acc = (Nil : Node[T]); i < arr.length(); {
continue i + 1,
meld(acc, Cons({ content: arr[i], sibling: Nil, child: Nil }), min)
} else {
{ min, len: arr.length(), top: acc }
// CR: bad formatting
let len = arr.length()
for i = 0, acc = Node::Nil {
if i < len {
continue i + 1,
meld(acc, Cons({ content: arr[i], sibling: Nil, child: Nil }), min)
} else {
break { min, len, top: acc }
}
}
}

Expand All @@ -84,11 +73,12 @@ fn meld[T : Compare](x : Node[T], y : Node[T], min : Bool) -> Node[T] {
(Nil, _) => y
(_, Nil) => x
(Cons(x_top), Cons(y_top)) =>
if if min {
// CR: bad formatting
if (if min {
x_top.content < y_top.content
} else {
x_top.content > y_top.content
} {
}) {
y_top.sibling = x_top.child
x_top.child = y
x
Expand Down Expand Up @@ -277,10 +267,7 @@ test "clear" {
/// let is_empty = queue.is_empty() // true
/// ```
pub fn is_empty[T](self : PriorityQueue[T]) -> Bool {
match self.top {
Nil => true
_ => false
}
self.len == 0
}

test "is_empty" {
Expand Down
26 changes: 26 additions & 0 deletions priority_queue/types.mbt
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
// 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.

priv struct Cont[T] {
content : T
mut sibling : Node[T]
mut child : Node[T]
}

priv enum Node[T] {
Nil
Cons(Cont[T])
}

struct PriorityQueue[T] {
min : Bool
mut len : Int
mut top : Node[T]
}
9 changes: 4 additions & 5 deletions vec/vec.mbt
Original file line number Diff line number Diff line change
Expand Up @@ -35,13 +35,12 @@ pub fn to_string[T : Show](self : Vec[T]) -> String {
return "Vec::[]"
}
let first = self.buf[0]
for i = 1, init = "Vec::[\(first)"; ; {
if i < self.len {
let cur = self.buf[i]
continue i + 1, "\(init),\(cur)"
} else {
for i = 1, init = "Vec::[\(first)" {
if i >= self.len {
break "\(init)]"
}
let cur = self.buf[i]
continue i + 1, "\(init),\(cur)"
}
}

Expand Down

0 comments on commit f479b0f

Please sign in to comment.