Нахождение трех элементов в массиве, сумма которого ближе всего к данному числу

155

Для данного массива целых чисел A 1 , A 2 , ..., A n , включая отрицательные и положительные числа, и еще одно целое число S. Теперь нам нужно найти три различных целых числа в массиве, сумма которых ближе всего к данному целому числу S Если существует более одного решения, любое из них в порядке.

Можно предположить, что все целые числа находятся в диапазоне int32_t, и при вычислении суммы не произойдет арифметического переполнения. S ничего особенного, но случайно выбранный номер.

Есть ли эффективный алгоритм, кроме перебора, чтобы найти три целых числа?

ZelluX
источник
1
Если вы ищете сумму, равную числу (а не ближайшему), это будет проблемой 3SUM .
Бернхард Баркер

Ответы:

186

Есть ли эффективный алгоритм, кроме перебора, чтобы найти три целых числа?

Ага; мы можем решить это за O (n 2 ) раз! Во-первых, учтите, что ваша проблема Pможет быть эквивалентно сформулирована несколько иначе, что устраняет необходимость в «целевом значении»:

Исходная задача P: Дано массив Aиз nцелых чисел и заданного значения S, существует ли 3-кортеж от того, Aчто суммы в S?

модифицированная проблема P': существует ли массив Aиз 3 nцелых чисел от Aэтой суммы до нуля?

Обратите внимание , что вы можете перейти от этой версии проблемы P'с Pвычитанием вашего S / 3 от каждого элемента A, но теперь вам не нужно целевое значение больше.

Ясно, что если бы мы просто протестировали все возможные 3-кортежи, мы бы решили проблему в O (n 3 ) - это базовая линия грубой силы. Можно ли сделать лучше? Что если мы выберем кортежи несколько умнее?

Во-первых, мы потратим некоторое время на сортировку массива, что обойдется нам в первоначальный штраф O (n log n). Теперь мы выполняем этот алгоритм:

for (i in 1..n-2) {
  j = i+1  // Start right after i.
  k = n    // Start at the end of the array.

  while (k >= j) {
    // We got a match! All done.
    if (A[i] + A[j] + A[k] == 0) return (A[i], A[j], A[k])

    // We didn't match. Let's try to get a little closer:
    //   If the sum was too big, decrement k.
    //   If the sum was too small, increment j.
    (A[i] + A[j] + A[k] > 0) ? k-- : j++
  }
  // When the while-loop finishes, j and k have passed each other and there's
  // no more useful combinations that we can try with this i.
}

Этот алгоритм работает путем размещения трех указателей, i, j, и kв различных точках массива. iначинается в начале и медленно продвигается к концу. kуказывает на самый последний элемент. jуказывает на то, где iначалось. Мы итеративно пытаемся суммировать элементы по их соответствующим индексам, и каждый раз происходит одно из следующего:

  • Сумма точно правильная! Мы нашли ответ.
  • Сумма была слишком мала. Подойдите jближе к концу, чтобы выбрать следующий по величине номер.
  • Сумма была слишком большой. Подойдите kближе к началу, чтобы выбрать следующее наименьшее число.

Для каждого iуказатели jи kбудут постепенно приближаться друг к другу. В конце концов они будут проходить друг друга, и в этот момент нам не нужно ничего для этого пробовать i, поскольку мы суммируем одни и те же элементы, просто в другом порядке. После этого мы попробуем следующее iи повторим.

В конце концов, мы либо исчерпаем полезные возможности, либо найдем решение. Вы можете видеть, что это O (n 2 ), поскольку мы выполняем внешний цикл O (n) раз и выполняем внутренний цикл O (n) раз. Это можно сделать субквадратично, если вы действительно придумаете, представив каждое целое число как битовый вектор и выполнив быстрое преобразование Фурье, но это выходит за рамки этого ответа.


Примечание. Поскольку это вопрос интервью, я немного обманываю: этот алгоритм позволяет выбирать один и тот же элемент несколько раз. То есть (-1, -1, 2) будет правильным решением, как (0, 0, 0). Он также находит только точные ответы, а не самый близкий ответ, как упоминается в заголовке. В качестве упражнения для читателя я дам вам понять, как заставить его работать только с отдельными элементами (но это очень простое изменение) и точными ответами (что также является простым изменением).

Джон Феминелла
источник
8
Кажется, что алгоритм может найти только 3-кортеж, равный S, а не ближайший к S.
ZelluX
7
ZelluX: Как я упоминал в заметке, я не хотел отдавать слишком много, потому что это проблема интервью. Надеюсь, вы сможете увидеть, как его изменить, чтобы он получил самый близкий ответ. (Подсказка: один из способов - отследить самый близкий ответ и переписать его, если вы найдете лучший.)
Джон Феминелла
12
Что, если мы не изменим постановку задачи, вместо этого мы будем искать aj и ak, которые суммируют с ai + S.
логическое значение
3
@ZelluX: Это похоже на то, как работает сортировка слиянием (вот как она впервые щелкнула для меня). Этот внутренний цикл пытается доказать, что ни A [j], ни A [k] не могут быть частью какого- либо удовлетворительного решения. Проблема в любой момент такова: «Есть ли пара j '> = j и k' <= k такая, что A [j] + A [k] = S - A [i]?» Глядя на текущую пару (i, j), есть 3 варианта: сумма взрыва (остановка - мы выиграли!), Она слишком низкая или слишком высокая. Если оно слишком мало, то сумма A [j] + A [k '] также должна быть слишком низкой для каждого k' <= k, поскольку в каждой такой сумме первый член (A [j]) будет одинаковым. ..
j_random_hacker
1
... и второе слагаемое (A [k ']) будет таким же или даже ниже, чем A [k]. Таким образом, в этом случае мы доказали, что A [j] не может участвовать ни в одной удовлетворяющей сумме, поэтому мы можем также отказаться от нее! Что мы и делаем, устанавливая j = j + 1 и начиная с начала (хотя это может помочь думать с точки зрения рекурсивного решения меньшей подзадачи). Аналогично, если сумма A [j] + A [k] слишком велика, то мы знаем, что A [j '] + A [k] также должна быть слишком высокой для каждого j'> = j, поскольку A [j '] должно быть не меньше, чем A [j], и мы уже слишком высоки. Это означает, что мы можем безопасно отказаться от A [k], установив k = k-1 и начав сначала.
j_random_hacker
28

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

Другое решение O (n ^ 2) (с использованием хэш-набора).

// K is the sum that we are looking for
for i 1..n
    int s1 = K - A[i]
    for j 1..i
        int s2 = s1 - A[j]
        if (set.contains(s2))
            print the numbers
    set.add(A[i])
Cem
источник
8
Недостатком является хранение O (N) вместо того, чтобы делать это на месте.
Чарльз Мангер
6
Использование хэш-набора не является строгим O (n ^ 2), так как хэш-набор может вырождаться в редких случаях, что приводит к линейному времени поиска.
Ext3h
@Charles - также для решения Джона нужно пространство O (N), так как при сортировке вы меняете исходный массив. Это означает, что вызывающему абоненту может потребоваться защитная копия перед использованием функции.
gamliela
Я думаю, что в вашем алгоритме есть ошибка. s2может быть уже выбранным элементом. Например, если массив есть 0,1,2и Kесть 2, ответа не должно быть. Я думаю, что ваш алгоритм будет выводить, 0,1,1что, очевидно, неверно.
Ямча
7

Решение Джона Феминеллы имеет ошибку.

На линии

if (A[i] + A[j] + A[k] == 0) return (A[i], A[j], A[k])

Нам нужно проверить, все ли i, j, k различны. В противном случае, если мой целевой элемент 6и если мой входной массив содержит {3,2,1,7,9,0,-4,6}. Если я распечатаю кортежи с суммой 6, то я также получу 0,0,6вывод. Чтобы избежать этого, нам нужно изменить условие таким образом.

if ((A[i] + A[j] + A[k] == 0) && (i!=j) && (i!=k) && (j!=k)) return (A[i], A[j], A[k])
Rambo7
источник
2
Решение Джона Феминеллы состоит в том, чтобы просто представить алгоритм решения проблемы, он также указал, что его решение не будет работать для определенного числа, и вам придется немного изменить вышеприведенный код, который он оставил читателю.
EmptyData
3
На самом деле, я никогда не буду j, поскольку вы всегда начинаете его с j = i + 1. Единственное реальное условие, которое вы должны проверить, это j == k. Однако, установив цикл while в j <k, вы решили проблемы без длинного оператора if, поскольку k всегда будет больше j, а j всегда больше i.
lorenzocastillo
2
Это не похоже на ответ на вопрос, а скорее на комментарий к ответу Джона Феминеллы.
Бернхард Баркер
6

Как насчет чего-то вроде этого, который является O (n ^ 2)

for(each ele in the sorted array)
{
    ele = arr[i] - YOUR_NUMBER;
    let front be the pointer to the front of the array;
    let rear be the pointer to the rear element of the array.;

    // till front is not greater than rear.                    
    while(front <= rear)
    {
        if(*front + *rear == ele)
        {
            print "Found triplet "<<*front<<","<<*rear<<","<<ele<<endl;
            break;
        }
        else
        {
            // sum is > ele, so we need to decrease the sum by decrementing rear pointer.
            if((*front + *rear) > ele)
                decrement rear pointer.
            // sum is < ele, so we need to increase the sum by incrementing the front pointer.
            else
                increment front pointer.
        }
    }

Это находит, если сумма из 3 элементов в точности равна вашему числу. Если вы хотите ближайший, вы можете изменить его так, чтобы он запомнил наименьшую дельту (разницу между вашим числом текущего триплета) и в конце напечатал триплет, соответствующий наименьшей дельте.

codaddict
источник
если вы хотите найти k элементов, чтобы получить сумму, в чем сложность? Как вы справляетесь с этим?
coder_15
При таком подходе сложность для k элементов составляет O (n ^ (k-1)) для k> = 2. Вам необходимо добавить внешний цикл для каждого дополнительного слагаемого.
Ext3h
5

Обратите внимание, что у нас есть отсортированный массив. Это решение похоже на решение Джона только в том, что оно ищет сумму и не повторяет один и тот же элемент.

#include <stdio.h>;

int checkForSum (int arr[], int len, int sum) { //arr is sorted
    int i;
    for (i = 0; i < len ; i++) {
        int left = i + 1;
        int right = len - 1;
        while (right > left) {
            printf ("values are %d %d %d\n", arr[i], arr[left], arr[right]);
            if (arr[right] + arr[left] + arr[i] - sum == 0) {
                printf ("final values are %d %d %d\n", arr[i], arr[left], arr[right]);
                return 1;
            }
            if (arr[right] + arr[left] + arr[i] - sum > 0)
                right--;
            else
                left++;
        }
    }
    return -1;
}
int main (int argc, char **argv) {
    int arr[] = {-99, -45, -6, -5, 0, 9, 12, 16, 21, 29};
    int sum = 4;
    printf ("check for sum %d in arr is %d\n", sum, checkForSum(arr, 10, sum));
}
пасада
источник
Необходимо рассчитать абсолютную разницу a[r] + a[l] + a[i] - sum. Примерить arr = [-1, 2, 1, -4] sum = 1.
Димитрий
3

Вот код C ++:

bool FindSumZero(int a[], int n, int& x, int& y, int& z)
{
    if (n < 3)
        return false;

    sort(a, a+n);

    for (int i = 0; i < n-2; ++i)
    {
        int j = i+1;
        int k = n-1;

        while (k >= j)
        {
            int s = a[i]+a[j]+a[k];

            if (s == 0 && i != j && j != k && k != i)
            {
                x = a[i], y = a[j], z = a[k];
                return true;
            }

            if (s > 0)
                --k;
            else
                ++j;
        }
    }

    return false;
}
Питер Ли
источник
2

Очень простое решение N ^ 2 * logN: отсортируйте входной массив, затем просмотрите все пары A i , A j (время N ^ 2), и для каждой пары проверьте, находится ли (S - A i - A j ) в массиве ( время входа).

Другое решение O (S * N) использует классический подход динамического программирования .

Коротко:

Создайте двумерный массив V [4] [S + 1]. Заполните его так, чтобы:

V [0] [0] = 1, V [0] [x] = 0;

V 1 [A i ] = 1 для любого i, V 1 [x] = 0 для всех остальных x

V [2] [A i + A j ] = 1, для любого i, j. V [2] [x] = 0 для всех остальных x

V [3] [сумма любых 3 элементов] = 1.

Чтобы заполнить его, переберите A i , для каждого перебирайте массив A i справа налево.

Olexiy
источник
небольшое изменение в первом алгоритме .. если элемент не существует, то в конце двоичного поиска нам нужно будет посмотреть на элемент слева, текущий и правый, чтобы увидеть, какой из них дает наиболее близкий результат ,
Анураг
Массив слишком большой, и это не O (s * N). Этот шаг O (N ^ 2): V [2] [Ai + Aj] = 1, для любого i, j. V [2] [x] = 0 для всех остальных x.
Ричард
1

Это может быть эффективно решено в O (n log (n)) следующим образом. Я даю решение, которое сообщает, равна ли сумма любых трех чисел данному числу.

import java.util.*;
public class MainClass {
        public static void main(String[] args) {
        int[] a = {-1, 0, 1, 2, 3, 5, -4, 6};
        System.out.println(((Object) isThreeSumEqualsTarget(a, 11)).toString());
}

public static boolean isThreeSumEqualsTarget(int[] array, int targetNumber) {

    //O(n log (n))
    Arrays.sort(array);
    System.out.println(Arrays.toString(array));

    int leftIndex = 0;
    int rightIndex = array.length - 1;

    //O(n)
    while (leftIndex + 1 < rightIndex - 1) {
        //take sum of two corners
        int sum = array[leftIndex] + array[rightIndex];
        //find if the number matches exactly. Or get the closest match.
        //here i am not storing closest matches. You can do it for yourself.
        //O(log (n)) complexity
        int binarySearchClosestIndex = binarySearch(leftIndex + 1, rightIndex - 1, targetNumber - sum, array);
        //if exact match is found, we already got the answer
        if (-1 == binarySearchClosestIndex) {
            System.out.println(("combo is " + array[leftIndex] + ", " + array[rightIndex] + ", " + (targetNumber - sum)));
            return true;
        }
        //if exact match is not found, we have to decide which pointer, left or right to move inwards
        //we are here means , either we are on left end or on right end
        else {

            //we ended up searching towards start of array,i.e. we need a lesser sum , lets move inwards from right
            //we need to have a lower sum, lets decrease right index
            if (binarySearchClosestIndex == leftIndex + 1) {
                rightIndex--;
            } else if (binarySearchClosestIndex == rightIndex - 1) {
                //we need to have a higher sum, lets decrease right index
                leftIndex++;
            }
        }
    }
    return false;
}

public static int binarySearch(int start, int end, int elem, int[] array) {
    int mid = 0;
    while (start <= end) {
        mid = (start + end) >>> 1;
        if (elem < array[mid]) {
            end = mid - 1;
        } else if (elem > array[mid]) {
            start = mid + 1;
        } else {
            //exact match case
            //Suits more for this particular case to return -1
            return -1;
        }
    }
    return mid;
}
}
Раджу Сингх
источник
Я не думаю, что это сработает. Конечно, у вас есть два простых случая, как продвигаться leftIndexили rightIndexкогда все элементы в середине либо строго меньше, либо больше, чем желаемое число. Но как насчет случая, когда бинарный поиск остановился где-то посередине? Вам нужно будет проверить обе ветви (где rightIndex--и leftIndex++). В своем решении вы просто игнорируете эту ситуацию. Но я не думаю, что есть способ преодолеть эту проблему.
Aivean
0

Сокращение: я думаю, что решение @John Feminella O (n2) является наиболее элегантным. Мы все еще можем уменьшить A [n] для поиска кортежа. Наблюдая A [k] так, чтобы все элементы находились в A [0] - A [k], когда наш массив поиска огромен, а SUM действительно малы.

[0] является минимальным: - отсортированный по возрастанию массив.

s = 2A [0] + A [k]: Учитывая s и A [], мы можем найти A [k], используя бинарный поиск по времени log (n).

d1val
источник
0

Вот программа в Java, которая является O (N ^ 2)

import java.util.Stack;


public class GetTripletPair {

    /** Set a value for target sum */
    public static final int TARGET_SUM = 32;

    private Stack<Integer> stack = new Stack<Integer>();

    /** Store the sum of current elements stored in stack */
    private int sumInStack = 0;
    private int count =0 ;


    public void populateSubset(int[] data, int fromIndex, int endIndex) {

        /*
        * Check if sum of elements stored in Stack is equal to the expected
        * target sum.
        * 
        * If so, call print method to print the candidate satisfied result.
        */
        if (sumInStack == TARGET_SUM) {
            print(stack);
        }

        for (int currentIndex = fromIndex; currentIndex < endIndex; currentIndex++) {

            if (sumInStack + data[currentIndex] <= TARGET_SUM) {
                ++count;
                stack.push(data[currentIndex]);
                sumInStack += data[currentIndex];

                /*
                * Make the currentIndex +1, and then use recursion to proceed
                * further.
                */
                populateSubset(data, currentIndex + 1, endIndex);
                --count;
                sumInStack -= (Integer) stack.pop();
            }else{
            return;
        }
        }
    }

    /**
    * Print satisfied result. i.e. 15 = 4+6+5
    */

    private void print(Stack<Integer> stack) {
        StringBuilder sb = new StringBuilder();
        sb.append(TARGET_SUM).append(" = ");
        for (Integer i : stack) {
            sb.append(i).append("+");
        }
        System.out.println(sb.deleteCharAt(sb.length() - 1).toString());
    }

    private static final int[] DATA = {4,13,14,15,17};

    public static void main(String[] args) {
        GetAllSubsetByStack get = new GetAllSubsetByStack();
        get.populateSubset(DATA, 0, DATA.length);
    }
}
М Сач
источник
Хороший подход, но я не мог получить точку, где вы ограничиваете количество результатов до триплета. Например, рассмотрите входные данные: [1,11,3,4,5,6,7,8, 2] и сумму 12, из вашего решения видно, что [1, 11] [4,8] [1,4, 5,2] и т. Д. Будет работать.
Анупам Сайни
0

Задача может быть решена в O (n ^ 2) путем расширения задачи 2-суммы с незначительными изменениями. A - вектор, содержащий элементы, а B - требуемая сумма.

int Solution :: threeSumClosest (vector & A, int B) {

sort(A.begin(),A.end());

int k=0,i,j,closest,val;int diff=INT_MAX;

while(k<A.size()-2)
{
    i=k+1;
    j=A.size()-1;

    while(i<j)
    {
        val=A[i]+A[j]+A[k];
        if(val==B) return B;
        if(abs(B-val)<diff)
        {
            diff=abs(B-val);
            closest=val;
        }
        if(B>val)
        ++i;
        if(B<val) 
        --j;
    }
    ++k;

}
return closest;
Анураг Семвал
источник
0

Вот код Python3

class Solution:
    def threeSumClosest(self, nums: List[int], target: int) -> int:
        result = set()
        nums.sort()
        L = len(nums)     
        for i in range(L):
            if i > 0 and nums[i] == nums[i-1]:
                continue
            for j in range(i+1,L):
                if j > i + 1 and nums[j] == nums[j-1]:
                    continue  
                l = j+1
                r = L -1
                while l <= r:
                    sum = nums[i] + nums[j] + nums[l]
                    result.add(sum)
                    l = l + 1
                    while l<=r and nums[l] == nums[l-1]:
                        l = l + 1
        result = list(result)
        min = result[0]
        for i in range(1,len(result)):
            if abs(target - result[i]) < abs(target - min):
                min = result[i]
        return min
Uday Reddy
источник
-1

Другое решение, которое проверяет и дает сбой рано

public boolean solution(int[] input) {
        int length = input.length;

        if (length < 3) {
            return false;
        }

        // x + y + z = 0  => -z = x + y
        final Set<Integer> z = new HashSet<>(length);
        int zeroCounter = 0, sum; // if they're more than 3 zeros we're done

        for (int element : input) {
            if (element < 0) {
                z.add(element);
            }

            if (element == 0) {
                ++zeroCounter;
                if (zeroCounter >= 3) {
                    return true;
                }
            }
        }

        if (z.isEmpty() || z.size() == length || (z.size() + zeroCounter == length)) {
            return false;
        } else {
            for (int x = 0; x < length; ++x) {
                for (int y = x + 1; y < length; ++y) {
                    sum = input[x] + input[y]; // will use it as inverse addition
                    if (sum < 0) {
                        continue;
                    }
                    if (z.contains(sum * -1)) {
                        return true;
                    }
                }
            }
        }
        return false;
    }

Я добавил несколько модульных тестов здесь: GivenArrayReturnTrueIfThreeElementsSumZeroTest .

Если набор использует слишком много места, я могу легко использовать java.util.BitSet, который будет использовать пространство O (n / w) .

Moxi
источник
-1

Программа, чтобы получить эти три элемента. Я только что отсортировал массив / список в первую очередь, и они обновляются minClosenessна основе каждого триплета.

public int[] threeSumClosest(ArrayList<Integer> A, int B) {
    Collections.sort(A);
    int ansSum = 0;
    int ans[] = new int[3];
    int minCloseness = Integer.MAX_VALUE;
    for (int i = 0; i < A.size()-2; i++){
        int j = i+1;
        int k = A.size()-1;
        while (j < k){
            int sum = A.get(i) + A.get(j) + A.get(k);
            if (sum < B){
                j++;
            }else{
                k--;
            }
            if (minCloseness >  Math.abs(sum - B)){
                minCloseness = Math.abs(sum - B);
                ans[0] = A.get(i); ans[1] = A.get(j); ans[2] = A.get(k);
            }
        }
    }
    return ans;
}
Саурав Саху
источник
-2

Я сделал это в п ^ 3, мой псевдокод ниже;

// Создаем hashMap с ключом как Integer и значением как ArrayList // перебираем список с помощью цикла for, для каждого значения в списке перебираем снова, начиная со следующего значения;

for (int i=0; i<=arr.length-1 ; i++){
    for (int j=i+1; j<=arr.length-1;j++){

// если сумма arr [i] и arr [j] меньше требуемой суммы, то существует вероятность найти третью цифру, поэтому сделайте другую для цикла

      if (arr[i]+arr[j] < sum){
        for (int k= j+1; k<=arr.length-1;k++)

// в этом случае мы сейчас ищем третье значение; если сумма arr [i] и arr [j] и arr [k] является желаемой суммой, то добавьте их в HashMap, сделав arr [i] ключом, а затем добавив arr [j] и arr [k] в ArrayList в значении этого ключа

          if (arr[i]+arr[j]+arr[k] ==  sum){              
              map.put(arr[i],new ArrayList<Integer>());
              map.get(arr[i]).add(arr[j]);
              map.get(arr[i]).add(arr[k]);}

после этого у вас теперь есть словарь, в котором есть все записи, представляющие три значения, добавляя к желаемой сумме. Извлеките все эти записи, используя функции HashMap. Это сработало отлично.

ниже нуля
источник