Как найти все возможные подмножества данного массива? - PullRequest
11 голосов
/ 24 марта 2009

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

Я ищу алгоритм. Я понимаю логику здесь, но я не смог реализовать это до сих пор.

Ответы [ 4 ]

12 голосов
/ 25 марта 2009

Учитывая набор S из N элементов и заданное подмножество, каждый элемент либо принадлежит, либо не принадлежит этому подмножеству. Следовательно, 2^N являются возможными подмножествами (если вы включаете исходные и пустые наборы), и существует прямое сопоставление битов в двоичном представлении x между 0 и 2^N с элементами в x th подмножество S.

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

Для нахождения подмножеств, равных суммарному t для больших N, одной из оптимизаций может быть запись тех подмножеств, которые превышают t, и не тестирование любых, которые являются правильными надмножествами этих подмножеств. Проверка того, является ли номер набора x надмножеством набора y, может быть выполнена с помощью одной побитовой операции и целочисленного сравнения.

7 голосов
/ 25 марта 2009

То, что вы ищете, называется блок питания . Rosetta Code имеет множество различных реализаций, но вот их код на C ++ (они используют вектор вместо массива, но это должно быть довольно легко переводить).

#include <iostream>
#include <set>
#include <vector>
#include <iterator>

typedef std::set<int> set_type;
typedef std::set<set_type> powerset_type;

powerset_type powerset(set_type const& set)
{
  typedef set_type::const_iterator set_iter;
  typedef std::vector<set_iter> vec;
  typedef vec::iterator vec_iter;

  struct local
  {
    static int dereference(set_iter v) { return *v; }
  };

  powerset_type result;

  vec elements;
  do
  {
    set_type tmp;
    std::transform(elements.begin(), elements.end(),
                   std::inserter(tmp, tmp.end()),
                   local::dereference);
    result.insert(tmp);
    if (!elements.empty() && ++elements.back() == set.end())
    {
      elements.pop_back();
    }
    else
    {
      set_iter iter;
      if (elements.empty())
      {
        iter = set.begin();
      }
      else
      {
        iter = elements.back();
        ++iter;
      }
      for (; iter != set.end(); ++iter)
      {
        elements.push_back(iter);
      }
    }
  } while (!elements.empty());

  return result;
}

int main()
{
  int values[4] = { 2, 3, 5, 7 };
  set_type test_set(values, values+4);

  powerset_type test_powerset = powerset(test_set);

  for (powerset_type::iterator iter = test_powerset.begin();
       iter != test_powerset.end();
       ++iter)
  {
    std::cout << "{ ";
    char const* prefix = "";
    for (set_type::iterator iter2 = iter->begin();
         iter2 != iter->end();
         ++iter2)
    {
      std::cout << prefix << *iter2;
      prefix = ", ";
    }
    std::cout << " }\n";
  }
}

Выход:

{  }
{ 2 }
{ 2, 3 }
{ 2, 3, 5 }
{ 2, 3, 5, 7 }
{ 2, 3, 7 }
{ 2, 5 }
{ 2, 5, 7 }
{ 2, 7 }
{ 3 }
{ 3, 5 }
{ 3, 5, 7 }
{ 3, 7 }
{ 5 }
{ 5, 7 }
{ 7 }
5 голосов
/ 25 марта 2009

Это был один из моих проектов в колледже 4/5 лет назад, и я не могу хорошо напомнить алгоритм. Как я вижу, моя память использует массив в качестве исходного набора и битовую маску, чтобы указать, какие элементы присутствуют в текущем подмножестве.
Вот не проверенный код из архива:

#include <iostream>

#ifndef H_SUBSET
#define H_SUBSET

template <class T>
class Subset {
    private:
        int Dimension;
        T *Set;
        bool *Bitmask;
    public:
        Subset(T *set, int dim);
        ~Subset(void);
        void Show(void);
        void NextSubset(void);
        void EmptySet(void);
        void FullSet(void);
        int SubsetCardinality(void);
        int SetCardinality(void);
        T operator[](int index);
};      

template <class T>
int Subset<T>::SetCardinality(void) {
    return Dimension;
}

template <class T>
int Subset<T>::SubsetCardinality(void) {
    int dim = 0;
    for(int i = 0; i<Dimension; i++) {
        if(Bitmask[i]) {
            dim++;
        }
    }
    return dim;
}

template <class T>
void Subset<T>::EmptySet(void) {
    for(int i = 0; i<Dimension; i++) {
        Bitmask[i] = 0;
    }
    return;
}

template <class T>
void Subset<T>::FullSet(void) {
    for(int i = 0; i<Dimension; i++) {
        Bitmask[i] = 1;
    }
    return;
}

template <class T>
void Subset<T>::NextSubset(void) {
    int i = Dimension - 1;
    while(!Bitmask[i]&&i>=0) {
        i--;
        if(i<0) {
            break;
        }
    }
    if(i>=0) {
        Bitmask[i] = !Bitmask[i];
    }
    for(int j = i+1; j < Dimension; j++) {
        Bitmask[j] = !Bitmask[j];
    }
    return;
}

template <class T>
void Subset<T>::Show(void) {
    std::cout << "{ ";
    for(int i=0; i<Dimension; i++) {
        if(Bitmask[i]) {
            std::cout << Set[i] << ", ";
        }
    }
    std::cout << "}";
    return;
}

template <class T>
Subset<T>::Subset(T *set, int dim) {
    Set = new T[dim];
    Bitmask = new bool[dim];
    Dimension = dim;
    for(int i=0; i<dim; i++) {
        Set[i] = set[i];
        Bitmask[i] = true;
    }
}

template <class T>
Subset<T>::~Subset(void) {
    delete [] Set;
    delete [] Bitmask;
}
#endif // H_SUBSET

А если это твоя домашняя работа, ты убиваешь себя, братан;)

1 голос
/ 25 марта 2009

Ты;

А) Действительно хотите найти все возможные подмножества?

или

B) Хотите узнать, сколько элементов в массиве можно объединить, чтобы получить заданное число, и увидеть A) как шаг к решению?

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

Если это B), то сначала вы должны отсортировать массив и работать оттуда.

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