배열의 반전은 다음을 나타냅니다. 배열을 정렬된 형식으로 변환하는 데 필요한 변경 수. 배열이 이미 정렬되어 있으면 반전이 0개 필요하고 다른 경우 배열이 반전되면 반전 수가 최대가 됩니다.
이 문제를 해결하기 위해 병합 정렬 방식을 따라 시간 복잡도를 줄이고 분할 정복 알고리즘으로 만들 것입니다.
입력 및 출력
Input: A sequence of numbers. (1, 5, 6, 4, 20). Output: The number of inversions required to arrange the numbers into ascending order. Here the number of inversions are 2. First inversion: (1, 5, 4, 6, 20) Second inversion: (1, 4, 5, 6, 20)
알고리즘
병합(배열, tempArray, 왼쪽, 중간, 오른쪽)
입력: 병합된 두 배열, 왼쪽, 오른쪽 및 중간 인덱스입니다.
출력: 정렬된 순서로 병합된 배열입니다.
Begin i := left, j := mid, k := right count := 0 while i <= mid -1 and j <= right, do if array[i] <= array[j], then tempArray[k] := array[i] increase i and k by 1 else tempArray[k] := array[j] increase j and k by 1 count := count + (mid - i) done while left part of the array has some extra element, do tempArray[k] := array[i] increase i and k by 1 done while right part of the array has some extra element, do tempArray[k] := array[j] increase j and k by 1 done return count End
mergeSort(배열, 임시 배열, 왼쪽, 오른쪽)
입력: 배열 및 임시 배열이 주어지면 배열의 왼쪽 및 오른쪽 인덱스입니다.
출력 - 정렬 후 반전 횟수입니다.
Begin count := 0 if right > left, then mid := (right + left)/2 count := mergeSort(array, tempArray, left, mid) count := count + mergeSort(array, tempArray, mid+1, right) count := count + merge(array, tempArray, left, mid+1, right) return count End
예
#include <iostream> using namespace std; int merge(intarr[], int temp[], int left, int mid, int right) { int i, j, k; int count = 0; i = left; //i to locate first array location j = mid; //i to locate second array location k = left; //i to locate merged array location while ((i <= mid - 1) && (j <= right)) { if (arr[i] <= arr[j]) { //when left item is less than right item temp[k++] = arr[i++]; }else{ temp[k++] = arr[j++]; count += (mid - i); //find how many convertion is performed } } while (i <= mid - 1) //if first list has remaining item, add them in the list temp[k++] = arr[i++]; while (j <= right) //if second list has remaining item, add them in the list temp[k++] = arr[j++]; for (i=left; i <= right; i++) arr[i] = temp[i]; //store temp Array to main array return count; } intmergeSort(intarr[], int temp[], int left, int right) { int mid, count = 0; if (right > left) { mid = (right + left)/2; //find mid index of the array count = mergeSort(arr, temp, left, mid); //merge sort left sub array count += mergeSort(arr, temp, mid+1, right); //merge sort right sub array count += merge(arr, temp, left, mid+1, right); //merge two sub arrays } return count; } intarrInversion(intarr[], int n) { int temp[n]; return mergeSort(arr, temp, 0, n - 1); } int main() { intarr[] = {1, 5, 6, 4, 20}; int n = 5; cout<< "Number of inversions are "<<arrInversion(arr, n); }
출력
Number of inversions are 2