Refactor `Permutations` in `EnumerableExtensions`, add `PermutationsWithRepeats` method, and update tests for improved validation and clarity
This commit is contained in:
parent
0665713725
commit
9d4051ece1
|
|
@ -9,9 +9,8 @@ public class EnumerableExtensionsTest
|
|||
int[] data = [1, 2];
|
||||
|
||||
var actual = data.Combinations(2).ToArray();
|
||||
|
||||
Assert.Contains([1, 2], actual);
|
||||
Assert.Single(actual);
|
||||
|
||||
Assert.Equal([[1, 2]], actual);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
|
|
@ -21,10 +20,7 @@ public class EnumerableExtensionsTest
|
|||
|
||||
var actual = data.Combinations(2).ToArray();
|
||||
|
||||
Assert.Contains([1, 2], actual);
|
||||
Assert.Contains([1, 3], actual);
|
||||
Assert.Contains([2, 3], actual);
|
||||
Assert.Equal(3, actual.Length);
|
||||
Assert.Equal([[1, 2], [1, 3], [2, 3]], actual);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
|
|
@ -33,9 +29,8 @@ public class EnumerableExtensionsTest
|
|||
int[] data = [1, 2, 3];
|
||||
|
||||
var actual = data.Combinations(3).ToArray();
|
||||
|
||||
Assert.Contains([1, 2, 3], actual);
|
||||
Assert.Single(actual);
|
||||
|
||||
Assert.Equal([[1, 2, 3]], actual);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
|
|
@ -69,33 +64,42 @@ public class EnumerableExtensionsTest
|
|||
}
|
||||
|
||||
[Fact]
|
||||
public void Permutations_12_2_equals_11_12_21_22()
|
||||
public void Permutations_repeated_12_2_equals_11_12_21_22()
|
||||
{
|
||||
int[] data = [1, 2];
|
||||
|
||||
var actual = data.PermutationsWithRepeats(2).ToArray();
|
||||
|
||||
Assert.Equal([[1, 1], [1, 2], [2, 1], [2, 2]], actual);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void Permutations_12_2_equals_12_21()
|
||||
{
|
||||
int[] data = [1, 2];
|
||||
|
||||
var actual = data.Permutations(2).ToArray();
|
||||
|
||||
Assert.Contains([1, 1], actual);
|
||||
Assert.Contains([1, 2], actual);
|
||||
Assert.Contains([2, 1], actual);
|
||||
Assert.Contains([2, 2], actual);
|
||||
Assert.Equal(4, actual.Length);
|
||||
Assert.Equal([[1, 2], [2, 1]], actual);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void Permutations_123_2_equals_11_12_13_21_22_23_31_32_33()
|
||||
public void Permutations_repeated_123_2_equals_11_12_13_21_22_23_31_32_33()
|
||||
{
|
||||
int[] data = [1, 2, 3];
|
||||
|
||||
var actual = data.PermutationsWithRepeats(2).ToArray();
|
||||
|
||||
Assert.Equal([[1, 1], [1, 2], [1, 3], [2, 1], [2, 2], [2, 3], [3, 1], [3, 2], [3, 3]], actual);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public void Permutations_123_2_equals_12_13_21_23_31_32()
|
||||
{
|
||||
int[] data = [1, 2, 3];
|
||||
|
||||
var actual = data.Permutations(2).ToArray();
|
||||
|
||||
Assert.Contains([1, 1], actual);
|
||||
Assert.Contains([1, 2], actual);
|
||||
Assert.Contains([2, 1], actual);
|
||||
Assert.Contains([2, 2], actual);
|
||||
Assert.Contains([3, 1], actual);
|
||||
Assert.Contains([3, 2], actual);
|
||||
Assert.Contains([3, 3], actual);
|
||||
Assert.Equal(9, actual.Length);
|
||||
Assert.Equal([[1, 2], [1, 3], [2, 1], [2, 3], [3, 1], [3, 2]], actual);
|
||||
}
|
||||
}
|
||||
|
|
@ -68,7 +68,7 @@ public static class EnumerableExtensions
|
|||
}
|
||||
}
|
||||
|
||||
public static IEnumerable<TValue[]> Permutations<TValue>(this IEnumerable<TValue> values, int count)
|
||||
public static IEnumerable<TValue[]> PermutationsWithRepeats<TValue>(this IEnumerable<TValue> values, int count)
|
||||
{
|
||||
var pool = values.ToArray();
|
||||
if (count > pool.Length)
|
||||
|
|
@ -96,6 +96,47 @@ public static class EnumerableExtensions
|
|||
}
|
||||
}
|
||||
|
||||
public static IEnumerable<TValue[]> Permutations<TValue>(this IEnumerable<TValue> values, int count)
|
||||
{
|
||||
var pool = values.ToArray();
|
||||
var poolLength = pool.Length;
|
||||
if (count > poolLength)
|
||||
yield break;
|
||||
var indices = Enumerable.Range(0, poolLength).ToArray();
|
||||
var cycles = Enumerable.Range(0, poolLength + 1)
|
||||
.Skip(poolLength - count + 1)
|
||||
.Reverse()
|
||||
.ToArray();
|
||||
yield return GetCombination(indices[..count], pool);
|
||||
var notDone = true;
|
||||
while (notDone)
|
||||
{
|
||||
notDone = false;
|
||||
for (var i = count - 1; i >= 0; i--)
|
||||
{
|
||||
cycles[i] -= 1;
|
||||
if (cycles[i] == 0)
|
||||
{
|
||||
var tmp = indices[i];
|
||||
for (var j = i; j < indices.Length - 1; j++)
|
||||
{
|
||||
indices[j] = indices[j + 1];
|
||||
}
|
||||
indices[^1] = tmp;
|
||||
cycles[i] = poolLength - i;
|
||||
}
|
||||
else
|
||||
{
|
||||
var j = cycles[i];
|
||||
(indices[i], indices[^j]) = (indices[^j], indices[i]);
|
||||
yield return GetCombination(indices[..count], pool);
|
||||
notDone = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static TValue[] GetCombination<TValue>(int[] innerIndices, TValue[] innerPool) =>
|
||||
innerIndices.Select(i => innerPool[i]).ToArray();
|
||||
}
|
||||
Loading…
Reference in New Issue