Walk a string of bracket characters. Push every opening bracket. On a closing bracket, pop and verify it matches; mismatch or empty-stack pop means unbalanced. At end of string, stack must be empty.

Algorithm

Canonical input "({[]})" (balanced) finishes with the stack empty and the result true.

Basic Implementation

basic.scala
import scala.collection.mutable.ArrayBuffer

object Main {
	def matchOpen(close: Char): Char = close match {
		case ')' => '('
		case ']' => '['
		case '}' => '{'
		case _   => ' '
	}

	def main(args: Array[String]): Unit = {
		val text = "({[]})"
		val stack = ArrayBuffer.empty[Char]
		var balanced = true
		var i = 0
		while (balanced && i < text.length) {
			val ch = text(i)
			if (ch == '(' || ch == '[' || ch == '{') {
				stack += ch
			} else {
				if (stack.isEmpty || stack.last != matchOpen(ch)) {
					balanced = false
				} else {
					stack.remove(stack.length - 1)
				}
			}
			i = i + 1
		}
		if (stack.nonEmpty) {
			balanced = false
		}
		println(if (balanced) "True" else "False")
	}
}

Complexity

  • Time: O(n)
  • Space: O(n) worst case

Implementation notes

  • Scala: ArrayBuffer[Char] with += / remove(length - 1) / last is the smallest honest stack shape; reaching for scala.collection.mutable.Stack would hide the explicit push/pop pattern. A var balanced plus a while (balanced && i < text.length) loop keeps the early-exit visible without leaning on scala.util.boundary.
  • The matchOpen helper documents the closing -> opening map without leaking runtime references into the replay.
  • The replay highlights the current character, shows the stack updating each frame, and surfaces the final balanced/unbalanced verdict.
stack push/pop Use an `ArrayBuffer[Char]` as the open-bracket stack; push by `stack += ch`, pop by `stack.remove(stack.length - 1)`.
matching map A small `matchOpen(close)` helper returns the expected opener for each closing bracket — three explicit `match` arms keep the lesson compact.