Mergesort: Unterschied zwischen den Versionen

Aus SibiWiki
Zur Navigation springen Zur Suche springen
(Die Seite wurde neu angelegt: „Kategorie:Sortierverfahren thumb|Mergesort für das Wort "informatik" |428px '''<font color='red'>nicht relevant fürs Zentralabitu…“)
 
 
(6 dazwischenliegende Versionen von 2 Benutzern werden nicht angezeigt)
Zeile 16: Zeile 16:
=Implementierung=  
=Implementierung=  
<code>
<code>
  private List teile(List pList){
  '''public List<String> mergesort(List<String> pList){'''
    '''//TODO'''  
  List<String> ergebnis = new List<>();
  }
  List<String> links = new List<>();
  List<String> rechts = new List<>();
  '''// Abbruchbedingung: nur 1 Element in pList'''
  pList.toFirst();
  pList.next();
  if(pList.hasAccess() == false) {
      return pList;
  }
  '''// verteilen auf links und rechts'''
  boolean nachLinks = true;
  for(pList.toFirst(); pList.hasAccess(); pList.next()) {
      String s = pList.getContent();
      if(nachLinks == true) {
        links.append(s);
        nachLinks = false;
      }
      else {
        rechts.append(s);
        nachLinks = true;
      }
  }
  '''//sortieren der Teillisten durch rekursive Aufrufe'''
  List<String> linksSortiert = '''mergesort(links);'''
  List<String> rechtsSortiert = '''mergesort(rechts);''' 
  '''// die beiden sortierten Listen zu einer einzigen verschmelzen'''
  ergebnis = merge(linksSortiert, rechtsSortiert);
  return ergebnis;
  }
</code> 


public List<Person> mergeSort(List<Person> pList){
'''Die folgende Hilfs-Methode verschmilzt zwei sortierte Listen zu einer einzigen.'''
    List<Person> ergebnis = new List<>();
    List<Person> hintereListe = new List<>();


    // abbruchbedingung: nur 1 element in pList
<code>   
    pList.toLast();
  '''private List<String> merge(List<String> l1, List<String> l2) {'''
    Person hinten = pList.getContent();
  List<String> ergebnis = new List<>();
    pList.toFirst();
  l1.toFirst();
    Person vorne = pList.getContent();
  l2.toFirst();
    if(vorne == hinten)
  // beide Listen parallel durchlaufen
    {
  while(l1.hasAccess() && l2.hasAccess()){
      // pList enthaelt nur ein Element!
      // jeweils die erste String auslesen
      return pList;
      String s1 = l1.getContent();
    }
      String s2 = l2.getContent();
 
      // die alphabetisch kleinere der beiden Strings
    hintereListe = teile (pList);
      // in die Liste ergebnis uebertragen
      // und in der Herkunftsliste eins weiter gehen
    //sortieren der Teillisten durch rekursiven Aufruf
      if(s1.compareTo(s2) < 0){
    List<Person> pListSortiert = mergeSort(pList);
        ergebnis.append(s1);
    List<Person> hintersteListSortiert = mergeSort(hintereListe);
        l1.next();
 
      }
    // verschmelzen
      else{
    ergebnis = verschmelzen(pListSortiert, hintersteListSortiert);
        ergebnis.append(s2);
    System.out.println("ende von mergesort:");
        l2.next();
    return ergebnis;
      }
}
  }
</code>
  // den Rest der ersten Liste (wenn es einen gibt)
 
  // in ergebnis uebertragen
mergeSort ruft die Methode verschmelzen auf:
  while(l1.hasAccess()){
 
      ergebnis.append(l1.getContent());
<code>
      l1.next();
  /**
  }
  * verschmilzt zwei sortierte Listen
  // den Rest der zweiten Liste (wenn es einen gibt)
  * zu einer sortierten Liste
  // in ergebnis uebertragen
  */
  while(l2.hasAccess()){
  private List<Person> verschmelzen(List<Person> l1, List<Person> l2) {
      ergebnis.append(l2.getContent());
    List<Person> ergebnis = new List<>();
      l2.next();
    l1.toFirst();
  }
    l2.toFirst();
  return ergebnis;
    // beide Listen parallel durchlaufen
    while(l1.hasAccess() && l2.hasAccess()){
      // jeweils die erste Person auslesen
      Person p1 = (Person) l1.getContent();
      Person p2 = (Person) l2.getContent();
      // die alphabetisch kleinere der beiden Personen
      // in die Liste ergebnis uebertragen
      // und aus der Herkunftsliste loeschen
      if(p1.getName().compareTo(p2.getName()) < 0){
    ergebnis.append(p1);
    l1.remove();
      }
      else{
    ergebnis.append(p2);
    l2.remove();
      }
    }
    // den Rest der ersten Liste (wenn es einen gibt)
    // in ergebnis uebertragen
    while(l1.hasAccess()){
      ergebnis.append(l1.getObject());
      l1.remove();
    }
    // den Rest der zweiten Liste (wenn es einen gibt)
    // in ergebnis uebertragen
    while(l2.hasAccess()){
      ergebnis.append(l2.getObject());
      l2.remove();
    }
    return ergebnis;
  }
  }
</code>
</code>

Aktuelle Version vom 15. Dezember 2022, 17:09 Uhr


Mergesort für das Wort "informatik"

nicht relevant fürs Zentralabitur!

Funktionsweise

Mergesort ist ein Divide&Conquer-Verfahren.

Im Divide wird die Liste in zwei gleich große Teile aufgeteilt (wenn die Anzahl der Elemente ungerade ist, dann natürlich nicht ganz gleich groß...).

Beim Conquer werden die sortierten Teillisten zusammengefügt. Durch geschickte Implementierung kann man das in O(n) Zeit bewerkstelligen.

Laufzeit

Mergesort hat auch im Worst Case eine Laufzeit von O(n*log(n)).
Für Arrays ist allerdings Quicksort schneller, weil man bei Mergesort ständig neue Arrays erzeugen muss - bei Quicksort dagegen kann man in situ (bzw. in place) arbeiten.

Implementierung

public List<String> mergesort(List<String> pList){
  List<String> ergebnis = new List<>();
  List<String> links = new List<>();
  List<String> rechts = new List<>();
  // Abbruchbedingung: nur 1 Element in pList
  pList.toFirst();
  pList.next();
  if(pList.hasAccess() == false) {
     return pList;
  }
  // verteilen auf links und rechts
  boolean nachLinks = true;
  for(pList.toFirst(); pList.hasAccess(); pList.next()) {
     String s = pList.getContent();
     if(nachLinks == true) {
        links.append(s);
        nachLinks = false;
     }
     else {
        rechts.append(s);
        nachLinks = true;
     }
  }
  //sortieren der Teillisten durch rekursive Aufrufe
  List<String> linksSortiert = mergesort(links);
  List<String> rechtsSortiert = mergesort(rechts);  
  // die beiden sortierten Listen zu einer einzigen verschmelzen
  ergebnis = merge(linksSortiert, rechtsSortiert);
  return ergebnis;
}	

Die folgende Hilfs-Methode verschmilzt zwei sortierte Listen zu einer einzigen.

private List<String> merge(List<String> l1, List<String> l2) {
  List<String> ergebnis = new List<>();
  l1.toFirst();
  l2.toFirst();
  // beide Listen parallel durchlaufen
  while(l1.hasAccess() && l2.hasAccess()){
     // jeweils die erste String auslesen
     String s1 = l1.getContent();
     String s2 = l2.getContent();
     // die alphabetisch kleinere der beiden Strings
     // in die Liste ergebnis uebertragen
     // und in der Herkunftsliste eins weiter gehen
     if(s1.compareTo(s2) < 0){
        ergebnis.append(s1);
        l1.next();
     }
     else{
        ergebnis.append(s2);
        l2.next();
     }
  }
  // den Rest der ersten Liste (wenn es einen gibt)
  // in ergebnis uebertragen
  while(l1.hasAccess()){
     ergebnis.append(l1.getContent());
     l1.next();
  }
  // den Rest der zweiten Liste (wenn es einen gibt)
  // in ergebnis uebertragen
  while(l2.hasAccess()){
     ergebnis.append(l2.getContent());
     l2.next();
  }
  return ergebnis;
}