Skip to content

Commit 5260991

Browse files
committed
Expose an interface for the stack package
1 parent d85364f commit 5260991

10 files changed

+340
-280
lines changed

structure/stack/stack_array.go structure/stack/array.go

+13-17
Original file line numberDiff line numberDiff line change
@@ -11,24 +11,22 @@ import "errors"
1111

1212
var ErrStackEmpty = errors.New("stack is empty")
1313

14-
/*
15-
The methods can also be implemented directly on the slice.
16-
```
17-
type Array[T any] []T
18-
```
19-
However, this exposes the underlaying storage (slice) outside the package.
20-
A struct is used instead, so that the underlying storage is not accessible outside the package.
21-
*/
22-
2314
// Array is an implementation of stack with slice as underlying storage.
2415
// ```
2516
// stack := stack.NewArray[int]()
2617
// ```
18+
// Note that the type `Array` could also be implemented directly using a slice.
19+
// ```
20+
// type Array[T any] []T
21+
// ```
22+
// However, this exposes the underlying storage (slice) outside the package.
23+
// A struct is used instead, so that the underlying storage is not accessible
24+
// outside the package.
2725
type Array[T any] struct {
2826
store []T
2927
}
3028

31-
func NewArray[T any]() *Array[T] {
29+
func NewArray[T any]() Interface[T] {
3230
return new(Array[T])
3331
}
3432

@@ -59,7 +57,7 @@ func (s *Array[T]) Empty() bool {
5957
return s.Len() == 0
6058
}
6159

62-
// Pop returns last inserted element and removes it from the underlaying storage
60+
// Pop returns last inserted element and removes it from the underlying storage
6361
// If the stack is empty, ErrStackEmpty error is returned
6462
func (s *Array[T]) Pop() (T, error) {
6563
var element T
@@ -80,10 +78,8 @@ func (s *Array[T]) Clear() {
8078
s.store = s.store[:0]
8179
}
8280

83-
// Truncate removes all elements and underlaying storage
84-
func (s *Array[T]) Truncate() {
85-
if s == nil {
86-
return
87-
}
88-
s.store = nil
81+
func (s *Array[T]) ToSlice() []T {
82+
out := make([]T, len(s.store))
83+
copy(out, s.store)
84+
return out
8985
}

structure/stack/stack_array_test.go structure/stack/array_test.go

+1-7
Original file line numberDiff line numberDiff line change
@@ -40,12 +40,6 @@ func Test_StackArray(t *testing.T) {
4040

4141
stack.Clear()
4242
if stack.Len() != 0 && !stack.Empty() {
43-
t.Errorf("Expected stack to be emtpy after Clear. Got len=%d, empty=%t", stack.Len(), stack.Empty())
44-
}
45-
46-
stack.Truncate()
47-
storeCapacity := cap(stack.store)
48-
if storeCapacity != 0 {
49-
t.Errorf("Expected store capacity to be zero after truncate. Got capacity=%d", storeCapacity)
43+
t.Errorf("Expected stack to be empty after Clear. Got len=%d, empty=%t", stack.Len(), stack.Empty())
5044
}
5145
}

structure/stack/doubly_linked_list.go

+93
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
package stack
2+
3+
import (
4+
"container/list"
5+
)
6+
7+
// doublyLinkedList is an implementation of stack.Interface using the doubly linked list provided by `container/list` as its underlying storage.
8+
type doublyLinkedList[T any] struct {
9+
stack *list.List
10+
}
11+
12+
func NewDoublyLinkedList[T any]() Interface[T] {
13+
return &doublyLinkedList[T]{
14+
stack: list.New(),
15+
}
16+
}
17+
18+
// Push add a value into our stack
19+
func (dl *doublyLinkedList[T]) Push(val T) {
20+
dl.stack.PushFront(val)
21+
}
22+
23+
// Peek return last inserted element(top of the stack) without removing it from the stack
24+
// If the stack is empty, ErrStackEmpty error is returned
25+
func (dl *doublyLinkedList[T]) Peek() (T, error) {
26+
var result T
27+
if dl.Empty() {
28+
return result, ErrStackEmpty
29+
}
30+
31+
element := dl.stack.Front()
32+
if element == nil {
33+
return result, ErrStackEmpty
34+
}
35+
36+
result = element.Value.(T)
37+
return result, nil
38+
}
39+
40+
// Pop is return last value that insert into our stack
41+
// also it will remove it in our stack
42+
func (dl *doublyLinkedList[T]) Pop() (T, error) {
43+
var result T
44+
if dl.Empty() {
45+
return result, ErrStackEmpty
46+
}
47+
48+
element := dl.stack.Front()
49+
if element == nil {
50+
return result, ErrStackEmpty
51+
}
52+
53+
dl.stack.Remove(element)
54+
result = element.Value.(T)
55+
return result, nil
56+
}
57+
58+
// Length returns the number of elements in the stack
59+
func (dl *doublyLinkedList[T]) Len() int {
60+
if dl == nil {
61+
return 0
62+
}
63+
return dl.stack.Len()
64+
}
65+
66+
// Empty returns true if stack has no elements and false otherwise.
67+
func (dl *doublyLinkedList[T]) Empty() bool {
68+
if dl == nil {
69+
return true
70+
}
71+
return dl.stack.Len() == 0
72+
}
73+
74+
// Clear initializes the underlying storage with a new empty doubly linked list, thus clearing the underlying storage.
75+
func (dl *doublyLinkedList[T]) Clear() {
76+
if dl == nil {
77+
return
78+
}
79+
dl.stack = list.New()
80+
}
81+
82+
// ToSlice returns the elements of stack as a slice
83+
func (dl *doublyLinkedList[T]) ToSlice() []T {
84+
var result []T
85+
if dl == nil {
86+
return result
87+
}
88+
89+
for e := dl.stack.Front(); e != nil; e = e.Next() {
90+
result = append(result, e.Value.(T))
91+
}
92+
return result
93+
}
+66
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
// Stack Test
2+
// description: based on `geeksforgeeks` description Stack is a linear data structure which follows a particular order in which the operations are performed.
3+
// The order may be LIFO(Last In First Out) or FILO(First In Last Out).
4+
// details:
5+
// Stack Data Structure : https://www.geeksforgeeks.org/stack-data-structure-introduction-program/
6+
// Stack (abstract data type) : https://en.wikipedia.org/wiki/Stack_(abstract_data_type)
7+
// author [Milad](https://github.com/miraddo)
8+
// see stackarray.go, stacklinkedlist.go, stacklinkedlistwithlist.go
9+
10+
package stack
11+
12+
import (
13+
"testing"
14+
)
15+
16+
// TestStackLinkedListWithList for testing Stack with Container/List Library (STL)
17+
func TestStackLinkedListWithList(t *testing.T) {
18+
st := NewDoublyLinkedList[int]()
19+
20+
t.Run("Stack Push", func(t *testing.T) {
21+
22+
st.Push(2)
23+
st.Push(3)
24+
25+
if st.Len() != 2 {
26+
t.Errorf("Expected 2 elements in the stack, found %d", st.Len())
27+
}
28+
})
29+
30+
t.Run("Stack Pop", func(t *testing.T) {
31+
pop, _ := st.Pop()
32+
33+
if pop != 3 {
34+
t.Errorf("Expected 3 from Pop operation, got %d", pop)
35+
}
36+
37+
if st.Len() != 1 {
38+
t.Errorf("Expected stack length to be 1 after Pop operation, got %d", st.Len())
39+
}
40+
})
41+
42+
t.Run("Stack Peek", func(t *testing.T) {
43+
st.Push(2)
44+
st.Push(83)
45+
peek, _ := st.Peek()
46+
if peek != 83 {
47+
t.Errorf("Expected value 83 from Peek operation, got %d", peek)
48+
}
49+
})
50+
51+
t.Run("Stack Len", func(t *testing.T) {
52+
if st.Len() != 3 {
53+
t.Errorf("Expected stack length to be 3, got %d", st.Len())
54+
}
55+
})
56+
57+
t.Run("Stack Empty", func(t *testing.T) {
58+
if st.Empty() {
59+
t.Error("Stack should not be empty")
60+
}
61+
st.Clear()
62+
if !st.Empty() {
63+
t.Error("Stack is expected to be empty")
64+
}
65+
})
66+
}

structure/stack/linked_list.go

+83
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
package stack
2+
3+
type node[T any] struct {
4+
Val T
5+
Next *node[T]
6+
}
7+
8+
// linkedList implements stack.Interface using a singly linked list as the underlying storage
9+
type linkedList[T any] struct {
10+
top *node[T]
11+
length int
12+
}
13+
14+
func NewLinkedList[T any]() Interface[T] {
15+
return new(linkedList[T])
16+
}
17+
18+
// Push value to the top of the stack
19+
func (ll *linkedList[T]) Push(n T) {
20+
newStack := new(node[T])
21+
22+
newStack.Val = n
23+
newStack.Next = ll.top
24+
25+
ll.top = newStack
26+
ll.length++
27+
}
28+
29+
// Pop returns last inserted element and removes it from the underlying storage
30+
// If the stack is empty, ErrStackEmpty error is returned
31+
func (ll *linkedList[T]) Pop() (T, error) {
32+
var element T
33+
if ll.Empty() {
34+
return element, ErrStackEmpty
35+
}
36+
element = ll.top.Val
37+
ll.top = ll.top.Next
38+
ll.length--
39+
return element, nil
40+
}
41+
42+
// Empty returns true if stack has no elements and false otherwise.
43+
func (ll *linkedList[T]) Empty() bool {
44+
return ll.length == 0
45+
}
46+
47+
// Len returns length of the stack
48+
func (ll *linkedList[T]) Len() int {
49+
return ll.length
50+
}
51+
52+
// Peek return last inserted element(top of the stack) without removing it from the stack
53+
// If the stack is empty, ErrStackEmpty error is returned
54+
func (ll *linkedList[T]) Peek() (T, error) {
55+
var element T
56+
if ll == nil || ll.length == 0 {
57+
return element, ErrStackEmpty
58+
}
59+
return ll.top.Val, nil
60+
}
61+
62+
// ToSlice returns the elements of stack as a slice
63+
func (ll *linkedList[T]) ToSlice() []T {
64+
var elements []T
65+
if ll == nil {
66+
return elements
67+
}
68+
69+
current := ll.top
70+
for current != nil {
71+
elements = append(elements, current.Val)
72+
current = current.Next
73+
}
74+
return elements
75+
}
76+
77+
func (ll *linkedList[T]) Clear() {
78+
if ll == nil {
79+
return
80+
}
81+
ll.top = nil
82+
ll.length = 0
83+
}

structure/stack/linked_list_test.go

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
package stack
2+
3+
import "testing"
4+
5+
// TestStackLinkedList for testing stack implementation using singly linked list
6+
func TestStack_SinglyLinkedList(t *testing.T) {
7+
st := NewLinkedList[int]()
8+
9+
st.Push(1)
10+
st.Push(2)
11+
12+
t.Run("Stack Push", func(t *testing.T) {
13+
result := st.ToSlice()
14+
expected := []int{2, 1}
15+
for x := range result {
16+
if result[x] != expected[x] {
17+
t.Errorf("Expected stack elements to be %v. Current elements: %v", expected, result)
18+
}
19+
}
20+
})
21+
22+
t.Run("Stack isEmpty", func(t *testing.T) {
23+
if st.Empty() {
24+
t.Error("Stack shouldn't be emtpy")
25+
}
26+
})
27+
28+
t.Run("Stack Length", func(t *testing.T) {
29+
if st.Len() != 2 {
30+
t.Errorf("Expected stack length to be 2, got %d", st.Len())
31+
}
32+
})
33+
34+
st.Pop()
35+
pop, _ := st.Pop()
36+
37+
t.Run("Stack Pop", func(t *testing.T) {
38+
if pop != 1 {
39+
t.Errorf("Expected 1 from Pop operation, got %d", pop)
40+
}
41+
})
42+
43+
st.Push(52)
44+
st.Push(23)
45+
st.Push(99)
46+
t.Run("Stack Peek", func(t *testing.T) {
47+
if val, _ := st.Peek(); val != 99 {
48+
t.Errorf("Expected 99 from Peek operation, got %d", val)
49+
}
50+
})
51+
}

0 commit comments

Comments
 (0)