본문으로 건너뛰기
Previous
Next
Time Complexity Optimization Checklist for Coding Interviews

Time Complexity Optimization Checklist for Coding Interviews

Time Complexity Optimization Checklist for Coding Interviews

이 글의 핵심

Reduce time complexity in coding interviews: patterns to convert O(N²) to O(N log N), eliminate duplicate calculations, and data structure selection checklist.

Introduction

Reducing time complexity in coding interviews starts with the habit of matching constraints (N range) with current complexity, rather than memorizing faster algorithms. Most TLE (Time Limit Exceeded) comes from unnecessary nested loops or repeatedly calculating same ranges. This guide is organized as a checklist for immediate use in exams. Numbers are approximate upper bounds and may vary with constants, language, and judge environment. The checklist is most effective when used as a 30-second scan habit after reading the problem, rather than memorization.

After Reading This

  • Intuitively grasp allowed complexity ranges by N scale
  • Recall typical patterns to reduce O(N²) to O(N log N)
  • Quickly choose between hash map, sorting, prefix sum, or binary search

Table of Contents

  1. Concepts
  2. Practical Implementation
  3. Advanced Usage
  4. Performance Comparison
  5. Real-World Cases
  6. Troubleshooting
  7. Conclusion

Concepts

Rough Guide (1 second limit, simple operations assumed)

N (approx)SafeRisky
≤ 20O(N!) barely
≤ 200O(N³)O(N⁴)
≤ 2×10³O(N²)O(N² log N) usually OK
≤ 10⁵O(N log N)O(N²) almost impossible
≤ 10⁶O(N)O(N log N) often possible
≥ 10⁷O(N) linear/simpleO(N log N) can be tight
First Step: Underline N, Q (query count), string length in problem, and write current solution’s complexity in one line. This is the first step in reducing coding interview time complexity.

Practical Implementation

✅ Step 1: Are Nested Loops Checking “All Pairs”?

  • All (i, j) pairs in array → baseline O(N²).
  • Check if you can reduce one axis with sorting + one-direction scan, two pointers, or binary search.
# Bad: All pairs — O(n²)
def two_sum_naive(a, t):
    for i in range(len(a)):
        for j in range(i + 1, len(a)):
            if a[i] + a[j] == t:
                return i, j
# Good: Sorting + two pointers — O(n log n)
def two_sum_sorted(a, t):
    a = sorted(enumerate(a), key=lambda x: x[1])
    i, j = 0, len(a) - 1
    while i < j:
        s = a[i][1] + a[j][1]
        if s == t:
            return a[i][0], a[j][0]
        if s < t:
            i += 1
        else:
            j -= 1
# Better: Hash map — O(n)
def two_sum_hash(a, t):
    seen = {}
    for i, num in enumerate(a):
        complement = t - num
        if complement in seen:
            return seen[complement], i
        seen[num] = i

✅ Step 2: Recalculating Same Range Sum/Min Every Time?

  • Prefix sum for range sum in O(1).
  • Sliding window for fixed-length range min/max in O(N).
# Bad: Recalculate sum — O(n*k)
def max_sum_naive(arr, k):
    max_sum = 0
    for i in range(len(arr) - k + 1):
        current_sum = sum(arr[i:i+k])  # Recalculate!
        max_sum = max(max_sum, current_sum)
    return max_sum
# Good: Sliding window — O(n)
def max_sum_window(arr, k):
    window_sum = sum(arr[:k])
    max_sum = window_sum
    for i in range(k, len(arr)):
        window_sum = window_sum - arr[i-k] + arr[i]
        max_sum = max(max_sum, window_sum)
    return max_sum

✅ Step 3: Using Linear Search for “Existence/Frequency”?

  • Hash map (dict / unordered_map) for average O(1) lookup.
  • TreeMap for O(log N) when order needed — only when sorted keys required.
# Bad: Linear search — O(n) per lookup
def count_frequency_naive(arr, target):
    count = 0
    for num in arr:
        if num == target:
            count += 1
    return count
# Good: Hash map — O(1) per lookup
from collections import Counter
def count_frequency_hash(arr):
    freq = Counter(arr)
    return freq  # O(1) lookup

✅ Step 4: Scanning Entire Array Per Query?

  • If offline, use Mo’s algorithm.
  • If fixed range values, use cumulative frequency array.
  • For range min, use segment tree to reduce cost per query.

✅ Step 5: Is BFS/DFS Enough for Graph?

  • For edge weights/shortest path, Dijkstra etc. changes complexity class. Connect with BFS vs DFS.

Advanced Usage

  • For repeated range sum/min/max updates, consider segment tree or Fenwick tree.
  • For bit operations/subset problems, bitset can reduce both space and time. Even in competitions/practice, remembering “changing data structure changes complexity class” makes direction easier.

Performance Comparison

NeedChoiceTypical Cost
Key existence/countHash mapAverage O(1)
Sorted order/next elementTree / sorted+binaryO(log N)
Range sumPrefix sumPreprocess O(N), query O(1)
Range min/updateSegment treeQuery/update O(log N)
PriorityHeappush/pop O(log N)

Real-World Cases

  • Log aggregation: Don’t scan entire log for daily counts, use date index array or map + accumulation.
  • API rate limiting: Sliding window + queue for O(request count) processing.

Troubleshooting

“Correct Algorithm But Still TLE”

  • I/O: Python uses sys.stdin.buffer for large input with input().
  • Recursion depth: sys.setrecursionlimit or convert to iterative BFS.
# Python I/O optimization
import sys
input = sys.stdin.readline
# Or for very large input
import sys
data = sys.stdin.buffer.read().decode()
lines = data.split('\n')

”O(N log N) But Still Worried”

  • Constants: dict lookup vs list index, remove unnecessary sorting.
  • Language choice: C++ is 10-100x faster than Python for same algorithm.
# Bad: Sort multiple times
for query in queries:
    arr.sort()  # O(n log n) per query!
    # process...
# Good: Sort once
arr.sort()
for query in queries:
    # process sorted arr...

Conclusion

Reducing coding interview time complexity is half done by writing N range and current solution’s order in one line. The rest is checking with just two axes: eliminate duplicate calculations and correct data structure. For pattern practice, continue with Two Pointers & Sliding Window.

Quick Reference

N ≤ 20:      O(N!) or O(2^N) acceptable
N ≤ 200:     O(N³) acceptable
N ≤ 2000:    O(N²) acceptable
N ≤ 100000:  O(N log N) required
N ≤ 1000000: O(N) or O(N log N)
N ≥ 10^7:    O(N) linear required
Common Optimizations:
O(N²) → O(N log N): Sorting + two pointers/binary search
O(N²) → O(N): Hash map, sliding window
O(N) → O(1): Prefix sum, math formula


Keywords

Time Complexity, Optimization, Coding Interview, TLE, Algorithm, Big O, Hash Map, Two Pointers, Sliding Window, Binary Search, Prefix Sum


자주 묻는 질문 (FAQ)

Q. 이 내용을 실무에서 언제 쓰나요?

A. Reduce time complexity in coding interviews: patterns to convert O(N²) to O(N log N), eliminate duplicate calculations, … 실무에서는 위 본문의 예제와 선택 가이드를 참고해 적용하면 됩니다.

Q. 선행으로 읽으면 좋은 글은?

A. 각 글 하단의 이전 글 또는 관련 글 링크를 따라가면 순서대로 배울 수 있습니다. C++ 시리즈 목차에서 전체 흐름을 확인할 수 있습니다.

Q. 더 깊이 공부하려면?

A. cppreference와 해당 라이브러리 공식 문서를 참고하세요. 글 말미의 참고 자료 링크도 활용하면 좋습니다.


같이 보면 좋은 글 (내부 링크)

이 주제와 연결되는 다른 글입니다.


이 글에서 다루는 키워드 (관련 검색어)

Algorithm, Time Complexity, Optimization, Coding Interview, TLE, Checklist 등으로 검색하시면 이 글이 도움이 됩니다.