-
Notifications
You must be signed in to change notification settings - Fork 126
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #1013 from dusunax/main
[οΏ½SunaDu] Week 10
- Loading branch information
Showing
3 changed files
with
294 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,184 @@ | ||
''' | ||
# 207 | ||
μ°Έκ³ μμ: https://www.youtube.com/watch?v=EgI5nU9etnU | ||
λ¬Έμ νμ΄: https://www.algodale.com/problems/course-schedule/ | ||
## λ¬Έμ μ 리 | ||
π prerequisitesλ? νμ μ μ κ³Όλͺ©μ΄λ€. | ||
π λ°©ν₯μ±μ΄ μλ μ°κ²° κ΄κ³μ΄λ―λ‘, Directed Graphλ€. | ||
π Cycle λ°μ μ, μ½μ€λ₯Ό μ΄μν μ μλ€.(μλ‘ μμ‘΄νλ μνμ΄ μμ΄μ λμμ΄ λκ² λλ κ²½μ°) | ||
## ν΄κ²° λ°©μ λκ°μ§ | ||
1. BFS, Queue, Topological Sort: μμ μ λ ¬ | ||
2. DFS, Cycle Detection: μν νμ§ | ||
### μμ μ λ ¬(Topological Sort) - BFS, Queue | ||
- μ§μ μ°¨μ(indegree): λ Έλλ‘ λ€μ΄μ€λ νμ΄ν μ | ||
- κ·Έλνλ‘ μΈμ 리μ€νΈ κ΅¬μ± | ||
- Queueμ λ£κΈ° | ||
- Queue BFS νμ | ||
- λͺ¨λ κ³Όλͺ©μ λ€μλμ§ νμΈ | ||
### μν νμ§(Cycle Detection) - DFS | ||
- κ·Έλνλ‘ μΈμ 리μ€νΈ κ΅¬μ± | ||
- λ°©λ¬Έ μν λ°°μ΄ μ΄κΈ°ν | ||
- dfs ν¨μ | ||
- λͺ¨λ λ Έλμ λν΄ dfs μ€ν | ||
## TC & SC | ||
- μκ° λ³΅μ‘λμ κ³΅κ° λ³΅μ‘λλ O(V + E)λ‘ λμΌνλ€. | ||
``` | ||
V: λ Έλ μ(κ³Όλͺ© μ) | ||
E: κ°μ μ(μ μ κ³Όλͺ© κ΄κ³ μ) | ||
``` | ||
### TC is O(V + E) | ||
λ λ°©λ² λͺ¨λ, κ·Έλνμ λͺ¨λ λ Έλμ κ°μ μ ν λ²μ© νμΈν¨ | ||
- BFS: λͺ¨λ Vλ₯Ό μννλ©΄μ, κ° λ Έλμμ λκ°λ Eλ₯Ό λ°λΌκ°λ©° μ°¨μλ₯Ό μ€μ | ||
- DFS: λͺ¨λ Vλ₯Ό μννλ©΄μ, κ° λ Έλμμ μ°κ²°λ Eλ₯Ό λ°λΌκ°λ©° κΉμ΄ νμ | ||
### SC is O(V + E) | ||
- O(V+E): V + Eλ₯Ό μ μ₯νλ μΈμ 리μ€νΈ κ·Έλν | ||
- O(V)'s: λ°©λ¬Έ μν μ μ₯, μ§μ μ°¨μ λ°°μ΄, BFS ν, DFS νΈμΆ μ€ν | ||
## μμμ λ ¬(BFS) vs μννμ§(DFS)π€ | ||
### BFSλ₯Ό μ¬μ©νμ λ | ||
- λ°λ³΅λ¬Έμ μ¬μ©ν BFSκ° indegree(μ§μ μ°¨μ) κ°λ μ΄ λ³΄λ€ μ§κ΄μ μ΄λ―λ‘ => "μμλλ‘ μ²λ¦¬ν μ μλμ§ νμΈ"ν λ λͺ ννκ² μ¬μ©ν μ μλ€. μ§μ μ°¨μκ° 0μΈ λ ΈλλΆν° μμν΄μ μ²λ¦¬ | ||
- μ μ κ³Όλͺ©μ λ€ λ€μ κ³Όλͺ©μ μ§μ μ°¨μκ° 0μ΄ λλ―λ‘ λ€μ΄κ° μ μλ κ³Όλͺ©μ΄λΌλ μ μ΄ λͺ νν¨ | ||
``` | ||
ν€μλ: μ²λ¦¬ μμλ₯Ό μΆλ ₯, μμλλ‘ μ²λ¦¬ν μ μλμ§ | ||
``` | ||
### DFSλ₯Ό μ¬μ©νμ λ | ||
- DFS μν νμ§λ "μν μ¬λΆ"κ° ν΅μ¬μΌ λ μμ°μ€λ½λ€. | ||
- μν(Status)λ₯Ό μ¬μ©ν΄μ, λ°©λ¬Έ μ€μΈ λ Έλ μνλ₯Ό λ€μ λ°©λ¬Ένλ€λ©΄ μνμ΄ μμμ λ°λ‘ μ μ μλ€. | ||
- μνμ΄ λ°κ²¬λλ©΄ λ°λ‘ μ€λ¨νλ―λ‘, μν νμ§μ μμ°μ€λ½λ€. | ||
``` | ||
ν€μλ: μ¬μ΄ν΄μ΄ μλμ§ νλ¨ | ||
``` | ||
### +a) `@cache`λ₯Ό νμ©ν΄λ³΄μ. | ||
- νμ΄μ 3.9~ λ©λͺ¨μ΄μ μ΄μ ν¨μ | ||
- μμ ν¨μ + μ¬κ· μ΅μ νμ μ¬μ© (μΈλΆ μμ‘΄μ±, λΆμν¨κ³Όμ μ£Όμν κ²) | ||
''' | ||
from enum import Enum | ||
|
||
class Status(Enum): # use it to dfs | ||
INITIAL = 1 | ||
IN_PROGRESS = 2 | ||
FINISHED = 3 | ||
|
||
class Solution: | ||
''' | ||
1. BFS | ||
μμ μ λ ¬ | ||
''' | ||
def canFinishTopologicalSort(self, numCourses: int, prerequisites: List[List[int]]) -> bool: | ||
indegree = [0] * numCourses | ||
graph = defaultdict(list) | ||
|
||
for dest, src in prerequisites: | ||
graph[src].append(dest) | ||
indegree[dest] += 1 | ||
|
||
queue = deque([i for i in range(numCourses) if indegree[i] == 0]) | ||
processed_count = 0 | ||
|
||
while queue: | ||
node = queue.popleft() | ||
processed_count += 1 | ||
for neighbor in graph[node]: | ||
indegree[neighbor] -= 1 | ||
if indegree[neighbor] == 0: | ||
queue.append(neighbor) | ||
|
||
return processed_count == numCourses | ||
|
||
''' | ||
2. DFS | ||
μν νμ§ | ||
''' | ||
def canFinishCycleDetection(self, numCourses: int, prerequisites: List[List[int]]) -> bool: | ||
graph = defaultdict(list) | ||
|
||
for dest, src in prerequisites: | ||
graph[src].append(dest) | ||
|
||
statuses = {i: Status.INITIAL for i in range(numCourses)} | ||
|
||
def dfs(node): | ||
if statuses[node] == Status.IN_PROGRESS: | ||
return False | ||
if statuses[node] == Status.FINISHED: | ||
return True | ||
|
||
statuses[node] = Status.IN_PROGRESS | ||
for neighbor in graph[node]: | ||
if not dfs(neighbor): | ||
return False | ||
statuses[node] = Status.FINISHED | ||
return True | ||
|
||
return all(dfs(crs) for crs in range(numCourses)) | ||
|
||
''' | ||
3. @cache | ||
νμ΄μ¬ 3.9 μ΄μμμ μ¬μ©νλ λ©λͺ¨μ΄μ μ΄μ λ°μ½λ μ΄ν° | ||
- λμΌ μ λ ₯ -> λμΌ μΆλ ₯μ 보μ₯νλ€. | ||
- 128κ° κΉμ§λ§ μ μ₯νλ @lru_cacheλ μλ€. | ||
''' | ||
def canFinishWithCache(self, numCourses: int, prerequisites: List[List[int]]) -> bool: | ||
graph = defaultdict(list) | ||
|
||
for dest, src in prerequisites: | ||
graph[src].append(dest) | ||
|
||
traversing = set() | ||
|
||
@cache | ||
def dfs(node): | ||
if node in traversing: | ||
return False | ||
|
||
traversing.add(node) | ||
result = all(dfs(pre) for pre in graph[node]) | ||
traversing.remove(node) | ||
return result | ||
|
||
return all(dfs(node) for node in range(numCourses)) | ||
|
||
''' | ||
4. visitedκ³Ό ν¨κ» μ¬μ©νκΈ° | ||
@cache λ°μ½λ μ΄ν°λ λ©λͺ¨μ΄μ μ΄μ , κ°μ μ λ ₯κ°μ λ°λΌ κ°μ κ²°κ³Όλ₯Ό λ°ννκ²ν¨ | ||
κ²°κ³Όκ° λ³νμ§ μμ λ μ μ©ν¨ => dfs(node)λ μΈλΆ μν μν traversingμ μμ‘΄ν΄μ λμμ΄ λ¬λΌμ§ μ μλ€. | ||
λ°λΌμ visited setμ΄ λ μμ°μ€λ¬μΈ μ μλ€ | ||
''' | ||
def canFinishWithVisited(self, numCourses: int, prerequisites: List[List[int]]) -> bool: | ||
graph = defaultdict(list) | ||
|
||
for dest, src in prerequisites: | ||
graph[src].append(dest) | ||
|
||
traversing = set() | ||
visited = set() | ||
|
||
def dfs(node): | ||
if node in traversing: | ||
return False | ||
if node in visited: | ||
return True | ||
|
||
traversing.add(node) | ||
for pre in graph[node]: | ||
if not dfs(pre): | ||
return False | ||
traversing.remove(node) | ||
|
||
visited.add(node) | ||
return True | ||
|
||
return all(dfs(i) for i in range(numCourses)) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,65 @@ | ||
''' | ||
# 226. Invert Binary Tree | ||
switch left and right child of each node | ||
## TC: O(N) | ||
visit each node once | ||
## SC: O(h) | ||
h is height of tree | ||
- best case: O(logN), balanced tree | ||
- worst case: O(N), skewed tree | ||
''' | ||
class Solution: | ||
''' | ||
DFS | ||
''' | ||
def invertTreeRecursive(self, root: Optional[TreeNode]) -> Optional[TreeNode]: | ||
if not root: | ||
return None | ||
|
||
root.left, root.right = root.right, root.left | ||
|
||
self.invertTree(root.left) | ||
self.invertTree(root.right) | ||
|
||
return root | ||
|
||
''' | ||
BFS | ||
- μ§κ΄μ μΈ stack νμ΄ | ||
''' | ||
def invertTree(self, root: Optional[TreeNode]) -> Optional[TreeNode]: | ||
stack = [root] | ||
|
||
while stack: | ||
node = stack.pop() | ||
if not node: | ||
continue | ||
|
||
node.left, node.right = node.right, node.left | ||
stack.append(node.left) | ||
stack.append(node.right) | ||
|
||
return root | ||
|
||
''' | ||
- μ°Έκ³ μ© deque νμ΄ | ||
''' | ||
def invertTreeDeque(self, root: Optional[TreeNode]) -> Optional[TreeNode]: | ||
dq = deque([root]) | ||
|
||
while dq: | ||
node = dq.popleft() | ||
if not node: | ||
continue | ||
|
||
node.left, node.right = node.right, node.left | ||
dq.append(node.left) | ||
dq.append(node.right) | ||
|
||
return root |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,45 @@ | ||
''' | ||
# 33. Search in Rotated Sorted Array | ||
binary search + condition check | ||
μ΄μ§ νμ μ€, μΌμͺ½μ΄ μ λ ¬λμμ λ | ||
- νκ²μ΄ μ λ ¬λ μΌμͺ½μ μλ κ²½μ°, μΌμͺ½ νμ (leftλΆν° mid - 1 μ¬μ΄μμ νκ²μ νμ) | ||
- νκ²μ΄ μ λ ¬λ μΌμͺ½μ μμ κ²½μ°, μ€λ₯Έμͺ½ νμ (mid + 1λΆν° right μ¬μ΄μμ νκ²μ νμ) | ||
μ΄μ§ νμ μ€, μ€λ₯Έμͺ½μ΄ μ λ ¬λμμ λ | ||
- νκ²μ΄ μ λ ¬λ μ€λ₯Έμͺ½μ μλ κ²½μ°, μ€λ₯Έμͺ½ νμ (mid + 1λΆν° right μ¬μ΄μμ νκ²μ νμ) | ||
- νκ²μ΄ μ λ ¬λ μ€λ₯Έμͺ½μ μμ κ²½μ°, μΌμͺ½ νμ (leftλΆν° mid - 1 μ¬μ΄μμ νκ²μ νμ) | ||
## TC: O(log n) | ||
binary search | ||
## SC: O(1) | ||
no extra space | ||
''' | ||
class Solution: | ||
def search(self, nums: List[int], target: int) -> int: | ||
left = 0 | ||
right = len(nums) - 1 | ||
|
||
while left <= right: | ||
mid = left + (right - left) // 2 | ||
|
||
if nums[mid] == target: | ||
return mid | ||
|
||
if nums[left] <= nums[mid]: # is_left_sorted | ||
if nums[left] <= target < nums[mid]: # is_target_left | ||
right = mid - 1 | ||
else: | ||
left = mid + 1 | ||
else: | ||
if nums[mid] < target <= nums[right]: # is_target_right | ||
left = mid + 1 | ||
else: | ||
right = mid - 1 | ||
|
||
return -1 |