EDIT: My solution is added to the end of the question. Thanks for the hint.
I'll just go with an example. Suppose I have an array with length n
:
arr = { 1, 4, 8, 2, 5, ... }
If I want to traverse all combinations of TWO elements I would write:
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
// do something with arr[i] and arr[j]
}
}
I If I want to traverse all configurations of THREE elements I would simply add another layer of for
iteration:
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
for (int k = 0; k < n; k++) {
// do something with arr[i] and arr[j]
}
}
}
What if the number of elements is given by user (say m
), and we don't know exactly what it is? What should I write then?
(I couldn't figure out the best title for this question. And the tags are not accurate either. Help me with these too, if you want.)
Answer
The solution is this function:
void configurations(int depth, int* array, int length, int* indices, int curDepth) {
if (curDepth == 0) {
for (int i = 0; i < depth; i++) {
printf("%d ", indices[i]);
}
printf("\n");
return;
}
for (int i = 0; i < length; i++) {
indices[curDepth - 1] = i;
configurations(depth, array, length, indices, curDepth - 1);
}
}
The usage of the function above is shown below:
int a[] = {9, 8, 7, 6, 5, 4, 3, 2, 1};
int configSize = 3;
int* indices = new int[configSize];
configurations(configSize, a, 9, indices, configSize);
The console will show all configurations of the given size of the elements in the array:
0 0 0
1 0 0
2 0 0
3 0 0
4 0 0
...
5 8 8
6 8 8
7 8 8
8 8 8
You can simply use recursion.
Some pseudo-code:
void someFunction(int[] arr, int n, int depth)
{
if (depth == 0)
{
// do something with the stored elements
return;
}
for (int i = 0; i < n; i++)
{
// store arr[i]
someFunction(arr, n, depth-1);
}
}
There are a few ways to store arr[i]
. One way could be to pass an array of size initialDepth
down via the recursive call, along with the current index in that array. We increase the index on every recursive call, and put arr[i]
at the current index. Then, when the depth == 0
if-statement triggers, we'll have an array containing a permutation, which we could do whatever with.
This, as your code, would repeat elements (i.e. one permutation will consist exclusively of the first element repeated a few times). If you wish to avoid this, you can instead swap the first element with each other element at the first step, then recurse, swapping the second element with each other element at the second step, and so on.
void someFunction(int[] arr, int n, int pos, int depth)
{
if (pos == depth)
{
// do something with the elements in arr from 0 to pos
return;
}
for (int i = pos; i < n; i++)
{
// swap arr[pos] and arr[i]
someFunction(arr, n, pos+1, depth);
// swap arr[pos] and arr[i] back
}
}
Call with someFunction(inputArray, n, 0, desiredDepth)
.
Use backtracking (depth-first search). The general idea will be something like this:
void Generate(int n) {
if ((0..n).count(i => used[i]) >= NUMBER_OF_ELEMENTS_DESIRED_FOR_A_PERMUTATION) {
print this permutation;
return;
}
used[n] = true;
foreach (int next in (0..n).where(i => !used[i]))
Generate(next);
used[n] = false;
}
You can be use recursion, Something like this:
public static void main(String[] args) {
char[] alphabet = new char[] {'a','f','j'};
possibleStrings(2, alphabet,"");
}
public static void possibleStrings(int maxLength, char[] alphabet, String curr) {
if(curr.length() == maxLength) {
System.out.println(curr);
} else {
for(int i = 0; i < alphabet.length; i++) {
String oldCurr = curr;
curr += alphabet[i];
possibleStrings(maxLength,alphabet,curr);
curr = oldCurr;
}
}
}