Stack a common special linear data structure, its special place is its order of operation, the following will be described in detail, but also because of its characteristics, so the stack can easily solve the expression value, bracket matching, recursive algorithms, backtracking algorithms and so on.
01Definitions
The particularity of the stack is manifested in the restricted operation, one of which allows the insertion and deletion of elements at one end of the stack, and the second operation of the stack follows the principle of last in first out (Last In First Out, or LIFO for short).
push on: When an element is inserted into the stack, this behavior is called entering the stack, also known as going into the stack or pressing the stack;
send a package to a warehouse: When an element is removed from the stack, this behavior is called out of the stack, also known as backing out of the stack;
stack top (computing): The end that allows elements to perform insertion and deletion operations is called the top of the stack;
the bottom of a stack: The other end corresponding to the top of the stack is called the bottom of the stack and element operations are not allowed;
warehouse: When there are no elements in the stack it is called an empty stack.
make a full stop at a hotel: When the stack is of finite capacity and the capacity is exhausted, it is called a full stack.
stack capacity: When the stack is of finite capacity, the stack capacity represents the maximum number of elements that the stack can hold.
stack size (computing): Indicates the number of elements in the current stack.
02Classification
Stacks are logical structures and therefore can be categorized into sequential stacks and chained stacks depending on how they are stored.
Sequential stack is to use a continuous address space to store all the stack elements, usually using an array implementation, which leads to a fixed size of the stack, not easy to expand the expansion capacity, easy to waste space and also pay attention to the element overflow and other issues.
Chain stacks, as the name suggests, are stored in a chained fashion, usually implemented as a unidirectional linked table, so they can be expanded indefinitely, used on demand, with efficient memory utilization, and there is no such thing as a full stack.
03, implementation (sequential stack)
We implement sequential stacks with the help of arrays, and the core idea is to use the start position of the array as the bottom of the stack and the tail direction of the array as the top of the stack.
We know that the array is not friendly to inserting and deleting elements, because it involves the problem of movement of already existing elements, but if you insert and delete elements directly at the end of the array is still very convenient, because it does not involve the problem of movement of the elements, we are using this feature to make the starting position of the array as the bottom of the stack, and insertion and deletion of the end of the array as the top of the stack for convenience.
Below we will implement a generalized sequential stack step by step.
1. Definition of ADT
We begin by defining the ADT for the sequential stack.
ADT Stack{
Data object: D is a non-empty collection of elements, D = {a1, a2, ... , an}, where ai denotes the ith element in the stack and n is the length of the stack.
Data Relationships: The elements in D are organized by their indexes (positions), which are integers from 0 to n-1, and follow the principle that elements can only be operated on the top of the stack, and that elements go in last in, first out.
Basic operations: [
Init(n) : Initialize an empty stack of specified capacity.
Capacity: Returns the capacity of the stack.
Length: return the length of the stack.
Top: return the top element of the stack, if the stack is empty, then report an exception.
IsEmpty(): return whether the stack is empty.
IsFull(): return whether the stack is full.
Push(): add an element to the stack, or report an exception if the stack is full.
Pop(): out of the stack that is to return to the top of the stack and remove it from the stack, when the stack is empty, then report an exception.
]
}
With the stack ADT defined, we can now begin our own implementation of the stack.
2、Initialization Init
Initializing a structure does a few things.
-
Initialize the capacity of the stack;
-
Initializes the array holding the stack elements;
-
Initialize the top-of-stack index;
The specific implementation code is as follows:
//store the stack elements
private T[] _array;
//stack capacity
private int _capacity;
//Top of stack index, -1 means empty stack
private int _top; //Initialize the stack to the specified capacity.
//Initialize the stack to the specified capacity
public MyselfArrayStack<T> Init(int capacity)
{
//Initialize the stack to capacity
_capacity = capacity;
// Initialize an array of the specified length to hold the stack elements.
_array = new T[_capacity]; //Initialize to empty stack.
//initialize to empty stack
_top = -1; //return to the stack
//Return to the stack
return this; }
}
3、Get stack capacity Capacity
It's easier to just return the stack capacity private field.
//stack capacity
public int Capacity
{
get
{
return _capacity;
}
}
4. Get stack length Length
Because the top of the stack index represents the array subscript, so you can add 1 to the top of the stack index to the length of the stack, at the same time, because we defined the empty stack is the top of the stack index is -1, so at this time the length of the stack is equal to [-1 +1] for 0, so the length of the stack that is [top of the stack index + 1].
//stack length
public int Length
{
get
{
//The stack length is equal to the top element of the stack plus 1
return _top + 1; }
}
}
5、Get the top element of the stack Top
Getting the top element of the stack can be obtained directly from the array through the private field of the top index of the stack, and at the same time pay attention to determine whether the stack is an empty stack, and report an exception if it is an empty stack. The specific code is as follows:
//Get the top element of the stack
public T Top
{
get
{
if (IsEmpty())
{
// Empty stack, no top-of-stack operations are allowed.
throw new InvalidOperationException("Empty Stack");
}
return _array[_top];
}
}
6, get whether the empty stack IsEmpty
Whether or not the stack is empty is simply a matter of determining whether or not the top-of-stack index is less than 0.
//Whether the empty stack
public bool IsEmpty()
{
//Top of the stack index is less than 0 means that the empty stack
return _top < 0; }
}
7、Get whether the full stack IsFull
Whether the stack is full just to determine whether the top of the stack index is equal to the stack capacity minus 1, the code is as follows:
//Is the stack full
public bool IsFull()
{
//Top of the stack index is equal to the capacity of the size of the full stack
return _top == _capacity - 1;
}
8、Into the stack Push
Into the stack is in the top of the stack index to the array after moving one, and then assign the new element to the top of the stack index of the corresponding element, but also need to check whether the full stack, if it is a full stack is reported as an error, the specific implementation of the code is as follows:
//Into the stack
public void Push(T value)
{
if (IsFull())
{
//Top of stack index is greater than or equal to capacity minus 1, indicating that the stack is full and cannot be loaded.
throw new InvalidOperationException("Full Stack");
}
// Move the top of the stack index back 1 bit before storing the top element.
_array[++_top] = value; }
}
9、Out Stack Pop
Out of the stack is first removed from the top of the stack elements, and then the top of the stack index to the front of the array to move one, and finally return to remove the top of the stack elements, but also need to check whether the empty stack, if it is an empty stack will report an error, the specific implementation of the code is as follows:
//out of the stack
public T Pop()
{
if (IsEmpty())
{
//Top-of-stack index less than 1 means empty stack, can not be out of the stack operation
throw new InvalidOperationException("Empty stack");;
}
// After returning the top element of the stack, the index of the top of the stack is shifted forward by 1 bit.
return _array[_top--];
}
04, realization (chain stack)
We implement chained stacks with the help of chained lists, the core idea of which is to use the tail node of the chained list as the bottom of the stack and the first element node of the chained list as the top of the stack.
This is because if we want to get the tail node of the chain table need to variable the entire chain table to get, but to get the first node can be obtained directly through the head pointer (no head node chain), so for the tail node of the chain table operation is not friendly suitable for doing the bottom of the stack, the first node of the chain table operation friendly suitable for doing the top of the stack.
Below we will implement a generic chain stack step by step.
1. Definition of ADT
Compared to sequential stack ADTs, chained stack ADTs have two fewer methods i.e. getting the stack capacity and whether the stack is full or not, which is a benefit of the chained table feature.
2、Initialization Init
The initialization structure mainly initializes the top node of the stack to be empty and the length of the stack to be 0. The specific implementation is as follows:
public class MyselfStackNode<T>
{
//Data field
public T Data.
// pointer field, i.e. the next node
public MyselfStackNode<T> Next;
public MyselfStackNode(T data)
{
Next = null; public MyselfStackNode(T) { data
Next = null;
}
}
public class MyselfStackLinkedList<T>
{
//Top stack node is the primitive node
private MyselfStackNode<T> _top;
//The length of the stack
private int _length;
//Initialize the stack
public MyselfStackLinkedList<T> Init()
{
//Initialize the top node of the stack to null
_top = null;
//Initialize the stack length to 0
_length = 0;
// Return the stack
return this; }
}
}
3. Get stack length Length
It's easier to just return the stack length private field.
//stack length (computing)
public int Length
{
get
{
return _length;
}
}
4、Get the top element of the stack Top
Getting the top element of the stack can be obtained directly through the top node of the stack, but pay attention to determine whether the stack is an empty stack, if it is an empty stack then report an exception. The specific code is as follows:
//Get the top element of the stack
public T Top
{
get
{
if (IsEmpty())
{
// Empty stack, no top-of-stack operations are allowed.
throw new InvalidOperationException("Empty Stack");
}
// Return the first element node data field
return _top.Data; }
}
}
5、Get whether the empty stack IsEmpty
Whether or not the stack is empty is simply a matter of determining whether or not the top node of the stack is empty.
//Whether the empty stack
public bool IsEmpty()
{
//Top of the stack node is null means that the empty stack
return _top == null;
}
6、Into the stack Push
Into the stack is the first to create a new node, and then the original top of the stack node assigned to the pointer field of the new node, and finally change the new node to the top of the stack node, at the same time the stack length is increased by 1, the specific implementation code is as follows:
//Into the stack
public void Push(T value)
{
// Create a new top stack node
var node = new MyselfStackNode<T>(value);
// Assign the old stack top node to the pointer field of the new node
= _top;
// change the top stack node to the newly created node
_top = node;
// add 1 to the stack length
_length++;
}
7、Out Stack Pop
Out of the stack is first removed from the top of the stack node corresponding to the data, and then the top of the stack node points to the original top of the stack node corresponding to the next node, at the same time the length of the stack is reduced by 1, of course, if the empty stack will be reported as an error, the specific implementation of the code is as follows:
//out of the stack
public T Pop()
{
if (IsEmpty())
{
//Empty stack, can not be out of the stack operation
throw new InvalidOperationException("Empty Stack");
}
// Get the data of the top node of the stack
var data = _top.Data;
//Change the top node to the next node corresponding to the original top node
_top = _top.Next; //change the stack top node to the next node corresponding to the original stack top node
//Subtract 1 from the stack length
_length--;; //return the stack top data
//Return the data at the top of the stack
return data; }
}
classifier for sums of money: The test method code as well as the sample source code have been uploaded to the code repository for those who are interested./hugogoos/Planner