Поиск недостающих элементов в массиве - PullRequest
11 голосов
/ 09 марта 2011

Учитывая, что у вас есть массив A [1..n] размера n, он содержит элементы из набора {1..n}. Однако два элемента отсутствуют (и, возможно, два элемента массива повторяются). Найдите недостающие элементы.

Например, если n = 5, A может быть A [5] = {1,2,1,3,2}; и поэтому недостающие элементы {4,5}

Подход, который я использовал, был:

int flag[n] = {0};  
int i;  
for(i = 0; i < n; i++)  {  
  flag[A[i]-1] = 1;  
 }  

for(i = 0; i < n; i++)  {  
 if(!flag[i]) {  
    printf("missing: %d", (i+1));  
}  

сложность пространства сводится к O (n). Я чувствую, что это очень детский и неэффективный код. Поэтому не могли бы вы предоставить лучший алгоритм с большей сложностью пространства и времени.

Ответы [ 9 ]

15 голосов
/ 09 марта 2011

Теоретически,

Это можно сделать в O (1) пространстве (в модели RAM, т.е. O (1) слов) и O (n) времени даже с массивом только для чтения.

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


Алгоритм

Предположим, что отсутствующие числа равны x иy.

Для массива есть две возможности:

1) Одно число повторяется три раза, а остальные числа в массиве появляются ровно один раз.

В этом случае сработает трюк с XOR с пакетом.

Выполните XOR для всех элементов массива с 1,2, ..., n.

В итоге вы получитес z = x XOR y.

Существует по крайней мере один бит z, который не равен нулю.

Теперь дифференцируем элементы массива на основе этого бита (два сегмента):XOR снова проходит через массив.

В результате вы получите x и y.

Получив x и y, вы можете подтвердить, действительно ли это отсутствующие элементы.

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

2) Два элементаs повторяется ровно дважды, а остальные появляются ровно один раз.

Пусть два повторяющихся элемента - это a и b (x и y - отсутствующие).

Предупреждение:Математика впереди.

Пусть S_k = 1^k + 2^k + .. + n^k

Например S_1 = n(n+1)/2, S_2 = n(n+1)(2n+1)/6 и т. Д.

Теперь мы вычисляем семь вещей:

T_1 = Sum of the elements of the array = S_1 + a + b - x - y.
T_2 = Sum of the squares = S_2 + a^2 + b^2 - x^2 - y^2
T_3 = Sum of cubes = S_3 + a^3 + b^3 - x^3 - y^3
T_4 = Sum of fourth powers = S_4 + a^4 + b^4 - x^4 - y^4
...
T_7 = Sum of seventh powers = S_7 + a^7 + b^7 - x^7 - y^7

Обратите внимание, что мы можем использовать O (1) слова (вместо одного) для решения проблем переполнения.(Я предполагаю, что 8-10 слов будет достаточно.)

Пусть Ci = T_i - S_i

Теперь предположим, что a, b, x, y - корни полинома 4-й степени P(z) = z^4 + pz^3 + qz^2 + rz + s

Теперь мы попытаемся преобразовать вышеупомянутые семь уравнений в четыре линейных уравнения в p,q,r,s.

Например, если мы сделаем 4th Eqn + p * 3rd Eqn + q* 2nd equation + r* 1st equation

, мы получим

C4 + p*C3 + q*C2 + r*C1 = 0

Аналогично получаем

C5 + p*C4 + q*C3 + r*C2 + s*C1 = 0
C6 + p*C5 + q*C4 + r*C3 + s*C2 = 0
C7 + p*C6 + q*C5 + r*C4 + s*C3 = 0

Это четыре линейных уравнения в p,q,r,s, которые могут быть решены с помощью методов линейной алгебры, таких как исключение Гаусса.

Обратите внимание, что p,q,r,s будет рациональным и поэтому может быть вычислено только с помощью целочисленной арифметики.

Теперь предположим, что нам дано решение p,q,r,s для вышеуказанной системы уравнений.

Рассмотрим P(z) = z^4 + pz^3 + qz^2 + rz + s.

То, что говорят приведенные выше уравнения, в основном

P(a) + P(b) - P(x) - P(y) = 0
aP(a) + bP(b) - xP(x) -yP(y) = 0
a^2 P(a) + b^2 P(b) - x^2 P(x) - y^2 P(y)  = 0
a^3 P(a) + b^3 P(b) - x^3 P(x) - y^3 P(y) = 0

Теперь матрица

   1   1  -1 -1
   a   b   -x   -y
   a^2 b^2 -x^2 -y^2
   a^3 b^3 -x^3 -y^3

имеет тот же определитель, что и матрица Вандермонда и, следовательно, является обратимым, если a,b,x,y различны.

Таким образом, мы должны иметь это P(a) = P(b) = P(x) = P(y) = 0.

Теперь проверьтекакие из 1,2,3,...,n являются корнями x^4 + px^3 + qx^2 + rx + s = 0.

Таким образом, это линейный алгоритм с постоянной постоянной времени.


Код

Я написал следующий код на C # (.Net 4.0), и он, кажется, работает для нескольких проб, которые я пробовал ... (Примечание: я не удосужился удовлетворить приведенный выше случай 1).

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

using System.Numerics;

namespace SOManaged
{
    class Program
    {
        static void Main(string[] args)
        {
            ulong[] inp = {1,3,2,1,2};
            ulong[] inp1 = { 1,2,3,4,5,6,7,8,
                             9,10,11,13,14,15,
                             16,17,18,19,20,21,5,14};

            int N = 100000;
            ulong[] inp2 = new ulong[N];
            for (ulong i = 0; i < (ulong)N; i++)
            {
                inp2[i] = i+1;
            }
            inp2[122] = 44;
            inp2[419] = 13;

            FindMissingAndRepeated(inp);
            FindMissingAndRepeated(inp1);
            FindMissingAndRepeated(inp2);
        }

        static void FindMissingAndRepeated(ulong [] nums)
        {
            BigInteger[] C = new BigInteger[8];

            // Compute the C_i
            for (int k = 0; k < 8; k++)
            {
                C[k] = 0;
            }

            BigInteger i = 1;
            BigInteger n = 0;

            for (int j = 0; j < nums.Length; j++)
            {
                n = nums[j];
                i = j + 1;
                for (int k = 1; k < 8; k++)
                {
                    C[k] += i - n;
                    n = n * nums[j];
                    i = i * (j + 1);
                }
            }


            for (int k = 1; k <= 7; k++)
            {
                Console.Write("C[" + k.ToString() + "] = " + 
                               C[k].ToString() +", ");
            }
            Console.WriteLine();

            // Solve for p,q,r,s
            BigInteger[] pqrs = new BigInteger[4];
            BigInteger[] constants = new BigInteger[4];
            BigInteger[,] matrix = new BigInteger[4, 4];

            int start = 4;
            for (int row = 0; row < 4; row++ )
            {
                constants[row] = -C[start];

                int k = start-1;
                for (int col = 0; col < 4; col++)
                {
                    matrix[row, col] = C[k];
                    k--;
                }

                start++;
            }

            Solve(pqrs, matrix, constants, 4);

            for (int k = 0; k < 4; k++)
            {
                Console.Write("pqrs[" + k.ToString() + "] = " 
                               + pqrs[k].ToString() + ", ");
            }
            Console.WriteLine();

            // Find the roots.
            for (int k = 1; k <= nums.Length; k++)
            {
                BigInteger x = new BigInteger(k);
                BigInteger p_k = x * x * x* x + pqrs[0] * x* x * x 
                                 + pqrs[1] * x * x + pqrs[2] * x 
                                 + pqrs[3];

                if (p_k == 0)
                {
                    Console.WriteLine("Found: " + k.ToString());
                }
            }
        }

        // Solve using Cramer's method.
        // matrix * pqrs = constants.
        static void Solve(BigInteger[] pqrs, BigInteger[,] matrix, 
                          BigInteger[] constants, int n)
        {
            BigInteger determinant = Determinant(matrix, n);

            for (int i = 0; i < n; i++)
            {
                BigInteger[,] numerator = Replace(matrix, constants, n, i);
                BigInteger numDet = Determinant(numerator,4);
                pqrs[i] = numDet/ determinant;
            }
        }

        // Replace a column of matrix with constants.
        static BigInteger[,] Replace(BigInteger[,] matrix, 
                           BigInteger[] constants, int n, int col)
        {
            BigInteger[,] newMatrix = new BigInteger[n, n];
            for (int i = 0; i < n; i++)
            {
                for (int j = 0; j < n; j++)
                {
                    if (j != col)
                    {
                        newMatrix[i, j] = matrix[i, j];
                    }
                    else
                    {
                        newMatrix[i, j] = constants[i];
                    }
                }
            }

            return newMatrix;
        }

        // Recursively compute determinant for matrix.
        static BigInteger Determinant(BigInteger[,] matrix, int n)
        {
            BigInteger determinant = new BigInteger(0);
            int multiplier = 1;

            if (n == 1)
            {
                return matrix[0,0];
            }

            for (int i = 0; i < n; i++)
            {
                BigInteger [,] subMatrix = new BigInteger[n-1,n-1];
                int row = 0;
                for (int j=1; j < n; j++)
                {
                    int col = 0;
                    for (int k = 0; k < n ; k++)
                    {
                        if (k == i)
                        {
                            continue;
                        }
                        subMatrix[row,col] = matrix[j,k];
                        col++;
                    }
                    row++;
                }

                BigInteger subDeterminant = Determinant(subMatrix, n - 1);
                determinant += multiplier * subDeterminant * matrix[0,i];
                multiplier = -multiplier;
            }

            return determinant;
        }
    }
}

вывод

C[1] = 6, C[2] = 36, C[3] = 180, C[4] = 864, C[5] = 4116, C[6] = 19656, C[7] = 9
4380,
pqrs[0] = -12, pqrs[1] = 49, pqrs[2] = -78, pqrs[3] = 40,
Found: 1
Found: 2
Found: 4
Found: 5


C[1] = 15, C[2] = 407, C[3] = 9507, C[4] = 215951, C[5] = 4861515, C[6] = 108820
727, C[7] = 2424698067,
pqrs[0] = -53, pqrs[1] = 980, pqrs[2] = -7396, pqrs[3] = 18480,
Found: 5
Found: 12
Found: 14
Found: 22


C[1] = 486, C[2] = 189424, C[3] = 75861486, C[4] = 31342069984, C[5] = 130971109
69326, C[6] = 5492487308851024, C[7] = 2305818940736419566,
pqrs[0] = -600, pqrs[1] = 83183, pqrs[2] = -3255216, pqrs[3] = 29549520,
Found: 13
Found: 44
Found: 123
Found: 420
7 голосов
/ 22 апреля 2011

Как указал @j_random_hacker, это очень похоже на Поиск дубликатов в O (n) времени и O (1) пространстве , и адаптация моего ответа там работает здесь тоже. В псевдокоде:

for i := 1 to n
    while A[A[i]] != A[i] 
        swap(A[i], A[A[i]])
    end if
end for

for i := 1 to n
    if A[i] != i then 
        print i
    end if
end for

Первый цикл переставляет массив так, что если элемент x присутствует хотя бы один раз, то одна из этих записей будет в позиции A[x].

Обратите внимание, что, хотя у него есть вложенный цикл, он все еще выполняется за O(N) время - своп происходит только при наличии i такого, что A[i] != i, и каждый своп устанавливает хотя бы один элемент, такой что A[i] == i, где это не было правдой раньше. Это означает, что общее количество перестановок (и, следовательно, общее количество выполнений тела цикла while) не более N-1.

4 голосов
/ 09 марта 2011

Ваше решение не плохое.Вот альтернатива - сортируйте список и перебирайте его, проверяя соседние числа.Если есть пробел, выведите все числа между ними.Если k - длина массива, а n - число, на которое нужно сосчитать, мы получаем O (k lg k + n) времени, O (1) пробел

1 голос
/ 02 февраля 2016

Пример кода для поиска отсутствующих элементов без сортировки массива ниже:

     public static void series(int[] arr) {
     for (int i = 0; i < arr.length; i++) {
        while (arr[i] != i + 1) {
            int jump = arr[arr[i] - 1];
            if (jump == arr[i]) {
                break;
            }
            arr[arr[i] - 1] = arr[i];
            arr[i] = jump;
        }
     }
     System.out.println("Missing number is ");
     for (int i = 0; i < arr.length; i++) {
        if (arr[i] != i + 1) {
            System.out.println(i + 1);
        } else {
            arr[i] = -1;
        }
     }

Этот код работает для серии чисел от 0 до N.

1 голос
/ 10 марта 2011

Ниже приведен один из подходов для идентификации всех пропущенных чисел, когда известно, что массив содержит только числа от 1 до n включительно, без использования дополнительного пробела. Временная сложность O (n).

Давайте возьмем наименьшее число k, так что оно не должно быть в массиве, поэтому k = n + 1 (давайте назовем его коэффициентом сложения).

первый цикл по каждому массиву, и для каждого a [i] мы будем обновлять a [a [i] - 1] + = k; после этого цикла каждый элемент массива содержит два набора информации, число которых изначально было в элементе массива + k * (число вхождений i-го числа в массиве).

во втором цикле вы можете узнать, сколько повторений i-го числа выполнив целочисленное деление числа в каждом месте на k. И исходное число в i-м месте будет [i]% k;

Давайте рассмотрим пример

A[5] = {1,2,1,3,2};

здесь (addfactor) k = 5 (длина массива) + 1 = 6

После того, как массив цикла fisrt будет выглядеть, как если бы исходный элемент был m, а вхождения i-го числа были O(i), результирующий элемент массива был бы m + k * O(i) делением этого элемента (целым числом) на k, вы получите вхождения i-го элемента и% k вы получите исходный массив.

A = {1 + 6*2, 2 + 6*2, 1 + 6*1, 3+6*0 , 2+6*0 }
A = {13, 14, 7, 3, 2 }

Ниже приведен код C #, который (извините, мой C немного заржавел, он давно уже) может быть перенесен на любой язык, просто заменив Printf & scanfs.

    static void Main(string[] args)
    {
        int[] A = { 1, 2, 1, 3, 2 };
        PrintDuplicateAndMissing(A);
        Console.ReadLine();
    }

    static void PrintDuplicateAndMissing(int[] array)
    {
        int addfactor = array.Length + 1;
        for (int i = 0; i < array.Length; i++)
        {
            array[array[i] - 1] += addfactor; // -1 only if array contains from 1 to n. if it is 0 to n (this -1 is not required)
        }
        for (int i = 0; i < array.Length; i++)
        {
            if ( (array[i] / addfactor) == 0 )
                Console.WriteLine(string.Format("{0} is missing", i + 1)); // i + 1 only if array is 1 to n, if 0 to n then +1 is not required
            array[i] %= addfactor; //restore original content of the array
        }
    }
1 голос
/ 09 марта 2011

Это немного qwirky Поскольку все ваши цифры положительны (по проблеме). Я сделаю число в позиции i-1 отрицательным, если я присутствую в массиве.

int i;  
for(i = 0; i < n; i++)  {  
    A[abs(A[i])-1] = -1*abs(A[abs(A[i])-1]);
 }  

for(i = 0; i < n; i++)  {  
 if(A[i]>0) {  
    printf("missing: %d", i+1);  
}  

Сложность O (n), пользователь вспомогательного массива отсутствует, но уничтожает входной массив.

0 голосов
/ 23 ноября 2013

As you have given an array of n size and find the missing number when it's in a sequence.

#include<stdio.h>
main()
{
print("value of n");
scan("%d",&n);
print("enter the elements");
for(i=0;i<n;i++)
scan("%d",&a[i]);
for(i=0;i<n;i++)
{
d1[i]=a[i+1]-a[i];
temp=d1[i];
d[i]=temp;
}
for(i=0;i<n;i++)
{
if(d[i]==d[i+1]
{
c=d[i];
break;
}
}
for(i=0;i<n;i++)
b[i]=a[0]+i*c;
for(i=0;i<n;i++)
{
miss=0;
for(j=0;j<n;j++)
{
if(b[i]!=a[j])
{
miss++;
}
if(miss==n)
print("missing no. %d",b[i]);
}
}

It would find the missing when its in sequence only.

0 голосов
/ 12 марта 2012

Как мы знаем, мы ищем элементы от 1 до N. Создайте хэш-набор, содержащий от 1 до N.

foreach(int i in input)
{
   if(hashset.contains(i))
   {
      hashset.delete(i);
   }
}

return "remaining elements in Hashset.";

Остальные элементы в Hashset - это недостающие элементы.

0 голосов
/ 09 марта 2011

Цикл каждого элемента 0 ... n-1.

x = abs(A[i]) (with i = 0...n-1);

A[x - 1] can be: 
> 0: we haven't checked the element, change its sign to negative:
    A[x - 1] = -A[x - 1]
< 0: we already found the same number

В конце цикла передайте каждый A [0 ... n-1].Индекс положительных элементов + 1 - это пропущенные числа.

Так что если

y = abs(A[i]) > 0: i + 1 is missing.

В C #

var arr = new[] { 1, 2, 1, 2, 4 };

for (int i = 0; i < arr.Length; i++) {
    int x = Math.Abs(arr[i]);
    int y = arr[x - 1];
    if (y > 0) {
        arr[x - 1] = -arr[x - 1];
    }
}

for (int i = 0; i < arr.Length; i++) {
    int x = arr[i];
    if (x > 0) {
        Console.WriteLine("Missing {0}", i + 1);
    } else {
        arr[i] = -arr[i];
    }
}

И массив как новый

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...