Arrangement, dictionary order and next arrangement
Suppose you have an array or sequence,Next ArrangementIt refers to an arrangement that is larger in dictionary order than the current arrangement. If the current arrangement is already the largest arrangement, the next arrangement is the smallest arrangement.
For example, given an array[1, 2, 3]
, its next arrangement is[1, 3, 2]
;The next one is[2, 1, 3]
; and for[3, 2, 1]
, it is already the largest arrangement; generally stipulate that the next arrangement is cycled to the smallest arrangement, that is,[1, 2, 3]
。
The key to finding the next arrangement is to start from the current arrangement and use the properties of the dictionary order to find the next larger arrangement. The core steps of this algorithm are as follows:
-
Scan the array from right to left, find the first drop position. Find an element
nums[i]
, makenums[i] < nums[i + 1]
. The existence of this position means that the current arrangement can be larger. If no suchi
, indicating that the current arrangement is already the largest arrangement. -
Find the
nums[i]
The smallest element of the big:turn upi
The smallest element that is larger than him (becausei
After that, they are all decreasing, that is, the last one). -
Swap elements:exchange
nums[i]
andnums[j]
, this step will ensure that the new arrangement is larger. -
The reversal part:turn out to be
i
The element at the location has been replaced with a larger element; to ensure the minimum dictionary order of the arrangement, thenums[i+1]
arrivenums[n-1]
The part of the part changes from descending order to ascending order, and becomes the smallest arrangement.
Why is this right?
This is determined by the nature of the dictionary order. Each arrangement is composed of step-by-step exchange from left to right in the dictionary order.
- First, find the first element in non-descending order from back to forward, and determine the "can be added" part.
- It should exchange with the smallest element afterwards and re-order other elements to ensure that the added dictionary order is minimal.
void nextPermutation(vector<int>& nums) {
int n = ();
int i = n - 2;
while (i >= 0 && nums[i] >= nums[i + 1]) i--;
if (i >= 0) swap(nums[i], nums[n - 1]);
reverse(() + i + 1, ());
}
Many languages (including C++) come with this function, and the complexity is obviously\(O(n)\)。
Full arrangement problem
The full arrangement problem is given a length asn
array, requiring all permutations of the array to be returned. For example, given an array[1, 2, 3]
, we need to find out[1, 2, 3]
、[1, 3, 2]
、[2, 1, 3]
、[2, 3, 1]
、[3, 1, 2]
、[3, 2, 1]
These six arrangements.
One feature of a full permutation is that it generates all possible permutations and each permutation contains a different order of all elements in the array. All arrangements are available in total\(n!\), you can directly enumerate with the next arrangement method above, the calculation amount is\(n! * n\)
Another method of backtracking is to gradually fill each position in the arrangement through recursive way, and select different elements at each step, and finally output all possible arrangements.
- initialization: Starting with an empty arrangement, fill each position recursively.
- Recursive fill arrangement: Each time we recurse, we select one of the remaining unused numbers in order and put it into the current permutation, recursively generating the remaining parts.
- Backtracking: When recursion reaches the bottom (i.e., a permutation is generated), we go back to the previous layer and try other options.
- Termination conditions: When the length of the arrangement is equal to the length of the array, it means that an arrangement has been completely generated and the result is added.
void permute(vector<int>& nums, vector<int>& path, vector<vector<int>>& result, vector<bool>& visited) {
if (() == ()) {
result.push_back(path);
return;
}
for (int i = 0; i < (); i++) {
if (visited[i]) continue;
visited[i] = true;
path.push_back(nums[i]);
permute(nums, path, result, visited);
path.pop_back();
visited[i] = false;
}
}
-
Time complexity: The time complexity of the full arrangement problem is
O(n!)
, because generating all permutations requiresn!
each arrangement needs to be generatedO(n)
Time to copy and save. -
Space complexity: The space complexity is
O(n)
, recursive depth.
Expand knowledge
The order of arrangement originates from dictionary order, so this method is not only suitable for integer arrays, but can also be extended to other types of arrays (such as strings, characters, etc.). It is not required that the numbers be continuous1 ~ n
。