Implement a Last-In-First-Out (LIFO) stack using a singly linked list. The implemented stack should support the following operations: push, pop, top, and isEmpty.
Implement the LinkedListStack class:
void push(int x): Pushes element x onto the stack.
int pop(): Removes and returns the top element of the stack.
int top(): Returns the top element of the stack without removing it.
boolean isEmpty(): Returns true if the stack is empty, false otherwise.
Input:
["LinkedListStack", "push", "push", "pop", "top", "isEmpty"]
[[], [3], [7], [], [], []]
Output: [null, null, null, 7, 3, false]
Explanation:
LinkedListStack stack = new LinkedListStack();
stack.push(3);
stack.push(7);
stack.pop(); // returns 7
stack.top(); // returns 3
stack.isEmpty(); // returns false
Input:
["LinkedListStack", "isEmpty"]
[[]]
Output: [null, true]
Explanation:
LinkedListStack stack = new LinkedListStack();
stack.isEmpty(); // returns true
Input:
["LinkedListStack", "push", "pop", "isEmpty"]
[[], [2], [], []]
Consider a stack of plates in a cafeteria. Each time a plate is added, it is placed on top of the stack, and when a plate is taken, it is removed from the top. This ensures that the last plate added is the first one to be used. If this stack of plates were managed using a linked list, each plate would be a node in the list, dynamically linked to the one below it. This structure allows adding or removing plates without worrying about a fixed capacity or the need to shuffle plates around, mimicking the behavior of a real-world stack with dynamic flexibility.
Implementing a stack using a linked list offers several advantages over using an array. Linked lists provide dynamic size flexibility, allowing the stack to grow and shrink as needed without worrying about predefined capacity or memory wastage. Each node in the linked list contains data and a reference to the next node, enabling efficient constant-time operations for both push and pop. This eliminates the risk of stack overflow and avoids the need for costly resizing operations, making linked lists an ideal choice for managing the dynamic nature of stack operations.
Approaching the problem of implementing a stack using a linked list can be likened to managing the stack of plates in the cafeteria example. Each plate represents a node in the linked list, and the top plate corresponds to the top node. To add a plate (push operation), a new node is created and placed on top of the stack, pointing to the previous top node. To remove a plate (pop operation), the top node is removed, and the next node in line becomes the new top. This dynamic linking and unlinking process allows the stack to grow and shrink efficiently, without any predefined capacity, just as the stack of plates can adjust to the number of plates being added or taken away.
#include <bits/stdc++.h>
using namespace std;
// Node structure
struct Node {
int val;
Node *next;
Node(int d) {
val = d;
next = NULL;
}
};
// Structure to represent stack
class LinkedListStack {
private:
Node *head; // Top of Stack
int size; // Size
public:
// Constructor
LinkedListStack() {
head = NULL;
size = 0;
}
// Method to push an element onto the stack
void push(int x) {
// Creating a node
Node *element = new Node(x);
element->next = head; // Updating the pointers
head = element; // Updating the top
// Increment size by 1
size++;
}
// Method to pop an element from the stack
int pop() {
// If the stack is empty
if (head == NULL) {
return -1; // Pop operation cannot be performed
}
int value = head->val; // Get the top value
Node *temp = head; // Store the top temporarily
head = head->next; // Update top to next node
delete temp; // Delete old top node
size--; // Decrement size
return value; // Return data
}
// Method to get the top element of the stack
int top() {
// If the stack is empty
if (head == NULL) {
return -1; // Top element cannot be accessed
}
return head->val; // Return the top
}
// Method to check if the stack is empty
bool isEmpty() {
return (size == 0);
}
};
int main() {
// Creating a stack
LinkedListStack st;
// List of commands
vector<string> commands = {"LinkedListStack", "push", "push",
"pop", "top", "isEmpty"};
// List of inputs
vector<vector<int>> inputs = {{}, {3}, {7}, {}, {}, {}};
for (int i = 0; i < commands.size(); ++i) {
if (commands[i] == "push") {
st.push(inputs[i][0]);
cout << "null ";
} else if (commands[i] == "pop") {
cout << st.pop() << " ";
} else if (commands[i] == "top") {
cout << st.top() << " ";
} else if (commands[i] == "isEmpty") {
cout << (st.isEmpty() ? "true" : "false") << " ";
} else if (commands[i] == "LinkedListStack") {
cout << "null ";
}
}
return 0;
}
import java.util.*;
// Node structure
class Node {
int val;
Node next;
Node(int d) {
val = d;
next = null;
}
}
// Structure to represent stack
class LinkedListStack {
private Node head; // Top of Stack
private int size; // Size
// Constructor
public LinkedListStack() {
head = null;
size = 0;
}
// Method to push an element onto the stack
public void push(int x) {
// Creating a node
Node element = new Node(x);
element.next = head; // Updating the pointers
head = element; // Updating the top
// Increment size by 1
size++;
}
// Method to pop an element from the stack
public int pop() {
// If the stack is empty
if (head == null) {
return -1; // Pop operation cannot be performed
}
int value = head.val; // Get the top value
Node temp = head; // Store the top temporarily
head = head.next; // Update top to next node
temp = null; // Delete old top node
size--; // Decrement size
return value; // Return data
}
// Method to get the top element of the stack
public int top() {
// If the stack is empty
if (head == null) {
return -1; // Top element cannot be accessed
}
return head.val; // Return the top
}
// Method to check if the stack is empty
public boolean isEmpty() {
return (size == 0);
}
}
class Main{
public static void main(String[] args) {
// Creating a stack
LinkedListStack st = new LinkedListStack();
// Array of commands
String[] commands = {"LinkedListStack", "push", "push",
"pop", "top", "isEmpty"};
// Array of inputs
int[][] inputs = {{}, {3}, {7}, {}, {}, {}};
for (int i = 0; i < commands.length; ++i) {
if (commands[i].equals("push")) {
st.push(inputs[i][0]);
System.out.print("null ");
} else if (commands[i].equals("pop")) {
System.out.print(st.pop() + " ");
} else if (commands[i].equals("top")) {
System.out.print(st.top() + " ");
} else if (commands[i].equals("isEmpty")) {
System.out.print((st.isEmpty() ? "true" : "false") + " ");
} else if (commands[i].equals("LinkedListStack")) {
System.out.print("null ");
}
}
}
}
# Node structure
class Node:
def __init__(self, d):
self.val = d
self.next = None
# Structure to represent stack
class LinkedListStack:
def __init__(self):
self.head = None # Top of Stack
self.size = 0 # Size
# Method to push an element onto the stack
def push(self, x):
# Creating a node
element = Node(x)
element.next = self.head # Updating the pointers
self.head = element # Updating the top
# Increment size by 1
self.size += 1
# Method to pop an element from the stack
def pop(self):
# If the stack is empty
if self.head is None:
return -1 # Pop operation cannot be performed
value = self.head.val # Get the top value
temp = self.head # Store the top temporarily
self.head = self.head.next # Update top to next node
del temp # Delete old top node
self.size -= 1 # Decrement size
return value # Return data
# Method to get the top element of the stack
def top(self):
# If the stack is empty
if self.head is None:
return -1 # Top element cannot be accessed
return self.head.val # Return the top
# Method to check if the stack is empty
def isEmpty(self):
return self.size == 0
# Creating a stack
st = LinkedListStack()
# List of commands
commands = ["LinkedListStack", "push", "push", "pop", "top", "isEmpty"]
# List of inputs
inputs = [[], [3], [7], [], [], []]
for i in range(len(commands)):
if commands[i] == "push":
st.push(inputs[i][0])
print("null", end=" ")
elif commands[i] == "pop":
print(st.pop(), end=" ")
elif commands[i] == "top":
print(st.top(), end=" ")
elif commands[i] == "isEmpty":
print("true" if st.is_empty() else "false", end=" ")
elif commands[i] == "LinkedListStack":
print("null", end=" ")
// Node structure
class Node {
constructor(d) {
this.val = d;
this.next = null;
}
}
// Structure to represent stack
class LinkedListStack {
constructor() {
this.head = null; // Top of Stack
this.size = 0; // Size
}
// Method to push an element onto the stack
push(x) {
// Creating a node
const element = new Node(x);
element.next = this.head; // Updating the pointers
this.head = element; // Updating the top
// Increment size by 1
this.size++;
}
// Method to pop an element from the stack
pop() {
// If the stack is empty
if (this.head === null) {
return -1; // Pop operation cannot be performed
}
const value = this.head.val; // Get the top value
const temp = this.head; // Store the top temporarily
this.head = this.head.next; // Update top to next node
this.size--; // Decrement size
return value; // Return data
}
// Method to get the top element of the stack
top() {
// If the stack is empty
if (this.head === null) {
return -1; // Top element cannot be accessed
}
return this.head.val; // Return the top
}
// Method to check if the stack is empty
isEmpty() {
return (this.size === 0);
}
}
// Creating a stack
const st = new LinkedListStack();
// List of commands
const commands = ["LinkedListStack", "push", "push",
"pop", "top", "isEmpty"];
// List of inputs
const inputs = [[], [3], [7], [], [], []];
for (let i = 0; i < commands.length; ++i) {
if (commands[i] === "push") {
st.push(inputs[i][0]);
console.log("null");
} else if (commands[i] === "pop") {
console.log(st.pop());
} else if (commands[i] === "top") {
console.log(st.top());
} else if (commands[i] === "isEmpty") {
console.log(st.isEmpty() ? "true" : "false");
} else if (commands[i] === "LinkedListStack") {
console.log("null");
}
}
Q: Why use a linked list instead of an array?
A: A linked list does not require resizing like an array, making it more memory efficient when dealing with unknown sizes. Insertions and deletions at the head take O(1) time, while an array-based stack may require shifting elements when using a fixed-size array.
Q: How does push() work in constant time?
A: A new node is created and inserted at the head of the linked list in O(1) time.
Q: What if we want a doubly linked list instead?
A: A doubly linked list would allow push() and pop() in O(1) but consumes extra memory for storing the prev pointer.
Q: How would you modify this implementation for a min-stack (stack that retrieves the minimum element in O(1))?
A: Maintain a secondary stack that keeps track of the minimum value at each push operation.