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

Zadanie Bracia

0 głosów
768 wizyt
pytanie zadane 16 sierpnia 2023 w C i C++ przez Dani Obywatel (1,450 p.)

Witam.

Rozwiązuję to zadanie << https://szkopul.edu.pl/problemset/problem/KkZDP16mELVF692Bf-NJsN1L/site/?key=statement. Nie mam pojęcia dlaczego nie przechodzi mi mój kod większości przypadków w testach. Ogólnie to wyznaczam końce i początki dla vectora bracia -> preprocessinguję tablicę początków i końców -> wyznaczam początek i koniec dla preprocessingu -> robię dp


Będę wdzięczny jak ktoś może rzucić oko na kod.
Z góry dziękuję.

#include <iostream>
#include <vector>
#include <algorithm>

using namespace std;

const int maxn = 1e6 + 5;
int poczatek[maxn];
int koniec[maxn];
std::vector<int> bracia;
std::vector<int> preprocessing;
int dp[maxn*2];

int main(){
    ios_base::sync_with_stdio(0);
    cin.tie(0);
    int n;
    cin >> n;
    
    bracia.push_back(0);

    for(int i=1;i<=n;++i){
        int x;
        std::cin >> x;

        bracia.push_back(x);
    }

    //poczatek dla bracia
    for(int i=1;i<=n;++i)
        poczatek[i] = -1;
    
    for(int i=1;i<=n;++i)
        if(poczatek[bracia[i]] == -1)
            poczatek[bracia[i]] = i;
    
    //koniec dla bracia
    for(int i=1;i<=n;++i)
        koniec[i] = -1;
    
    for(int i=n;i>=1;--i)
        if(koniec[bracia[i]] == -1)
            koniec[bracia[i]] = i;
    
    //preprocessing 3 2 4 2 5 - > 3 3 2 4 4 2 5 5
    preprocessing.push_back(0);
    for(int i=1;i<=n;++i){
        if(poczatek[bracia[i]] == koniec[bracia[i]])
        {
            preprocessing.push_back(bracia[i]);
            preprocessing.push_back(bracia[i]);
        }
        else if(poczatek[bracia[i]] == i || koniec[bracia[i]] == i)
            preprocessing.push_back(bracia[i]);
    }

    //poczatek dla preprocessing
    for(int i=1;i<=preprocessing.size()-1;++i)
        poczatek[i] = -1;
    
    for(int i=1;i<=preprocessing.size()-1;++i)
        if(poczatek[preprocessing[i]] == -1)
            poczatek[preprocessing[i]] = i;
    
    //koniec dla preprocessing
    for(int i=1;i<=preprocessing.size()-1;++i)
        koniec[i] = -1;
    
    for(int i=preprocessing.size()-1;i>=1;--i)
        if(koniec[preprocessing[i]] == -1)
            koniec[preprocessing[i]] = i;
    
    //dp
    dp[0] = 0;
    for(int i=1;i<=preprocessing.size()-1;++i){
        if(i == koniec[preprocessing[i]])
            dp[i] = max(dp[poczatek[preprocessing[i]]] + 1,dp[i-1]);
        else
            dp[i] = dp[i-1];
    }

    cout << dp[preprocessing.size()-1] << '\n';
    return 0;
}

 

1 odpowiedź

0 głosów
odpowiedź 17 sierpnia 2023 przez Whistleroosh Maniak (57,400 p.)
wybrane 17 sierpnia 2023 przez Dani
 
Najlepsza
for(int i=1;i<=preprocessing.size()-1;++i)
    poczatek[i] = -1;

czyścisz pozycje od 1 do preprocessing.size(), a potem w kodzie robisz poczatek[preprocessing[i]] = i; Tylko, że w treści zadania nie ma założenia, że preprocessing[i] <= preprocessing.size().

komentarz 17 sierpnia 2023 przez Dani Obywatel (1,450 p.)
Bardzo dziękuję Ci za pomoc. Teraz wchodzi już prawie na 100 pkt. W ostatnim teście wypisuje memory limit exceeded. Z tego co rozumiem to za dużo używam tablic/vectorów.
1
komentarz 17 sierpnia 2023 przez Whistleroosh Maniak (57,400 p.)

Możesz próbować skorzystać z reserve, żeby zarezerwować dokładnie tyla pamięci ile trzeba. Tak samo tablicę dp możesz tworzyć dopiero wtedy gdy wiadomo ile będzie miała elementów

komentarz 17 sierpnia 2023 przez reaktywny Nałogowiec (46,230 p.)
Zainteresował mnie ten reserve. Widzę, że sporo kodu trzeba wpisać żeby z tego korzystać, chyba, że źle zrozumiałem kod w przykładzie.

W Rust jest podobny mechanizm, można podać przewidywaną liczbę elementów a później rozszerzać wielkość kontenera. Ale tam to jest jedna linijka kodu.
1
komentarz 17 sierpnia 2023 przez Whistleroosh Maniak (57,400 p.)
Tu też wystarczy jedna linia np. v.reserve(50)
komentarz 17 sierpnia 2023 przez reaktywny Nałogowiec (46,230 p.)

To dobrze, mnie zmyliła:

struct NAlloc

Czy Szkopuł w ogólnym rankingu bierze pod uwagę liczbę wersji nadesłanego kodu? Niektóre strony biorą to pod uwagę.

komentarz 17 sierpnia 2023 przez Whistleroosh Maniak (57,400 p.)
Raczej nie
1
komentarz 17 sierpnia 2023 przez Dani Obywatel (1,450 p.)

@Whistleroosh, Działa! Dzięki wielkie (;

komentarz 17 sierpnia 2023 przez Dani Obywatel (1,450 p.)

@Whistleroosh, Czy reserve dodaje też na swoje miejsca zera tak jak resize?

komentarz 17 sierpnia 2023 przez Whistleroosh Maniak (57,400 p.)
Reserve nie zmienia rozmiaru wektora. Jak masz wektora v rozmiaru 10, to po v.reserve(100) on wciąż ma rozmiar 10. Tablica z której wewnętrznie korzysta v zwiększyła się, ale i tak nie masz bezpośredniego dostępu do nowych pól, więc nie ważne czy tam jest zero czy cokolwiek.

Podobne pytania

0 głosów
1 odpowiedź 318 wizyt
0 głosów
2 odpowiedzi 1,254 wizyt
pytanie zadane 22 maja 2023 w C i C++ przez Dani Obywatel (1,450 p.)
0 głosów
1 odpowiedź 541 wizyt
pytanie zadane 19 maja 2023 w C i C++ przez Dani Obywatel (1,450 p.)

93,720 zapytań

142,641 odpowiedzi

323,264 komentarzy

63,268 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

Twierdza Linux. Bezpieczeństwo dla dociekliwych

Aby uzyskać rabat -10%, użyjcie kodu pasja-linux, wpisując go w specjalne pole w koszyku.

...