6. Klasser

Programmeringsteknik

(Skillnad mellan versioner)
Hoppa till: navigering, sök
Versionen från 6 juli 2007 kl. 09.01 (redigera)
KTH.SE:u1w7eri0 (Diskussion | bidrag)
(Objektorientering)
← Gå till föregående ändring
Nuvarande version (12 januari 2011 kl. 15.21) (redigera) (ogör)
KTH.SE:u1h4cq58 (Diskussion | bidrag)
m (Exempel: Husdjursprogrammet)
 
(5 mellanliggande versioner visas inte.)
Rad 27: Rad 27:
sig just nu). Användaren får själv välja namn på sitt husdjur, och husdjurets sig just nu). Användaren får själv välja namn på sitt husdjur, och husdjurets
skick ska påverkas av hur den behandlas. skick ska påverkas av hur den behandlas.
 +
==Attribut== ==Attribut==
Rad 52: Rad 53:
# ugnsvolym # ugnsvolym
'''Svar:'''<span style="color:white">Alla alternativen skulle passa bra som attribut! </span> '''Svar:'''<span style="color:white">Alla alternativen skulle passa bra som attribut! </span>
 +
==Metoder== ==Metoder==
Rad 90: Rad 92:
'''Svar:'''<span style="color:white">Alla utom sista alternativet - det hör '''Svar:'''<span style="color:white">Alla utom sista alternativet - det hör
inte till själva kundvagnen.</span> inte till själva kundvagnen.</span>
 +
==Speciella metoder: konstruktorn och str== ==Speciella metoder: konstruktorn och str==
Rad 139: Rad 142:
# ett returvärde # ett returvärde
'''Svar:'''<span style="color:white">Parametern self är det enda som måste vara med. Attributen kan få värden via parametrar (som namn ovan), men det är inte nödvändigt. Konstruktorn ska inte ha något returvärde.</span> '''Svar:'''<span style="color:white">Parametern self är det enda som måste vara med. Attributen kan få värden via parametrar (som namn ovan), men det är inte nödvändigt. Konstruktorn ska inte ha något returvärde.</span>
 +
==Klassen== ==Klassen==
Rad 198: Rad 202:
'''Svar:'''<span style="color:white">Nej, man kan bara banna ett husdjursobjekt, inte hela klassen. </span> '''Svar:'''<span style="color:white">Nej, man kan bara banna ett husdjursobjekt, inte hela klassen. </span>
 +
==Exempel: Husdjursprogrammet== ==Exempel: Husdjursprogrammet==
Rad 221: Rad 226:
print "glad: (^_^)" print "glad: (^_^)"
elif self.skick > 0: elif self.skick > 0:
- print "ledsen: (T_T)"+ print "trött: (T_T)"
else: else:
print "hungrig: ('o')" print "hungrig: ('o')"
Rad 368: Rad 373:
'''Svar:'''<span style="color:white">Man kan enkelt vända på listan efter sorteringen med djurlista.reverse()</span> '''Svar:'''<span style="color:white">Man kan enkelt vända på listan efter sorteringen med djurlista.reverse()</span>
 +
==Test== ==Test==
Rad 374: Rad 380:
på kursens förstasida. Även detta test rättas automatiskt och du har möjlighet på kursens förstasida. Även detta test rättas automatiskt och du har möjlighet
att göra om testet flera gånger om du inte lyckas på första försöket. att göra om testet flera gånger om du inte lyckas på första försöket.
 +
==Inlämningsuppgift 3 (Nöjesfält)== ==Inlämningsuppgift 3 (Nöjesfält)==
 +
 +''Inlämningsuppgiften är individuell och får inte lösas i grupp.''
[[Bild:Rollercoaster.jpg|frame|right]] [[Bild:Rollercoaster.jpg|frame|right]]

Nuvarande version

Innehåll

[redigera] Objektorientering

Objektorienterad programmering är ett sätt att kombinera data och funktioner för att bilda objekt. Objekten är ofta modeller av verkliga ting. Om vi vill skriva ett program för att simulera ett rymdskepp kan vi till exempel ha hastighet och lägeskoordinater som data. Rymdskeppets funktioner kan vara att accelerera i en viss riktning och att rotera.

Två rymdskepp

För att kunna använda objekt i sitt program måste man först definiera en klass. Klassen beskriver vilka data ett objekt ska kunna innehålla och vilka funktioner det ska ha. Om vi definierat en klass för rymdskepp kan vi sedan skapa hur många rymdskepps-objekt vi vill!

Fråga: Vi vill skriva ett program för att lägga patiens. Vilken av följande klasser ska vi definiera i programmet?

  1. Poäng
  2. Spader
  3. Kort

Svar:Klassen Kort, så att vi kan skapa många kort-objekt som kan användas i patiensen.

Det stora exemplet i det här avsnittet kommer att handla om ett virtuellt husdjur. Ett husdjur kan ju ha många egenskaper, men vi väljer ut två att ta med i vår klass: namn och skick (ett heltal som talar om hur den känner sig just nu). Användaren får själv välja namn på sitt husdjur, och husdjurets skick ska påverkas av hur den behandlas.


[redigera] Attribut

De variabler som används för att lagra data som hör till objektet kallas attribut. Våra husdjursobjekt ska alltså ha attributen namn och skick. Om man har flera objekt av samma typ är attributens värde helt oberoende av varandra. Två olika husdjur får ha olika namn och olika skick.

Det finns en synlig skillnad mellan attribut och vanliga variabler. Inuti klassen använder man alltid ordet self när man vill komma åt ett attribut. Man kan till exempel skriva så här:

self.namn = "Fido"

Med self menar man objektet självt.


Fråga: Anta att vi jobbar med ett program för vitvaruhandeln. Vilka av följande alternativ skulle passa som attribut för ett spis-objekt?

  1. bredd
  2. energiklass
  3. ugnsvolym

Svar:Alla alternativen skulle passa bra som attribut!


[redigera] Metoder

Funktionerna som hör till ett objekt kallas metoder. För att en funktion ska bli en metod måste den definieras inuti klassen. Den första parametern i en metod måste alltid vara self. Via self kan metoden använda objektets attribut.

Säg att vi vill ha en metod som ger vårt husdjur bannor (i uppfostrande syfte). Metoden ska visa bannorna, och även minska värdet på skick eftersom husdjuret blir på sämre humör.

    def banna(self):
        print
        print "- Fy på dig", self.namn, "!"
        self.skick -= 3

Som synes skriver man en metod precis som en funktion, sånär som på ordet self. Samma regler för parametrar och returvärden gäller.

Husdjursklassen ska så småningom få metoder som gör följande:

  • skapar ett husdjursobjekt och ger attributen värden
  • visar i vilket skick husdjuret är
  • bannar husdjuret
  • matar husdjuret
  • leker med husdjuret
  • tar avsked av husdjuret


Fråga: I en internetbutik vill vi ha ett objekt som representerar en kundvagn. Vilka av följande kan tänkas bli metoder i ett kundvagns-objekt?

  1. lägga ner en vara i kundvagnen
  2. beräkna totalpriset för alla varor i kundvagnen
  3. räkna ut dagens vinst för internetbutiken

Svar:Alla utom sista alternativet - det hör inte till själva kundvagnen.


[redigera] Speciella metoder: konstruktorn och str

Metoder får man döpa precis som man vill. Men det finns ett par metodnamn som har speciell betydelse för Python. Här tar vi upp två av dessa. Metodnamnen börjar och slutar med två understrykningstecken, så det finns ingen risk att man råkar välja namnen av misstag.

Konstruktorn är en speciell metod som man skriver för att initiera objekt, dvs ge objektets attribut de värden de ska ha från början. Konstruktorn anropas automatiskt varje gång ett objekt skapas. I Python måste man döpa sin konstruktor till __init__ (alltså init med två understrykningstecken både före och efter).

Konstruktorn för vårt husdjursobjekt kommer att se ut så här:

    def __init__(self, djurnamn):
        self.namn = djurnamn
        self.skick = 0

Den här konstruktorn har två parametrar, self (som ju alltid måste vara med) och djurnamn. Raden self.namn = djurnamn skapar attributet namn och initierar det till värdet av parametern djurnamn. På samma sätt skapar raden self.skick = 0 attributet skick och initierar det till noll. Varje gång man gör ett nytt objekt kommer konstruktorn att anropas.

Den andra specialmetoden vi ska ta upp här heter __str__, och det speciella med den är att den anropas varje gång man försöker skriva ut ett objekt med print. Här kan man alltså tala om hur man vill att objektets data ska skrivas ut. Exempel:

    def __str__(self):
        return self.namn + "*" + str(self.skick)

Metoden __str__ är särskilt användbar under tiden man arbetar med programmet, för att det blir enklare att göra kontrollutskrifter.

Fråga: Vilket av följande måste en konstruktor ha?

  1. parametern self
  2. en parameter för varje attribut
  3. ett returvärde

Svar:Parametern self är det enda som måste vara med. Attributen kan få värden via parametrar (som namn ovan), men det är inte nödvändigt. Konstruktorn ska inte ha något returvärde.


[redigera] Klassen

För att kunna skapa ett objekt måste man först definiera en klass. Det är i klassen man skriver de metoder som objektet ska ha. Klasser brukar ha namn som börjar med versal, så att man lätt ska kunna se att det är en klass. Så här långt har vi hunnit i vår husdjursklass hittills:

# En klass som beskriver ett virtuellt husdjur.
# Attribut:
#    namn - djurets nanm
#    skick - ett heltal som beskriver djurets skick
class Husdjur:

    # Konstruktorn
    def __init__(self, djurnamn):
        self.namn = djurnamn
        self.skick = 0

    # För utskrift med print
    def __str__(self):
        return self.namn + "*" + str(self.skick)

    # Ger husdjuret bannor. Skick minskas.
    def banna(self):
        print
        print "- Fy på dig", self.namn, "!"
        self.skick -= 3

Med en klass är det likadant som med en funktion - det händer inget förrän man använder den i huvudprogrammet. För att skapa ett Husdjurs-objekt kan vi skriva:

djur = Husdjur("Muffin")

Det som händer här är att konstruktorn __init__ anropas, så att attributen skapas och initieras. Sen returneras ett Husdjurs-objekt, som lagras i variabeln djur

Nu har vi ett husdjursobjekt i variabeln djur. Hur anropar vi en av djurets metoder? Just nu finns bara metoden banna att välja på:

djur.banna()

När vi vill komma åt ett attribut inuti klassen använder vi ju en punkt, och ett liknande skrivsätt använder man alltså när man vill anropa en metod från ett objekt.

Fråga: Kan man skriva Husdjur.banna()?

Svar:Nej, man kan bara banna ett husdjursobjekt, inte hela klassen.


[redigera] Exempel: Husdjursprogrammet

Här följer hela programmet, först Husdjurs-klassen, och sedan huvudprogrammet. Lägg märke till att alla metoderna i klassen är indenterade.

# En klass som beskriver ett virtuellt husdjur.
# Attribut:
#    namn - djurets nanm
#    skick - ett heltal som beskriver djurets skick
class Husdjur:

    # Konstruktorn, initierar attributen namn och skick.
    def __init__(self, djurnamn):
        self.namn = djurnamn
        self.skick = 0

    # Visar husdjurets namn och skick
    def visaSkick(self):       
        print self.namn, "är ",
        if self.skick > 5:
            print "glad: (^_^)"
        elif self.skick > 0:
            print "trött: (T_T)"
        else:
            print "hungrig: ('o')"
        print

    # Ger husdjuret bannor. Skick minskas.
    def banna(self):
        print
        print "- Fy på dig", self.namn, "!"
        self.skick -= 3

    # Ger husdjuret mat. Skick ökar.
    def mata(self, mat):
        print
        for i in range(mat):
            print "GLUFS",
        self.skick += mat

    # Leker med husdjuret. Skick kan öka eller minska.
    def leka(self):
        print
        if self.skick < 0:
            self.skick -= 1
            print self.namn, " vill inte leka."
        else:
            self.skick += 1
            print "~~~~~~~~~~~ WHEEEEEEE! ~~~~~~~~~~~"

    # Skriver ut avskedet.
    def avsked(self):
        print
        print "Hejdå,", self.namn, "kommer att sakna dig!"

# Här slutar Husdjursklassen


#  --------  Här börjar huvudprogrammet -----------
djurnamn = raw_input("Vad vill du döpa ditt husdjur till? ")
djur = Husdjur(djurnamn)
djur.visaSkick()
svar = raw_input(" Vill du \n  banna \n  mata \n  leka med \n ditt husdjur? " )
while svar:
    if svar[0]=="m":
        bullar = input("Hur många bullar? ")
        djur.mata(bullar)    
    elif svar[0]=="b":
        djur.banna()
    elif svar[0]=="l":
        djur.leka()
    else:
        print "Hursa? "
    djur.visaSkick()
    svar = raw_input(" Vill du \n  banna \n  mata \n  leka med \n ditt husdjur? " )
djur.avsked()

Provkör programmet och svara på följande frågor.

  • Fråga: Hur många objekt skapas i programmet?

Svar:Bara ett - variabeln djur.

  • Fråga: När metoden visaSkick skriver ut att husdjuret är trött - vilka värden har attributet skick då?

Svar:Attributet skick har något av värdena 1,2,3,4 eller 5

  • Fråga: Hur avslutar man programmet?

Svar:När man trycker på Retur avslutas programmet. Lägg gärna till en print-sats som upplyser användaren om det!

  • Fråga: Hur många metoder finns det i klassen Husdjur?

Svar:Det finns sex metoder, inklusive konstruktorn.

  • Fråga: Kan man få programmet att krascha?

Svar:Javisst, prova att skriva tre (istället för 3) när du matar husdjuret med bullar. Här skulle det behövas lite felhantering!

[redigera] Sortering av en lista med objekt

Oftast skapar man fler än ett objekt i sitt program. Då är det praktiskt att spara objekten i en lista. En lista med objekt kan enkelt sorteras efter sina attribut. Men på något sätt måste man ange vilket attribut man vill sortera på. Är det namn eller skick?

Så här kan sortering av en lista skrivas:

   lista.sort(key = lambda x:x.attr)

Parametern key i sort visar att man vill ange vilket attribut man vill sortera på. Uttrycket lambda x:x.attribut är som en liten funktion i miniatyr där parametern står före kolonet och returvärdet efter. Om vi tänker oss att x är objektet så är attr det av objektets attribut som vi vill sortera på.

Om det attribut man sorterar på är en sträng görs sorteringen efter bokstavsordning. Är attributet ett tal så blir objekten sorterade i stigande ordning. Vi förenklar Husdjursklassen en aning och visar med exemplet nedan hur sorteringen kan göras i praktiken:

import random

# En klass som beskriver ett virtuellt husdjur.
# Attribut:
#    namn - djurets nanm
#    skick - ett heltal som beskriver djurets skick
class Husdjur:

    # Konstruktorn, initierar attributen namn och skick.
    def __init__(self, djurnamn):
        self.namn = djurnamn
        self.skick = random.randrange(-5,6)
        
    # För utskrift av ett objekt med print
    def __str__(self):
        return self.namn + "*" + str(self.skick)
    
#  -------  Här slutar Husdjursklassen ---------

# En funktion för utskrift av listan
def skrivListan(listan):
    for o in listan:
        print o,
    print "\n"

#  --------  Här börjar huvudprogrammet ---------

djurlista = [Husdjur("Missan"), Husdjur("Blixten"), Husdjur("Fido"), Husdjur("Miso")]

print "Så här ser listan ut från början:"
skrivListan(djurlista)

djurlista.sort(key = lambda husdjur:husdjur.namn)
print "Nu är listan sorterad efter namn, i bokstavsordning:"
skrivListan(djurlista)
        
djurlista.sort(key = lambda husdjur:husdjur.skick)
print "Nu är listan sorterad efter skick, i stigande ordning:"
skrivListan(djurlista)

Fråga: Hur gör man om man vill sortera i avtagande istället för i stigande ordning?

Svar:Man kan enkelt vända på listan efter sorteringen med djurlista.reverse()


[redigera] Test

Dags för test nummer 6. Testet hittar du som vanligt under rubriken Examination på kursens förstasida. Även detta test rättas automatiskt och du har möjlighet att göra om testet flera gånger om du inte lyckas på första försöket.


[redigera] Inlämningsuppgift 3 (Nöjesfält)

Inlämningsuppgiften är individuell och får inte lösas i grupp.

Du ska skriva ett program som simulerar ett nöjesfält. Använd objekt för att representera attraktionerna. En attraktion kan till exempel ha namn, minimilängd (för att få åka), antal passagerare och magpirrfaktor. Den ska kunna göra reklam för sig, startas, stoppas och haverera.

Skriv också ett huvudprogram där användaren får välja mellan tre olika attraktioner. Vald attraktion ska startas, kanske haverera (slumpa fram haverier) och stannas. När attraktionen är igång ska det synas på skärmen, till exempel med Iiiih från berg-och-dalbanan, skratt från lustiga huset och så vidare.

Kravet för att bli godkänd är att du med ditt program visar att du kan skriva en egen klass med attribut och metoder, och använda den i ett program. I övrigt har du fria händer!

Programmet ska finnas i en fil med namnet Uppgift3.py. Följ den här mallen:

# Programmeringsteknik webbkurs KTH inlämningsuppgift 3.
# <Ditt namn>
# <Datum>
# <Kort beskrivning av vad programmet gör>

<Programkod>

Innan du skickar in programmet så ska du testa att det fungerar. Se till att du har provat alla klassens metoder, och fått se alla utskrifter som ditt program gör.

När du har testat ditt program grundligt så går du till studentportalen och klickar dig fram till Inlämningsuppgift 3 och där lämnar du in filen Uppgift3.py. Observera att filen Uppgift3.py måste vara i textformat - vilket den automatiskt blir om du skapar den med hjälp av IDLE. Detta för att vi ska kunna prova och testköra ditt program.

Personliga verktyg