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

Usuwanie duplikatów - Lista jednokierunkowa

Object Storage Arubacloud
0 głosów
184 wizyt
pytanie zadane 13 sierpnia 2023 w C i C++ przez Janchess Początkujący (480 p.)

Cześć

Napisałem taki oto kod, którego zadaniem jest usunięcie duplikatów z listy jednokierunkowej. Niestety otrzymuję błąd "naruszenie dostępu do odczytu". Co w tym kodzie jest nie tak i gdzie powinienem szukać błędu gdy ta sytuacja powtórzy się w przyszłości, w innych programach.

struct lista {
	int dane;
	lista* nastepny;
};

void duplikaty(lista*& head) {
	for (lista* i = head; i != NULL; i = i->nastepny) {
		lista* poprzedni = i;
		for (lista* j = i->nastepny; j != NULL; j = j->nastepny) {
			if (i->dane == j->dane) {
				poprzedni->nastepny = j->nastepny;
				lista* tmp = j;
				delete tmp;
			}
			else poprzedni = j;
		}
		
	}
}

 

komentarz 13 sierpnia 2023 przez reaktywny Nałogowiec (41,050 p.)
edycja 13 sierpnia 2023 przez reaktywny

Pytania pomocnicze: A masz w ogóle jakiś pomysł jak pozbyć się duplikatów? Chcesz sortować elementy na liście? Czy w jakiś inny sposób?

Zamieniłbym nazwę lista na węzeł :)

Zrób sobie funkcje dodaj_element(), usun_element(), itd. itd.  Łatwiej będzie pracować na liście.

komentarz 13 sierpnia 2023 przez Janchess Początkujący (480 p.)
Trzymam pierwszy element listy (element bieżący) i potem sprawdzam po całej liście czy ten element ma jakieś duplikaty jeśli tak to usuwam duplikat i kontynuuje przeglądanie aż do końca listy potem zmieniam element bieżący o jeden itd...
komentarz 13 sierpnia 2023 przez Janchess Początkujący (480 p.)

@reaktywny, Przepraszam bardzo ale nie pytam oto jak poprawić estetykę kodu, jak nazwać węzły itp. Chce się dowiedzieć jak poprawić błąd programu...

komentarz 13 sierpnia 2023 przez reaktywny Nałogowiec (41,050 p.)
Musisz pamiętać. że nieco inaczej kasujesz element z końca, z środka i z początku listy. U ciebie oczywiście odpada przypadek kasowania pierwszego elementu. Po skasowaniu jednego elementu lista musi pozostać prawidłowo funkcjonującą listą.

2 odpowiedzi

+2 głosów
odpowiedź 13 sierpnia 2023 przez Oscar Nałogowiec (29,320 p.)
Spójrz na linie 12 i 13 - w sumie kasujesz element wskazywany przez j, a zaraz w ramach pętli wykonujesz j=j->następny.
+1 głos
odpowiedź 13 sierpnia 2023 przez Wiciorny Ekspert (270,230 p.)

Trzymam pierwszy element listy (element bieżący) i potem sprawdzam po całej liście czy ten element ma jakieś duplikaty jeśli tak to usuwam duplikat i kontynuuje przeglądanie aż do końca listy potem zmieniam element bieżący o jeden itd...

Kod wydaje się próbować usuwać duplikaty, ale jego działanie nie zawsze będzie poprawnie  W momencie usunięcia j-tego elementu, nie powinieneś przesuwać poprzedni na j, ponieważ teraz poprzedni i j wskazują na ten sam element, co może prowadzić do problemów.

Co ważne w takim podejściu twój kod nie uwzględnia usuwania duplikatów z pierwszego elementu, druga sprawa kod ma złożoność czasową O(n^2), co może być kosztowne dla dłuższych list.
Kolejna sprawa, że poo usunięciu elementu z listy, tworzysz tymczasowy wskaźnik tmp, ale nadal używasz wskaźników do elementów, które zostały już usunięte

Zakładając liste np. : 
 

1 -> 1 -> 2 -> 2 -> 3

 

Podobne pytania

+1 głos
1 odpowiedź 150 wizyt
pytanie zadane 5 czerwca 2020 w C i C++ przez kamylmeister Nowicjusz (190 p.)
0 głosów
0 odpowiedzi 243 wizyt
pytanie zadane 25 kwietnia 2018 w C i C++ przez damianoom Nowicjusz (240 p.)
–1 głos
1 odpowiedź 1,495 wizyt
pytanie zadane 15 stycznia 2017 w Algorytmy przez Mariusz M Obywatel (1,640 p.)

92,584 zapytań

141,434 odpowiedzi

319,671 komentarzy

61,968 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!

...