Practicum Datastructuren – voorjaar 2006 – Opgave 9: Backtrack–algoritmen 1 Achtergrond Veel problemen zijn niet analytisch op te lossen. We kunnen zon probleem soms met een computer oplossen door van alle denkbare megelijkheden na te gaan of ze inderdaad werkelijke oplossingen zijn. Een te volgen aanpak kan dan zijn: zolang niet alle denkbare mogelijkheden geprobeerd doe: pak een volgende mogelijkheid en controleer of die een oplossing is. Afhankelijk van het specifieke probleem moet deze aanpak bijgeschaafd en verder uitgewerkt worden. Vaak kan zon trial–and–error –aanpak elegant op een recursieve wijze geformuleerd worden. Een simpele aanpak hierbij gaat op de brute force manier (alle mogelijke combinaties proberen). Een intelligentere maar toch relatief eenvoudige aanpak, is die met een ‘terugkrabbel’ backtrack-algoritme. 2 Leerdoelen Na afloop van deze opdracht ben je in staat om: • aan te geven wat de basisprincipes zijn van backtrack -algoritmen; • een situatie voor toepassing van een backtrack -aanpak te herkennen; • voor eenvoudige problemen een correcte backtrack -oplossing in de vorm van een algoritme te bedenken. 3 Instructie Bestudeer allereerst de onderdelen over backtracking uit het Algoritmiek-dictaat (het [eerste deel van] hoofdstuk 12) en de sheets over dit onderwerp zoals die gebruikt zijn op het college (bekijk beslist ook de programma-voorbeelden) en uiteraard je op het hoorcollege gemaakte eigen aantekeningen en begin pas daarna aan de volgende deelopdrachten. 4 Probleemschets 1: Aantal combinaties Beschouw de volgende situatie met een rijtje getallen, die allemaal groter dan 1 zijn, bijvoorbeeld: const int Rijlengte = 10; int rij [ Rijlengte ] = { 2 , 3 , 5 , 7 , 8 , 11 , 13 , 17 , 19 , 23 , 29 } ; We kunnen ons nu afvragen op hoeveel manieren een gegeven getal als product van de getallen in de rij te schrijven is. De getallen in de rij mogen hierbij zo vaak gebruikt worden als je maar wilt. Het getal 20 is bijvoorbeeld te schrijven als 2 ∗ 2 ∗ 5, of 2 ∗ 5 ∗ 2, of 5 ∗ 2 ∗ 2, op 3 manieren dus. Er blijkt geen enkele manier te zijn om het getal 31 als product van getallen in deze rij te schrijven. Het moet mogelijk zijn om via het toetsenbord de waarde van het getal in te voeren, waarvoor bepaald moet worden óf (hoe vaak etc.) het als product van die getallen geschreven kan worden. Deelopdracht 1 Ontwerp en implementeer de functie int bepaalAantalCombinaties ( int doelwaarde, int product) waarmee backtrackend bepaald wordt hoe vaak het getal doelwaarde gemaakt kan worden uit de gegeven globale rij van getallen. De parameter product wordt gebruikt om het product van de getallen die op het moment van de aanroep uit de gegeven rij zijn gekozen. De beginwaarde bij de hoofdaanroep (bijvoorbeeld vanuit je main-functie) is 1. Waarschijnlijk kan je backtrack-aanpak impliciet zijn. Deelopdracht 2 Hetzelfde als bij 1 waarbij echter oplossingen waarvoor precies dezelfde reeks cijfers is gebruikt maar één keer meetellen. Dus 2 ∗ 2 ∗ 5, 2 ∗ 5 ∗ 2 en 5 ∗ 2 ∗ 2 tellen als één oplossing voor het doelgetal 20. Bovendien moet(en) de oplossing(en) getoond worden. Test je programma uit en neem de uitvoer op als commentaarregel(s) aan het einde van je programmacode. 5 Probleemschets 2: Som van kwadraten Voor sommige gehele getallen is het mogelijk hun kwadraat te schrijven als de som van de kwadraten van andere gehele getallen. Zo kunnen we bijvoorbeeld 5 schrijven als 52 = 32 + 42 . We beperken ons tot positieve gehele getallen en tot situaties waarbij in de kwadratenserie elk getal slechts éénmaal mag voorkomen (niet toegestaan is dus bijvoorbeeld: 22 = 12 +12 +12 +12 ). Voor sommige getallen is het niet mogelijk om het kwadraat als de som van andere kwadraten te schrijven, voor andere getallen kan het wel, soms zelfs op meerdere manieren (bijvoorbeeld 92 = 82 + 42 + 12 = 62 + 52 + 42 + 22 ). Deelopdracht 3: Bepaal alle mogelijke kwadratenseries Stel nu een backtrack procedure void bepaalKwadratenSeries(int getal) op (plus eventueel benodigde hulpprocedures), die alle mogelijke manieren afdrukt om het kwadraat van opgegeven getal als een som van kwadraten te schrijven. Hint: je zult hier een hulprij en zeer waarschijnlijk expliciet backtracken bij nodig hebben. Bij de hoofdaanroep: bepaalKwadratenSeries( 9 ) ; krijg je als mogelijke uitvoer (je mag kiezen of je de kwadraten van laag naar hoog of van hoog naar laag afdrukt): Gevonden kwadratenserie(s) voor het getal 9 : 9∗9 = 1∗1 + 4∗4 + 8∗8 9∗9 = 2∗2 + 4∗4 + 5∗5 + 6∗6 Test het programma uit en neem de uitvoer op als commentaarregel(s) op het einde van je programma-code. Deelopdracht 4: Bepaal kortste kwadratenserie Breid je vorige procedure(s) nu zodanig uit, dat via een aanroep als bepaalKortsteKwadratenSeries(9); je bij de uitvoer alleen de kortste kwadratenserie te zien krijgt. 6 Producten Als producten moet je (uitgeteste!) C++-code hebben, die ervoor zorgt dat je programma voldoet aan de gevraagde specificaties van de diverse deelopdrachten en waarbij je uitwerking voldoet aan de kwaliteitscriteria zoals die gesteld worden (zie bij punt 7. Zelfreflectie). Geef bovendien de gevraagde uitvoer van deelopdracht 2 en 3. 7 Zelfreflectie Via de cursuswebpagina kom je uit op een aantal criteria die we ook bij deze cursus gebruiken voor de kwaliteit van je ingeleverde werk. Ga voor jezelf na [voor zover dat nu al van toepassing is] in hoeverre je uitwerkingen voldoen aan deze criteria. Je uitwerking van deze zelfreflectie moet je echter niet inleveren. 8 Inleveren van je producten Vóór maandag 24 april, 9.00 uur, en wel door jullie programma één keer via e-mail met de tekst PI2 opgave9 in de Subject:-regel naar S.Smetsers@science.ru.nl te sturen. Vergeet niet om in de uitwerking duidelijk jullie namen te vermelden.