Как проверить, является ли вектор подмножеством другого в c ++ - PullRequest
11 голосов
/ 02 августа 2011

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

std::includes, кажется, работает только для отсортированных диапазонов.Как мне это сделать?

Ответы [ 4 ]

20 голосов
/ 02 августа 2011

Скопируйте векторы. Сортировать копии. Затем используйте std::includes на копиях.

template <typename T>
bool IsSubset(std::vector<T> A, std::vector<T> B)
{
    std::sort(A.begin(), A.end());
    std::sort(B.begin(), B.end());
    return std::includes(A.begin(), A.end(), B.begin(), B.end());
}
8 голосов
/ 02 августа 2011

Мой ответ предполагает, что когда вы говорите «подмножество», вы действительно ищете больше эквивалента «подстроки»;то есть поддержание порядка элементов во время поиска.


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

template <typename T1, typename T2>
bool contains(std::vector<T1> const& a, std::vector<T2> const& b) {
   for (typename std::vector<T1>::const_iterator i = a.begin(), y = a.end(); i != y; ++i) {
      bool match = true;

      typename std::vector<T1>::const_iterator ii = i;
      for (typename std::vector<T2>::const_iterator j = b.begin(), z = b.end(); j != z; ++j) {
          if (ii == a.end() || *j != *ii) {
              match = false;
              break;
          }
          ii++;
      }

      if (match)
         return true;
   }

   return false;
}

Демонстрация в реальном времени.

(Возможно, вы могли бы быть более креативными с параметрами шаблона.)

3 голосов
/ 02 августа 2011

Предполагается, что дубликаты НЕ имеют значения.Таким образом, если у вас есть два экземпляра числа 99 в векторе a, то, если вектор b имеет хотя бы один экземпляр числа 99, он будет объявлен как подмножество.

bool isSubset(const std::vector<int>& a, const std::vector<int>& b)
{
    for (std::vector<int>::const_iterator i = a.begin(); i != a.end(); i++)
    {
        bool found = false;

        for (std::vector<int>::const_iterator j = b.begin(); j != b.end(); j++)
        {
            if (*i == *j)
            {
                found = true;
                break;
            }
        }

        if (!found)
        {
            return false;
        }
    }

    return true;
}
0 голосов
/ 21 декабря 2017

Без сортировки ...

template <typename T>
bool isSubsetOrEqual(std::vector<T> const& a, std::vector<T> const& b) {
   for(auto const& av:a){
      if(std::find(b.begin(),b.end(),av)==b.end())
          return false;
   }
   return true;
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...