deck:: Logseq/coding tip - ## **► 개념(Concept)** - 1차원 배열(리스트)에서 {{c1 두 개의 포인터(인덱스)}}를 조작하여 원하는 결과를 찾아내는 기법. id:: 6972072d-bf82-4f1a-a003-ca1527fb112b 보통 이중 반복문 로직 {{c1 ($O(N^2)$)}}을 최적화하여 {{c1 ($O(N)$)}}의 시간 복잡도로 결과를 얻어내게 됨. - - ## **► 유형(pattern)** - ### ◼︎ 양 끝에서 좁혀오는 방식(Collision) - #### ◉ 동작 원리 - id:: 69720ad1-9712-4934-b0e4-daa540fdd95f 1. 길이가 N인 1차원 배열에 대해 두 포인터 left, right가 각각 {{c1 0}}, {{c1 N-1}}에서 시작. 2. 조건에 따라 left는 {{c1 오른쪽으로(+1)}}, right는 {{c1 왼쪽으로(-1)}} 이동. 3. 두 포인터가 {{c1 교차하거나 만날 때 (left >= right)}} 종료 - #### ◉ 주요 용도 #card id:: 69720bbd-d10e-49b7-bcf7-5cc75d2c7428 - 정렬된 배열에서 두 수의 합 찾기. - 팰린드롬 검사 extra:: 팰린드롬(palindrome)이란, 뒤집어도 원본과 일치하는 배열을 의미 - #### ◉ 예제 - 정렬된 배열 nums에서 합이 target이 되는 두 수를 찾는 파이썬 함수를 작성하라. id:: 69720c2d-cb74-49c2-9b61-41979aca8468 (target을 입력으로 받는 함수이고 nums는 전역변수로 선언되어 있고, 정답 조합은 반드시 존재하고 오직 한개만 존재한다고 가정한다. 출력값은 튜플 형태로 두 수의 인덱스를 반환한다.) #card - ```python def twoSum(target) : left, right = 0, len(nums) - 1 while left < right : curr_sum = nums[left] + nums[right] if curr_sum == target : return (left, right) elif curr_sum < target : left += 1 else : right -= 1 ``` - ### ◼︎ 같은 방향으로 이동하는 방식(Forwarding) - #### ◉ 동작 원리 - id:: d44956cd-a48f-43c5-9b1b-e7a7d0b54e9c 1. 길이가 N인 1차원 배열에 대해 두 포인터 left, right가 각각 {{c1 0}}, {{c1 0}}에서 시작. 2. {{c1 right}}를 오른쪽으로 늘려가며 범위를 확장. 3. 특정 조건을 만족하면 {{c1 left}}를 오른쪽으로 이동해서 범위를 축소. 4. {{c1 right}}가 끝에 도달하면 종료 - 위의 방식의 응용으로 만약 left와 right의 폭이 고정적인 상태로 움직인다면 이를 {{c1 슬라이딩 윈도우(sliding window)}} 기법이라고도 부른다. id:: 6972102a-a070-4903-8a30-1b9c29ed7922 - #### ◉ 주요 용도 #card id:: fcd552eb-b6a4-436c-8cb4-93844aa1a1de - 부분 배열의 합 - 특정 조건을 만족하는 연속 구간 찾기 - #### ◉ 예제 - 배열 nums에서 합이 M이 되는 연속 부분 수열의 개수를 찾는 파이썬 코드를 작성하라. id:: 6d5397f3-9cad-4a0f-9d2c-1a093047290c #card - ```python count = 0 interval_sum = 0 right = 0 for left in range(len(nums)) : while interval_sum < M and right < len(nums) : interval_sum += nums[right] right += 1 if interval_sum == M : count += 1 interval_sum -= nums[left] print(count) ```