Поиск всех комбинаций чисел из набора чисел, которые равны цели, код пропускает некоторые, но не все - PullRequest
1 голос
/ 08 марта 2019

Я использовал этот вопрос Выясните, какие комбинации чисел в наборе суммируются до заданной суммы , которая использует пример C #, чтобы попытаться найти все возможные комбинации чисел, которые суммируют с данным числом .

Я использовал предоставленный код и попытался его понять, и думаю, что понимаю, однако не могу понять, почему возникает эта ошибка.

Вот настройка:

public class Program
    {
        public static void Main(string[] args)
        {
            // subtotal list
            List<double> totals = new List<double>(new double[] { 17.5, 14.3, 10.9, 7.8, 6.3, 3.8, 3.2, 2.7, 1.8, 1.0 });
            List<double> totals2 = new List<double>(new double[] { 17.5, 14.3, 7.8, 6.3, 3.2, 1.8});

        // get matches
        List<double[]> results = Knapsack.MatchTotal(50.9, totals2);

        // print results`
        foreach (var result in results)
        {
            Console.WriteLine(string.Join(",", result));
        }

        Console.WriteLine("Done.");
        Console.ReadKey();
    }
}

и основная часть кода не изменилась по сравнению с примером.

public class Knapsack
    {
        internal static List<double[]> MatchTotal(double theTotal, List<double> subTotals)
        {
            List<double[]> results = new List<double[]>();

        while (subTotals.Contains(theTotal))
        {
            results.Add(new double[1] { theTotal });
            subTotals.Remove(theTotal);
        }

        // if no subtotals were passed
        // or all matched the Total
        // return
        if (subTotals.Count == 0)
            return results;

        subTotals.Sort();

        double mostNegativeNumber = subTotals[0];
        if (mostNegativeNumber > 0)
            mostNegativeNumber = 0;

        // if there aren't any negative values
        // we can remove any values bigger than the total
        if (mostNegativeNumber == 0)
            subTotals.RemoveAll(d => d > theTotal);

        // if there aren't any negative values
        // and sum is less than the total no need to look further
        if (mostNegativeNumber == 0 && subTotals.Sum() < theTotal)
            return results;

        // get the combinations for the remaining subTotals
        // skip 1 since we already removed subTotals that match
        for (int choose = 2; choose <= subTotals.Count; choose++)
        {
            // get combinations for each length
            IEnumerable<IEnumerable<double>> combos = Combination.Combinations(subTotals.AsEnumerable(), choose);

            // add combinations where the sum mathces the total to the result list
            results.AddRange(from combo in combos
                             where combo.Sum() == theTotal
                             select combo.ToArray());
        }

        return results;
    }
}

public static class Combination
{
    public static IEnumerable<IEnumerable<T>> Combinations<T>(this IEnumerable<T> elements, int choose)
    {
        return choose == 0 ?                        // if choose = 0
            new[] { new T[0] } :                    // return empty Type array
            elements.SelectMany((element, i) =>     // else recursively iterate over array to create combinations
            elements.Skip(i + 1).Combinations(choose - 1).Select(combo => (new[] { element }).Concat(combo)));
    }
}

Я предоставил набор чисел, который содержится в totals и используется для расчета суммы.

Я ввел 50,9 как итог и не дал результатов.

Второй набор чисел totals2 - это набор чисел, которые составляют до 50,9 и являются подмножеством totals. Это также не может вернуть результат за 50,9 По какой-то причине код не может найти 50,9 в основном наборе или в подмножестве (которое само по себе составляет 50,9).

Я не могу понять, почему это происходит, кто-нибудь может мне сказать?

...