• Najnowsze pytania
  • Bez odpowiedzi
  • Zadaj pytanie
  • Kategorie
  • Tagi
  • Zdobyte punkty
  • Ekipa ninja
  • IRC
  • FAQ
  • Regulamin
  • Książki warte uwagi

Funkcja sprawdzająca, czy pierwszy vector zawiera wszystkie elementy drugiego

Object Storage Arubacloud
+1 głos
286 wizyt
pytanie zadane 22 grudnia 2021 w C i C++ przez MKolaj15 Bywalec (2,270 p.)

Witam, mam do napisania funkcję, która będzie sprawdzać, czy pierwszy vector zawiera wszystkie elementy drugiego vectora. Problem w tym, że nie wiem jak obsłużyć sytuację, w której pojawiają się duplikaty. Oto mój kod:

#include <iostream>
#include <string>
#include <vector>

void contains_all(std::vector<int> vec, std::vector<int> vec2)
{
        size_t count = 0;
        for(size_t i = 0; i < vec.size(); i++)
        {
                for(size_t j = 0; j < vec2.size(); j++)
                {
                        if(vec2[j]==vec[i])
                        {
                                count++;
                        }       
                } 

        }

        if(count>=vec2.size())
        {
                std::cout<<"Pierwszy vector zawiera wszystkie elementy drugiego vectora"<<std::endl;
        }
        else
        {
                std::cout<<"Pierwszy vector nie zawiera wszystkich elementów drugiego vectora"<<std::endl;
        }

}

auto main() -> int
{
        auto vec = std::vector<int>{2,3,5,2,1,4,5,6,7,8};
        auto vec2 = std::vector<int>{2,3,5,2,9};
         
        contains_all(vec, vec2);

return 0;
}


Byłbym wdzięczny za pomoc z napisaniem poprawnego kodu. Z góry dzięki!

2 odpowiedzi

+2 głosów
odpowiedź 22 grudnia 2021 przez Tomasz Sobczak Bywalec (2,830 p.)
wybrane 23 grudnia 2021 przez MKolaj15
 
Najlepsza

Jak można rozwiązać taki problem:

1. Sprawdzić czy istnieją już funkcje, które to rozwiązują lub upraszczają rozwiązanie.  Przykładowo:

//** funkcja FIND  -  sprawdzenie czy wektor V zawiera element X  **/

#include <algorithm>

if( std:: find(  v .begin() , v .end() , x ) != v .end() ) {
    /* ZAWIERA*/
} else {
    /* NIE ZAWIERA */
}


//** funkcja INCLUDES -  sprawdzenie czy posortowany zakres zawiera inny posortowany zakres   **/
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() );
}

2. Twój pomysł może osiągnąć poziom rozwiązania poprawnego. Wystarczy przeanalizować jego działanie i szukać co nie działa tak jak planowałeś.  Gdyby wystarczył rezultat: "Pierwszy vector zawiera wszystkie elementy drugiego vectora w przynajmniej jednej kopii" wtedy wystarczyła by prosta zmiana:

count++;
break;

W przeciwnym razie podszedłbym to rozwiązania inaczej:

bool contains_all( std::vector<int> vec, std::vector<int> vec2 )
{
//** KROK 1: Sortowanie wektorów **/
        std::sort( vec .begin(), vec .end() ); 
        std::sort( vec2 .begin(), vec2 .end() );

        auto v_pos { vec .begin() };

//** KROK 2: Czy każdy element vectora 2... **/      
        for ( auto& elem : vec2 ) 
        {
            //** KROK 3: Zawiera się w wektorze pierwszym **/
            while ( v_pos < vec .end() )
            {
                //cout << format( " [ {0} ] == [ {1} ] " , elem , *v_pos ) << endl;
                if ( elem == *v_pos )
                {
                    /** Tak, więc przechodzę do następnego elementu vec **/
                    ++v_pos;
                    /** Sprawdzam kolejny element vec2 **/
                    break;
                }
                else if ( *v_pos > elem )
                {
                    /** Dalej już nie znajdziemy bo wektory są posortowane. **/
                    return false;
                }
                v_pos++;
            }
        }
      
        return( ( v_pos < vec .end() ) ); // TRUE gdy Nie przekroczyło zakresu vec i sprawdzono wszystkie elementy vec2.
 
}

Przepis masz podany wraz z rozwiązaniem, które cię pewnie nie zainteresuje ;), ale korzystając z przepisu możesz ugotować własną funkcję.

komentarz 23 grudnia 2021 przez MKolaj15 Bywalec (2,270 p.)
Wielkie dzięki, naprawdę bardzo mi pomogłeś :)
+1 głos
odpowiedź 22 grudnia 2021 przez tkz Nałogowiec (42,000 p.)
#include <iostream>
#include <string>
#include <vector>
#include <set>

bool contains_all(const std::vector<int>& vec, const std::vector<int>& vec2)
{
    std::set<int> a1(vec.begin(), vec.end());
    std::set<int> a2(vec2.begin(), vec2.end());
    if(a1.size()<=a2.size())
    {
        for(const auto& i :a1)
        {
            if(a2.find(i)==a2.end())
            {
                return false;
            }
        }
    }
    return true;
}
 
auto main() -> int
{
    auto vec1 = std::vector<int>{2,4,2,3,3,5,0,9};
    auto vec2 = std::vector<int>{2,3,5,2,1,4,5,9,7,8};
      
    std::cout<<contains_all(vec1, vec2);
}

 

komentarz 22 grudnia 2021 przez j23 Mędrzec (194,920 p.)
Wydaje mi się, że OP chodziło o to, by uwzględniać ilość wystąpień.
komentarz 22 grudnia 2021 przez MKolaj15 Bywalec (2,270 p.)
Tak, ilość wystąpień ma być uwzględniona, ale dzięki i tak bardzo pomogłeś.
komentarz 22 grudnia 2021 przez j23 Mędrzec (194,920 p.)

Możesz użyć mapy wartość -> ilość wystąpień, wtedy będziesz mógł uwzględnić wystąpienia.

komentarz 26 grudnia 2021 przez TOWaD Mądrala (5,700 p.)

@tkz, 

bool contains_all(const std::vector<int>& vec, const std::vector<int>& vec2) {
    std::set<int> a1(vec.begin(), vec.end());
    std::set<int> a2(vec2.begin(), vec2.end());
    a1.insert(a2.begin(),a2.end());
    if(a1.size()==a2.size()) return true;
    else return false;
}

Wolniejsze, ale krócej zapisane.

komentarz 26 grudnia 2021 przez TOWaD Mądrala (5,700 p.)

Może <algorithm>

bool contains_all(std::vector<int>& vec, std::vector<int>& vec2) {
    std::vector<int>diff{};
    std::sort(vec.begin(),vec.end());
    std::sort(vec2.begin(),vec2.end());
    std::set_difference(vec.begin(), vec.end(),vec2.begin(), vec2.end(),std::back_inserter(diff));
    return diff.empty();
}

 

Podobne pytania

0 głosów
0 odpowiedzi 298 wizyt
0 głosów
3 odpowiedzi 1,471 wizyt
0 głosów
1 odpowiedź 174 wizyt

92,568 zapytań

141,422 odpowiedzi

319,634 komentarzy

61,956 pasjonatów

Motyw:

Akcja Pajacyk

Pajacyk od wielu lat dożywia dzieci. Pomóż klikając w zielony brzuszek na stronie. Dziękujemy! ♡

Oto polecana książka warta uwagi.
Pełną listę książek znajdziesz tutaj.

Akademia Sekuraka

Kolejna edycja największej imprezy hakerskiej w Polsce, czyli Mega Sekurak Hacking Party odbędzie się już 20 maja 2024r. Z tej okazji mamy dla Was kod: pasjamshp - jeżeli wpiszecie go w koszyku, to wówczas otrzymacie 40% zniżki na bilet w wersji standard!

Więcej informacji na temat imprezy znajdziecie tutaj. Dziękujemy ekipie Sekuraka za taką fajną zniżkę dla wszystkich Pasjonatów!

Akademia Sekuraka

Niedawno wystartował dodruk tej świetnej, rozchwytywanej książki (około 940 stron). Mamy dla Was kod: pasja (wpiszcie go w koszyku), dzięki któremu otrzymujemy 10% zniżki - dziękujemy zaprzyjaźnionej ekipie Sekuraka za taki bonus dla Pasjonatów! Książka to pierwszy tom z serii o ITsec, który łagodnie wprowadzi w świat bezpieczeństwa IT każdą osobę - warto, polecamy!

...