13.2 全順列問題¶
全順列問題はバックトラッキングアルゴリズムの典型的な応用例です。これは、ある集合(配列や文字列など)が与えられたとき、その要素のあり得るすべての順列を求める問題です。
下表に、入力配列とそれに対応するすべての順列から成る例をいくつか示します。
表 13-2 全順列の例
| 入力配列 | すべての順列 |
|---|---|
| \([1]\) | \([1]\) |
| \([1, 2]\) | \([1, 2], [2, 1]\) |
| \([1, 2, 3]\) | \([1, 2, 3], [1, 3, 2], [2, 1, 3], [2, 3, 1], [3, 1, 2], [3, 2, 1]\) |
13.2.1 等しい要素がない場合¶
Question
重複要素を含まない整数配列を入力として受け取り、あり得るすべての順列を返します。
バックトラッキングアルゴリズムの観点から見ると、順列生成の過程は一連の選択の結果として捉えられます。入力配列が \([1, 2, 3]\) だとすると、最初に \(1\) を選び、次に \(3\) を選び、最後に \(2\) を選べば、順列 \([1, 3, 2]\) が得られます。戻る操作は 1 つの選択を取り消し、その後で別の選択を試し続けることを表します。
バックトラッキングコードの観点では、候補集合 choices は入力配列中のすべての要素であり、状態 state は現時点までに選ばれた要素です。各要素は 1 回しか選べないことに注意してください。したがって state 内の要素はすべて一意でなければなりません。
下図のように、探索過程は再帰木として展開できます。木の各ノードは現在の状態 state を表します。根ノードから始めて 3 ラウンドの選択を経て葉ノードに到達し、各葉ノードが 1 つの順列に対応します。

図 13-5 全順列の再帰木
1. 重複選択の枝刈り¶
各要素が 1 回しか選ばれないようにするため、ブール配列 selected の導入を考えます。ここで selected[i] は choices[i] がすでに選ばれているかどうかを表し、これに基づいて次の枝刈りを行います。
- 選択
choice[i]を行った後、selected[i]を \(\text{True}\) に設定し、その要素が選択済みであることを表します。 - 選択肢リスト
choicesを走査するとき、すでに選ばれたノードはすべてスキップします。これが枝刈りです。
下図のように、1 回目に 1、2 回目に 3、3 回目に 2 を選ぶとします。このとき 2 回目では要素 1 の分岐を、3 回目では要素 1 と要素 3 の分岐を刈り取る必要があります。

図 13-6 全順列の枝刈り例
上図から、この枝刈りにより探索空間の大きさは \(O(n^n)\) から \(O(n!)\) へ削減されることがわかります。
2. コード実装¶
以上を整理できれば、フレームワークコードの「穴埋め」を行えます。全体のコードを短くするため、フレームワークコード中の各関数を個別には実装せず、これらを backtrack() 関数内に展開します。
def backtrack(
state: list[int], choices: list[int], selected: list[bool], res: list[list[int]]
):
"""バックトラッキング:順列 I"""
# 状態の長さが要素数に等しければ、解を記録
if len(state) == len(choices):
res.append(list(state))
return
# すべての選択肢を走査
for i, choice in enumerate(choices):
# 枝刈り:要素の重複選択を許可しない
if not selected[i]:
# 試行: 選択を行い、状態を更新
selected[i] = True
state.append(choice)
# 次の選択へ進む
backtrack(state, choices, selected, res)
# バックトラック:選択を取り消し、前の状態に戻す
selected[i] = False
state.pop()
def permutations_i(nums: list[int]) -> list[list[int]]:
"""全順列 I"""
res = []
backtrack(state=[], choices=nums, selected=[False] * len(nums), res=res)
return res
/* バックトラッキング:順列 I */
void backtrack(vector<int> &state, const vector<int> &choices, vector<bool> &selected, vector<vector<int>> &res) {
// 状態の長さが要素数に等しければ、解を記録
if (state.size() == choices.size()) {
res.push_back(state);
return;
}
// すべての選択肢を走査
for (int i = 0; i < choices.size(); i++) {
int choice = choices[i];
// 枝刈り:要素の重複選択を許可しない
if (!selected[i]) {
// 試行: 選択を行い、状態を更新
selected[i] = true;
state.push_back(choice);
// 次の選択へ進む
backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.pop_back();
}
}
}
/* 全順列 I */
vector<vector<int>> permutationsI(vector<int> nums) {
vector<int> state;
vector<bool> selected(nums.size(), false);
vector<vector<int>> res;
backtrack(state, nums, selected, res);
return res;
}
/* バックトラッキング:順列 I */
void backtrack(List<Integer> state, int[] choices, boolean[] selected, List<List<Integer>> res) {
// 状態の長さが要素数に等しければ、解を記録
if (state.size() == choices.length) {
res.add(new ArrayList<Integer>(state));
return;
}
// すべての選択肢を走査
for (int i = 0; i < choices.length; i++) {
int choice = choices[i];
// 枝刈り:要素の重複選択を許可しない
if (!selected[i]) {
// 試行: 選択を行い、状態を更新
selected[i] = true;
state.add(choice);
// 次の選択へ進む
backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.remove(state.size() - 1);
}
}
}
/* 全順列 I */
List<List<Integer>> permutationsI(int[] nums) {
List<List<Integer>> res = new ArrayList<List<Integer>>();
backtrack(new ArrayList<Integer>(), nums, new boolean[nums.length], res);
return res;
}
/* バックトラッキング:順列 I */
void Backtrack(List<int> state, int[] choices, bool[] selected, List<List<int>> res) {
// 状態の長さが要素数に等しければ、解を記録
if (state.Count == choices.Length) {
res.Add(new List<int>(state));
return;
}
// すべての選択肢を走査
for (int i = 0; i < choices.Length; i++) {
int choice = choices[i];
// 枝刈り:要素の重複選択を許可しない
if (!selected[i]) {
// 試行: 選択を行い、状態を更新
selected[i] = true;
state.Add(choice);
// 次の選択へ進む
Backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.RemoveAt(state.Count - 1);
}
}
}
/* 全順列 I */
List<List<int>> PermutationsI(int[] nums) {
List<List<int>> res = [];
Backtrack([], nums, new bool[nums.Length], res);
return res;
}
/* バックトラッキング:順列 I */
func backtrackI(state *[]int, choices *[]int, selected *[]bool, res *[][]int) {
// 状態の長さが要素数に等しければ、解を記録
if len(*state) == len(*choices) {
newState := append([]int{}, *state...)
*res = append(*res, newState)
}
// すべての選択肢を走査
for i := 0; i < len(*choices); i++ {
choice := (*choices)[i]
// 枝刈り:要素の重複選択を許可しない
if !(*selected)[i] {
// 試行: 選択を行い、状態を更新
(*selected)[i] = true
*state = append(*state, choice)
// 次の選択へ進む
backtrackI(state, choices, selected, res)
// バックトラック:選択を取り消し、前の状態に戻す
(*selected)[i] = false
*state = (*state)[:len(*state)-1]
}
}
}
/* 全順列 I */
func permutationsI(nums []int) [][]int {
res := make([][]int, 0)
state := make([]int, 0)
selected := make([]bool, len(nums))
backtrackI(&state, &nums, &selected, &res)
return res
}
/* バックトラッキング:順列 I */
func backtrack(state: inout [Int], choices: [Int], selected: inout [Bool], res: inout [[Int]]) {
// 状態の長さが要素数に等しければ、解を記録
if state.count == choices.count {
res.append(state)
return
}
// すべての選択肢を走査
for (i, choice) in choices.enumerated() {
// 枝刈り:要素の重複選択を許可しない
if !selected[i] {
// 試行: 選択を行い、状態を更新
selected[i] = true
state.append(choice)
// 次の選択へ進む
backtrack(state: &state, choices: choices, selected: &selected, res: &res)
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false
state.removeLast()
}
}
}
/* 全順列 I */
func permutationsI(nums: [Int]) -> [[Int]] {
var state: [Int] = []
var selected = Array(repeating: false, count: nums.count)
var res: [[Int]] = []
backtrack(state: &state, choices: nums, selected: &selected, res: &res)
return res
}
/* バックトラッキング:順列 I */
function backtrack(state, choices, selected, res) {
// 状態の長さが要素数に等しければ、解を記録
if (state.length === choices.length) {
res.push([...state]);
return;
}
// すべての選択肢を走査
choices.forEach((choice, i) => {
// 枝刈り:要素の重複選択を許可しない
if (!selected[i]) {
// 試行: 選択を行い、状態を更新
selected[i] = true;
state.push(choice);
// 次の選択へ進む
backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.pop();
}
});
}
/* 全順列 I */
function permutationsI(nums) {
const res = [];
backtrack([], nums, Array(nums.length).fill(false), res);
return res;
}
/* バックトラッキング:順列 I */
function backtrack(
state: number[],
choices: number[],
selected: boolean[],
res: number[][]
): void {
// 状態の長さが要素数に等しければ、解を記録
if (state.length === choices.length) {
res.push([...state]);
return;
}
// すべての選択肢を走査
choices.forEach((choice, i) => {
// 枝刈り:要素の重複選択を許可しない
if (!selected[i]) {
// 試行: 選択を行い、状態を更新
selected[i] = true;
state.push(choice);
// 次の選択へ進む
backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.pop();
}
});
}
/* 全順列 I */
function permutationsI(nums: number[]): number[][] {
const res: number[][] = [];
backtrack([], nums, Array(nums.length).fill(false), res);
return res;
}
/* バックトラッキング:順列 I */
void backtrack(
List<int> state,
List<int> choices,
List<bool> selected,
List<List<int>> res,
) {
// 状態の長さが要素数に等しければ、解を記録
if (state.length == choices.length) {
res.add(List.from(state));
return;
}
// すべての選択肢を走査
for (int i = 0; i < choices.length; i++) {
int choice = choices[i];
// 枝刈り:要素の重複選択を許可しない
if (!selected[i]) {
// 試行: 選択を行い、状態を更新
selected[i] = true;
state.add(choice);
// 次の選択へ進む
backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.removeLast();
}
}
}
/* 全順列 I */
List<List<int>> permutationsI(List<int> nums) {
List<List<int>> res = [];
backtrack([], nums, List.filled(nums.length, false), res);
return res;
}
/* バックトラッキング:順列 I */
fn backtrack(mut state: Vec<i32>, choices: &[i32], selected: &mut [bool], res: &mut Vec<Vec<i32>>) {
// 状態の長さが要素数に等しければ、解を記録
if state.len() == choices.len() {
res.push(state);
return;
}
// すべての選択肢を走査
for i in 0..choices.len() {
let choice = choices[i];
// 枝刈り:要素の重複選択を許可しない
if !selected[i] {
// 試行: 選択を行い、状態を更新
selected[i] = true;
state.push(choice);
// 次の選択へ進む
backtrack(state.clone(), choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.pop();
}
}
}
/* 全順列 I */
fn permutations_i(nums: &mut [i32]) -> Vec<Vec<i32>> {
let mut res = Vec::new(); // 状態(部分集合)
backtrack(Vec::new(), nums, &mut vec![false; nums.len()], &mut res);
res
}
/* バックトラッキング:順列 I */
void backtrack(int *state, int stateSize, int *choices, int choicesSize, bool *selected, int **res, int *resSize) {
// 状態の長さが要素数に等しければ、解を記録
if (stateSize == choicesSize) {
res[*resSize] = (int *)malloc(choicesSize * sizeof(int));
for (int i = 0; i < choicesSize; i++) {
res[*resSize][i] = state[i];
}
(*resSize)++;
return;
}
// すべての選択肢を走査
for (int i = 0; i < choicesSize; i++) {
int choice = choices[i];
// 枝刈り:要素の重複選択を許可しない
if (!selected[i]) {
// 試行: 選択を行い、状態を更新
selected[i] = true;
state[stateSize] = choice;
// 次の選択へ進む
backtrack(state, stateSize + 1, choices, choicesSize, selected, res, resSize);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
}
}
}
/* 全順列 I */
int **permutationsI(int *nums, int numsSize, int *returnSize) {
int *state = (int *)malloc(numsSize * sizeof(int));
bool *selected = (bool *)malloc(numsSize * sizeof(bool));
for (int i = 0; i < numsSize; i++) {
selected[i] = false;
}
int **res = (int **)malloc(MAX_SIZE * sizeof(int *));
*returnSize = 0;
backtrack(state, 0, nums, numsSize, selected, res, returnSize);
free(state);
free(selected);
return res;
}
/* バックトラッキング:順列 I */
fun backtrack(
state: MutableList<Int>,
choices: IntArray,
selected: BooleanArray,
res: MutableList<MutableList<Int>?>
) {
// 状態の長さが要素数に等しければ、解を記録
if (state.size == choices.size) {
res.add(state.toMutableList())
return
}
// すべての選択肢を走査
for (i in choices.indices) {
val choice = choices[i]
// 枝刈り:要素の重複選択を許可しない
if (!selected[i]) {
// 試行: 選択を行い、状態を更新
selected[i] = true
state.add(choice)
// 次の選択へ進む
backtrack(state, choices, selected, res)
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false
state.removeAt(state.size - 1)
}
}
}
/* 全順列 I */
fun permutationsI(nums: IntArray): MutableList<MutableList<Int>?> {
val res = mutableListOf<MutableList<Int>?>()
backtrack(mutableListOf(), nums, BooleanArray(nums.size), res)
return res
}
### バックトラッキング法:全順列 I ###
def backtrack(state, choices, selected, res)
# 状態の長さが要素数に等しければ、解を記録
if state.length == choices.length
res << state.dup
return
end
# すべての選択肢を走査
choices.each_with_index do |choice, i|
# 枝刈り:要素の重複選択を許可しない
unless selected[i]
# 試行: 選択を行い、状態を更新
selected[i] = true
state << choice
# 次の選択へ進む
backtrack(state, choices, selected, res)
# バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false
state.pop
end
end
end
### 全順列 I ###
def permutations_i(nums)
res = []
backtrack([], nums, Array.new(nums.length, false), res)
res
end
コードの可視化
13.2.2 等しい要素を考慮する場合¶
Question
整数配列を入力として受け取り、配列には重複要素が含まれる場合があります。重複しない順列をすべて返します。
入力配列が \([1, 1, 2]\) だと仮定します。2 つの重複する要素 \(1\) を区別しやすくするため、2 つ目の \(1\) を \(\hat{1}\) と記します。
下図のように、上述の方法で生成される順列の半分は重複しています。

図 13-7 重複した順列
では、重複した順列をどのように取り除けばよいのでしょうか。最も直接的なのは、ハッシュ集合を用いて順列結果をそのまま重複排除する方法です。しかしこのやり方は十分に洗練されていません。なぜなら、重複順列を生成する探索分岐はそもそも不要であり、事前に見つけて枝刈りすべきだからです。そうすることで、アルゴリズム効率をさらに高められます。
1. 等しい要素の枝刈り¶
下図を見ると、1 回目のラウンドでは \(1\) を選ぶことと \(\hat{1}\) を選ぶことは等価であり、これら 2 つの選択の下で生成される順列はすべて重複します。したがって \(\hat{1}\) を枝刈りすべきです。
同様に、1 回目で \(2\) を選んだ後では、2 回目のラウンドにおける \(1\) と \(\hat{1}\) も重複分岐を生むため、2 回目の \(\hat{1}\) も枝刈りすべきです。
本質的には、各ラウンドの選択において、等しい複数の要素が 1 回しか選ばれないようにすることが目標です。

図 13-8 重複順列の枝刈り
2. コード実装¶
前問のコードを土台として、各ラウンドの選択でハッシュ集合 duplicated を 1 つ用意し、そのラウンドですでに試した要素を記録して、重複要素を枝刈りすることを考えます。
def backtrack(
state: list[int], choices: list[int], selected: list[bool], res: list[list[int]]
):
"""バックトラッキング:順列 II"""
# 状態の長さが要素数に等しければ、解を記録
if len(state) == len(choices):
res.append(list(state))
return
# すべての選択肢を走査
duplicated = set[int]()
for i, choice in enumerate(choices):
# 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if not selected[i] and choice not in duplicated:
# 試行: 選択を行い、状態を更新
duplicated.add(choice) # 選択済みの要素値を記録
selected[i] = True
state.append(choice)
# 次の選択へ進む
backtrack(state, choices, selected, res)
# バックトラック:選択を取り消し、前の状態に戻す
selected[i] = False
state.pop()
def permutations_ii(nums: list[int]) -> list[list[int]]:
"""全順列 II"""
res = []
backtrack(state=[], choices=nums, selected=[False] * len(nums), res=res)
return res
/* バックトラッキング:順列 II */
void backtrack(vector<int> &state, const vector<int> &choices, vector<bool> &selected, vector<vector<int>> &res) {
// 状態の長さが要素数に等しければ、解を記録
if (state.size() == choices.size()) {
res.push_back(state);
return;
}
// すべての選択肢を走査
unordered_set<int> duplicated;
for (int i = 0; i < choices.size(); i++) {
int choice = choices[i];
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if (!selected[i] && duplicated.find(choice) == duplicated.end()) {
// 試行: 選択を行い、状態を更新
duplicated.emplace(choice); // 選択済みの要素値を記録
selected[i] = true;
state.push_back(choice);
// 次の選択へ進む
backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.pop_back();
}
}
}
/* 全順列 II */
vector<vector<int>> permutationsII(vector<int> nums) {
vector<int> state;
vector<bool> selected(nums.size(), false);
vector<vector<int>> res;
backtrack(state, nums, selected, res);
return res;
}
/* バックトラッキング:順列 II */
void backtrack(List<Integer> state, int[] choices, boolean[] selected, List<List<Integer>> res) {
// 状態の長さが要素数に等しければ、解を記録
if (state.size() == choices.length) {
res.add(new ArrayList<Integer>(state));
return;
}
// すべての選択肢を走査
Set<Integer> duplicated = new HashSet<Integer>();
for (int i = 0; i < choices.length; i++) {
int choice = choices[i];
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if (!selected[i] && !duplicated.contains(choice)) {
// 試行: 選択を行い、状態を更新
duplicated.add(choice); // 選択済みの要素値を記録
selected[i] = true;
state.add(choice);
// 次の選択へ進む
backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.remove(state.size() - 1);
}
}
}
/* 全順列 II */
List<List<Integer>> permutationsII(int[] nums) {
List<List<Integer>> res = new ArrayList<List<Integer>>();
backtrack(new ArrayList<Integer>(), nums, new boolean[nums.length], res);
return res;
}
/* バックトラッキング:順列 II */
void Backtrack(List<int> state, int[] choices, bool[] selected, List<List<int>> res) {
// 状態の長さが要素数に等しければ、解を記録
if (state.Count == choices.Length) {
res.Add(new List<int>(state));
return;
}
// すべての選択肢を走査
HashSet<int> duplicated = [];
for (int i = 0; i < choices.Length; i++) {
int choice = choices[i];
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if (!selected[i] && !duplicated.Contains(choice)) {
// 試行: 選択を行い、状態を更新
duplicated.Add(choice); // 選択済みの要素値を記録
selected[i] = true;
state.Add(choice);
// 次の選択へ進む
Backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.RemoveAt(state.Count - 1);
}
}
}
/* 全順列 II */
List<List<int>> PermutationsII(int[] nums) {
List<List<int>> res = [];
Backtrack([], nums, new bool[nums.Length], res);
return res;
}
/* バックトラッキング:順列 II */
func backtrackII(state *[]int, choices *[]int, selected *[]bool, res *[][]int) {
// 状態の長さが要素数に等しければ、解を記録
if len(*state) == len(*choices) {
newState := append([]int{}, *state...)
*res = append(*res, newState)
}
// すべての選択肢を走査
duplicated := make(map[int]struct{}, 0)
for i := 0; i < len(*choices); i++ {
choice := (*choices)[i]
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if _, ok := duplicated[choice]; !ok && !(*selected)[i] {
// 試す: 選択を行って状態を更新
// 選択済みの要素値を記録
duplicated[choice] = struct{}{}
(*selected)[i] = true
*state = append(*state, choice)
// 次の選択へ進む
backtrackII(state, choices, selected, res)
// バックトラック:選択を取り消し、前の状態に戻す
(*selected)[i] = false
*state = (*state)[:len(*state)-1]
}
}
}
/* 全順列 II */
func permutationsII(nums []int) [][]int {
res := make([][]int, 0)
state := make([]int, 0)
selected := make([]bool, len(nums))
backtrackII(&state, &nums, &selected, &res)
return res
}
/* バックトラッキング:順列 II */
func backtrack(state: inout [Int], choices: [Int], selected: inout [Bool], res: inout [[Int]]) {
// 状態の長さが要素数に等しければ、解を記録
if state.count == choices.count {
res.append(state)
return
}
// すべての選択肢を走査
var duplicated: Set<Int> = []
for (i, choice) in choices.enumerated() {
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if !selected[i], !duplicated.contains(choice) {
// 試行: 選択を行い、状態を更新
duplicated.insert(choice) // 選択済みの要素値を記録
selected[i] = true
state.append(choice)
// 次の選択へ進む
backtrack(state: &state, choices: choices, selected: &selected, res: &res)
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false
state.removeLast()
}
}
}
/* 全順列 II */
func permutationsII(nums: [Int]) -> [[Int]] {
var state: [Int] = []
var selected = Array(repeating: false, count: nums.count)
var res: [[Int]] = []
backtrack(state: &state, choices: nums, selected: &selected, res: &res)
return res
}
/* バックトラッキング:順列 II */
function backtrack(state, choices, selected, res) {
// 状態の長さが要素数に等しければ、解を記録
if (state.length === choices.length) {
res.push([...state]);
return;
}
// すべての選択肢を走査
const duplicated = new Set();
choices.forEach((choice, i) => {
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if (!selected[i] && !duplicated.has(choice)) {
// 試行: 選択を行い、状態を更新
duplicated.add(choice); // 選択済みの要素値を記録
selected[i] = true;
state.push(choice);
// 次の選択へ進む
backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.pop();
}
});
}
/* 全順列 II */
function permutationsII(nums) {
const res = [];
backtrack([], nums, Array(nums.length).fill(false), res);
return res;
}
/* バックトラッキング:順列 II */
function backtrack(
state: number[],
choices: number[],
selected: boolean[],
res: number[][]
): void {
// 状態の長さが要素数に等しければ、解を記録
if (state.length === choices.length) {
res.push([...state]);
return;
}
// すべての選択肢を走査
const duplicated = new Set();
choices.forEach((choice, i) => {
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if (!selected[i] && !duplicated.has(choice)) {
// 試行: 選択を行い、状態を更新
duplicated.add(choice); // 選択済みの要素値を記録
selected[i] = true;
state.push(choice);
// 次の選択へ進む
backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.pop();
}
});
}
/* 全順列 II */
function permutationsII(nums: number[]): number[][] {
const res: number[][] = [];
backtrack([], nums, Array(nums.length).fill(false), res);
return res;
}
/* バックトラッキング:順列 II */
void backtrack(
List<int> state,
List<int> choices,
List<bool> selected,
List<List<int>> res,
) {
// 状態の長さが要素数に等しければ、解を記録
if (state.length == choices.length) {
res.add(List.from(state));
return;
}
// すべての選択肢を走査
Set<int> duplicated = {};
for (int i = 0; i < choices.length; i++) {
int choice = choices[i];
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if (!selected[i] && !duplicated.contains(choice)) {
// 試行: 選択を行い、状態を更新
duplicated.add(choice); // 選択済みの要素値を記録
selected[i] = true;
state.add(choice);
// 次の選択へ進む
backtrack(state, choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.removeLast();
}
}
}
/* 全順列 II */
List<List<int>> permutationsII(List<int> nums) {
List<List<int>> res = [];
backtrack([], nums, List.filled(nums.length, false), res);
return res;
}
/* バックトラッキング:順列 II */
fn backtrack(mut state: Vec<i32>, choices: &[i32], selected: &mut [bool], res: &mut Vec<Vec<i32>>) {
// 状態の長さが要素数に等しければ、解を記録
if state.len() == choices.len() {
res.push(state);
return;
}
// すべての選択肢を走査
let mut duplicated = HashSet::<i32>::new();
for i in 0..choices.len() {
let choice = choices[i];
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if !selected[i] && !duplicated.contains(&choice) {
// 試行: 選択を行い、状態を更新
duplicated.insert(choice); // 選択済みの要素値を記録
selected[i] = true;
state.push(choice);
// 次の選択へ進む
backtrack(state.clone(), choices, selected, res);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
state.pop();
}
}
}
/* 全順列 II */
fn permutations_ii(nums: &mut [i32]) -> Vec<Vec<i32>> {
let mut res = Vec::new();
backtrack(Vec::new(), nums, &mut vec![false; nums.len()], &mut res);
res
}
/* バックトラッキング:順列 II */
void backtrack(int *state, int stateSize, int *choices, int choicesSize, bool *selected, int **res, int *resSize) {
// 状態の長さが要素数に等しければ、解を記録
if (stateSize == choicesSize) {
res[*resSize] = (int *)malloc(choicesSize * sizeof(int));
for (int i = 0; i < choicesSize; i++) {
res[*resSize][i] = state[i];
}
(*resSize)++;
return;
}
// すべての選択肢を走査
bool duplicated[MAX_SIZE] = {false};
for (int i = 0; i < choicesSize; i++) {
int choice = choices[i];
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if (!selected[i] && !duplicated[choice]) {
// 試行: 選択を行い、状態を更新
duplicated[choice] = true; // 選択済みの要素値を記録
selected[i] = true;
state[stateSize] = choice;
// 次の選択へ進む
backtrack(state, stateSize + 1, choices, choicesSize, selected, res, resSize);
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false;
}
}
}
/* 全順列 II */
int **permutationsII(int *nums, int numsSize, int *returnSize) {
int *state = (int *)malloc(numsSize * sizeof(int));
bool *selected = (bool *)malloc(numsSize * sizeof(bool));
for (int i = 0; i < numsSize; i++) {
selected[i] = false;
}
int **res = (int **)malloc(MAX_SIZE * sizeof(int *));
*returnSize = 0;
backtrack(state, 0, nums, numsSize, selected, res, returnSize);
free(state);
free(selected);
return res;
}
/* バックトラッキング:順列 II */
fun backtrack(
state: MutableList<Int>,
choices: IntArray,
selected: BooleanArray,
res: MutableList<MutableList<Int>?>
) {
// 状態の長さが要素数に等しければ、解を記録
if (state.size == choices.size) {
res.add(state.toMutableList())
return
}
// すべての選択肢を走査
val duplicated = HashSet<Int>()
for (i in choices.indices) {
val choice = choices[i]
// 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if (!selected[i] && !duplicated.contains(choice)) {
// 試行: 選択を行い、状態を更新
duplicated.add(choice) // 選択済みの要素値を記録
selected[i] = true
state.add(choice)
// 次の選択へ進む
backtrack(state, choices, selected, res)
// バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false
state.removeAt(state.size - 1)
}
}
}
/* 全順列 II */
fun permutationsII(nums: IntArray): MutableList<MutableList<Int>?> {
val res = mutableListOf<MutableList<Int>?>()
backtrack(mutableListOf(), nums, BooleanArray(nums.size), res)
return res
}
### バックトラッキング法:全順列 II ###
def backtrack(state, choices, selected, res)
# 状態の長さが要素数に等しければ、解を記録
if state.length == choices.length
res << state.dup
return
end
# すべての選択肢を走査
duplicated = Set.new
choices.each_with_index do |choice, i|
# 枝刈り:要素の重複選択を許可せず、同値要素の重複選択も許可しない
if !selected[i] && !duplicated.include?(choice)
# 試行: 選択を行い、状態を更新
duplicated.add(choice)
selected[i] = true
state << choice
# 次の選択へ進む
backtrack(state, choices, selected, res)
# バックトラック:選択を取り消し、前の状態に戻す
selected[i] = false
state.pop
end
end
end
### 全順列 II ###
def permutations_ii(nums)
res = []
backtrack([], nums, Array.new(nums.length, false), res)
res
end
コードの可視化
要素どうしがすべて互いに異なると仮定すると、\(n\) 個の要素には全部で \(n!\) 通りの順列(階乗)があります。結果を記録する際には、長さ \(n\) のリストをコピーする必要があり、これに \(O(n)\) 時間を要します。したがって時間計算量は \(O(n!n)\) です。
再帰の最大深さは \(n\) であり、\(O(n)\) のスタックフレーム空間を使います。selected は \(O(n)\) 空間を使用します。同時刻に存在する duplicated は最大で \(n\) 個であり、\(O(n^2)\) 空間を要します。したがって空間計算量は \(O(n^2)\) です。
3. 2 種類の枝刈りの比較¶
selected と duplicated はどちらも枝刈りに用いられますが、目的は異なる点に注意してください。
- 重複選択の枝刈り:探索全体を通して
selectedは 1 つだけです。これは現在の状態にどの要素が含まれているかを記録し、ある要素がstateに重複して現れるのを防ぎます。 - 等しい要素の枝刈り:各ラウンドの選択、すなわち各回の
backtrack呼び出しにはduplicatedが含まれます。これはそのラウンドの走査(forループ)でどの要素がすでに選ばれたかを記録し、等しい要素が 1 回しか選ばれないことを保証します。
下図は、2 つの枝刈り条件が有効になる範囲を示しています。木の各ノードは 1 つの選択を表し、根ノードから葉ノードまでの経路上の各ノードが 1 つの順列を構成することに注意してください。

図 13-9 2 種類の枝刈り条件の作用範囲