Minimal steps down to 1: Difference between revisions

Content added Content deleted
(Rename Perl 6 -> Raku, alphabetize, minor clean-up)
(Added C#)
Line 47: Line 47:
;Reference:
;Reference:
* [https://www.youtube.com/watch?v=f2xi3c1S95M Learn Dynamic Programming (Memoization & Tabulation)] Video of similar task.
* [https://www.youtube.com/watch?v=f2xi3c1S95M Learn Dynamic Programming (Memoization & Tabulation)] Video of similar task.

=={{header|C sharp}}==
{{works with|C sharp|7}}
<lang csharp>using System;
using System.Collections.Generic;
using System.Linq;

public static class MinimalSteps
{
public static void Main() {
var (divisors, subtractors) = (new int[] { 2, 3 }, new [] { 1 });
var lookup = CreateLookup(2_000, divisors, subtractors);
Console.WriteLine($"Divisors: [{divisors.Delimit()}], Subtractors: [{subtractors.Delimit()}]");
PrintRange(lookup, 10);
PrintMaxMins(lookup);
lookup = CreateLookup(20_000, divisors, subtractors);
PrintMaxMins(lookup);
Console.WriteLine();

subtractors = new [] { 2 };
lookup = CreateLookup(2_000, divisors, subtractors);
Console.WriteLine($"Divisors: [{divisors.Delimit()}], Subtractors: [{subtractors.Delimit()}]");
PrintRange(lookup, 10);
PrintMaxMins(lookup);
lookup = CreateLookup(20_000, divisors, subtractors);
PrintMaxMins(lookup);
}

private static void PrintRange((char op, int param, int steps)[] lookup, int limit) {
for (int goal = 1; goal <= limit; goal++) {
var x = lookup[goal];
if (x.param == 0) {
Console.WriteLine($"{goal} cannot be reached with these numbers.");
continue;
}
Console.Write($"{goal} takes {x.steps} {(x.steps == 1 ? "step" : "steps")}: ");
for (int n = goal; n > 1; ) {
Console.Write($"{n},{x.op}{x.param}=> ");
n = x.op == '/' ? n / x.param : n - x.param;
x = lookup[n];
}
Console.WriteLine("1");
}
}

private static void PrintMaxMins((char op, int param, int steps)[] lookup) {
var maxSteps = lookup.Max(x => x.steps);
var items = lookup.Select((x, i) => (i, x)).Where(t => t.x.steps == maxSteps).ToList();
Console.WriteLine(items.Count == 1
? $"There is one number below {lookup.Length-1} that requires {maxSteps} steps: {items[0].i}"
: $"There are {items.Count} numbers below {lookup.Length-1} that require {maxSteps} steps: {items.Select(t => t.i).Delimit()}"
);
}

private static (char op, int param, int steps)[] CreateLookup(int goal, int[] divisors, int[] subtractors)
{
var lookup = new (char op, int param, int steps)[goal+1];
lookup[1] = ('/', 1, 0);
for (int n = 1; n < lookup.Length; n++) {
var ln = lookup[n];
if (ln.param == 0) continue;
for (int d = 0; d < divisors.Length; d++) {
int target = n * divisors[d];
if (target > goal) break;
if (lookup[target].steps == 0 || lookup[target].steps > ln.steps) lookup[target] = ('/', divisors[d], ln.steps + 1);
}
for (int s = 0; s < subtractors.Length; s++) {
int target = n + subtractors[s];
if (target > goal) break;
if (lookup[target].steps == 0 || lookup[target].steps > ln.steps) lookup[target] = ('-', subtractors[s], ln.steps + 1);
}
}
return lookup;
}

private static string Delimit<T>(this IEnumerable<T> source) => string.Join(", ", source);
}</lang>
{{out}}
<pre>
Divisors: [2, 3], Subtractors: [1]
1 takes 0 steps: 1
2 takes 1 step: 2,-1=> 1
3 takes 1 step: 3,/3=> 1
4 takes 2 steps: 4,-1=> 3,/3=> 1
5 takes 3 steps: 5,-1=> 4,-1=> 3,/3=> 1
6 takes 2 steps: 6,/2=> 3,/3=> 1
7 takes 3 steps: 7,-1=> 6,/2=> 3,/3=> 1
8 takes 3 steps: 8,/2=> 4,-1=> 3,/3=> 1
9 takes 2 steps: 9,/3=> 3,/3=> 1
10 takes 3 steps: 10,-1=> 9,/3=> 3,/3=> 1
There are 16 numbers below 2000 that require 14 steps: 863, 1079, 1295, 1439, 1511, 1583, 1607, 1619, 1691, 1727, 1823, 1871, 1895, 1907, 1919, 1943
There are 5 numbers below 20000 that require 20 steps: 12959, 15551, 17279, 18143, 19439

Divisors: [2, 3], Subtractors: [2]
1 takes 0 steps: 1
2 takes 1 step: 2,/2=> 1
3 takes 1 step: 3,-2=> 1
4 takes 2 steps: 4,-2=> 2,/2=> 1
5 takes 2 steps: 5,-2=> 3,-2=> 1
6 takes 2 steps: 6,/2=> 3,-2=> 1
7 takes 3 steps: 7,-2=> 5,-2=> 3,-2=> 1
8 takes 3 steps: 8,-2=> 6,/2=> 3,-2=> 1
9 takes 2 steps: 9,/3=> 3,-2=> 1
10 takes 3 steps: 10,/2=> 5,-2=> 3,-2=> 1
There is one number below 2000 that requires 17 steps: 1699
There is one number below 20000 that requires 24 steps: 19681</pre>


=={{header|Go}}==
=={{header|Go}}==