Seite 1 von 2

Funktionen in Python nur am Anfang

Verfasst: Mittwoch 26. August 2009, 20:51
von zenker
Hallo,

ich bin neu her und wage gerade meine ersten Schritte in Python unter Linux.
Früher hab ich ein bisschen mit C rumgespielt und diverse Sachen unter Windows mit "Pure Basic" gemacht.
Und jetzt also Python - schaun wir mal! :D

In C und Pure Basic konnte ich Funktionen ans Ende des Sourcecodes stellen
und mit "declare" am Anfang deklarieren. Geht das bei Python auch oder muß die Funktion am Anfang des Source stehen?
Im "byteofpython_120.pdf" konnte ich bisher dazu nichts finden (oder hab es überlesen!)

Für Antworten im voraus herzlichen Dank!

Verfasst: Mittwoch 26. August 2009, 21:00
von cofi
Du musst Funktionen vor deren Verwendung definieren, am Anfang stehen muessen sie deshalb nicht.

Verfasst: Mittwoch 26. August 2009, 21:14
von derdon
Funktionen können (wie alle anderen Objekte in Python auch) nicht *deklariert* werden. cofi hat es nicht direkt gesagt, deswegen betone ich es lieber noch mal. Python ≠ C

Verfasst: Mittwoch 26. August 2009, 21:30
von sma
Dies geht nicht:

Code: Alles auswählen

geht_nicht()

def geht_nicht():
  print "ich werde nicht gefunden"
Dies aber schon:

Code: Alles auswählen

def ich_kann():
  die_andere_funktion_aufrufen()

def die_andere_funktion_aufrufen():
  print "ich werde gefunden"

ich_kann()
Stefan

Verfasst: Mittwoch 26. August 2009, 21:35
von zenker
derdon hat geschrieben:Funktionen können (wie alle anderen Objekte in Python auch) nicht *deklariert* werden.
Ok, das war dann ein Fehler von mir, das als "deklariert" zu bezeichnen.
In Pure Basic gab's halt das Schlüsselwort "declare", was ich dann auf "deklariert" übernommen habe!
Ich find's bloß ganz angenehm, wenn eine Funktion (oder Prozedur in Pure Basic) am Ende des Source steht.
Auf der anderen Seite hatte ich bisher wenig am Hut mit OO-Programmierung. OKay, meine Frage war ein Schnellschuß (nach Entdeckung von "Import") - kommt nicht nochmal vor! :wink:

Verfasst: Mittwoch 26. August 2009, 23:43
von problembär
Bzw. noch näher an C:

Code: Alles auswählen

def main():
  andere_funktion()

def andere_funktion():
  print "ich werde gefunden."

if __name__ == "__main__":
  main()
Die letzten beiden Zeilen siehst Du so oder so ähnlich recht häufig in Python-Skripten. Sie bedeuten: Rufe "main()" auf, wenn das Skript direkt aufgerufen und nicht nur von einem anderen Skript als Modul eingebunden wurde.

Gruß

Verfasst: Donnerstag 27. August 2009, 06:06
von BlackJack
@zenker: Du hast das was Du möchtest schon richtig als "deklariert" beschrieben. Nur gibt's in Python halt (fast) keine Deklarationen. Du kannst Funktionen "nur" definieren.

Das ist nur indirekt eine Frage der Position im Quelltext. Du kannst zur Laufzeit ein Objekt erst verwenden, wenn es vorher definiert wurde. Macht ja auch irgendwie Sinn. Definitionen passieren zur Laufzeit und der Code auf Modulebene wird von oben nach unten ausgeführt.

Verfasst: Dienstag 1. September 2009, 08:44
von J.D.
sma hat geschrieben: Dies aber schon:

Code: Alles auswählen

def ich_kann():
  die_andere_funktion_aufrufen()

def die_andere_funktion_aufrufen():
  print "ich werde gefunden"

ich_kann()
Stefan
Allerdings:

Code: Alles auswählen

def ich_kann():
  die_andere_funktion_aufrufen()

def die_andere_funktion_aufrufen(ich_kann):
  print "ich werde gefunden"

ich_kann()
Da bekomme ich Probleme. Was kann ich in diesem Fall tun?

Nochmal ausführlich: Ich habe zwei Funktionen A und B. Funktion B ist von A abgeleitet und überschreibt ein Attribut. In Klasse A möchte ich nun an einer Stelle Klasse B aufrufen. Egal wie rum ich es auch drehe, es funzt leider nicht.

Kann wer helfen?
Danke & Grüße

Verfasst: Dienstag 1. September 2009, 08:55
von audax
J.D. hat geschrieben:
sma hat geschrieben: Dies aber schon:

Code: Alles auswählen

def ich_kann():
  die_andere_funktion_aufrufen()

def die_andere_funktion_aufrufen():
  print "ich werde gefunden"

ich_kann()
Stefan
Allerdings:

Code: Alles auswählen

def ich_kann():
  die_andere_funktion_aufrufen()

def die_andere_funktion_aufrufen(ich_kann):
  print "ich werde gefunden"

ich_kann()
In dem Fall meckert der Interpreter nur darüber, dass "die_andere_.." ein Argument verlangt, was du ihr bei Aufruf in "ich_kann" nicht übergibst.

Verfasst: Dienstag 1. September 2009, 09:10
von /me
J.D. hat geschrieben:Nochmal ausführlich: Ich habe zwei Funktionen A und B. Funktion B ist von A abgeleitet und überschreibt ein Attribut. In Klasse A möchte ich nun an einer Stelle Klasse B aufrufen. Egal wie rum ich es auch drehe, es funzt leider nicht.
Ich bin mir momentan nicht ganz sicher, ob ich das Problem wirklich verstanden habe, aber so etwas geht ohne Probleme:

Code: Alles auswählen

class foo(object):
	def do_it(self):
		self.make_it()

		
class bar(foo):
	def make_it(self):
		self.x = self.x + 1

		
baz = bar()
baz.x = 1
baz.do_it()

print baz.x
Wenn es das nicht ist, dann solltest du vielleicht mal einen kleinen eingedampften Beispielcode zeigen.

Verfasst: Dienstag 1. September 2009, 09:17
von nemomuk
Ich glaube, du hast noch nicht ganz verstanden, was Klassen, Funktionen, Methoden und Attribute sind. Du kannst nicht von einer Funktion erben oder ableiten, falls du das mit diesem Code errreichen wolltest:

Code: Alles auswählen

def ich_kann():
  die_andere_funktion_aufrufen()

def die_andere_funktion_aufrufen(ich_kann):
  print "ich werde gefunden"

ich_kann()

Verfasst: Dienstag 1. September 2009, 09:47
von cofi
J.D. hat geschrieben:Ich habe zwei Funktionen A und B. Funktion B ist von A abgeleitet und überschreibt ein Attribut. In Klasse A möchte ich nun an einer Stelle Klasse B aufrufen.
Kann es sein, dass du Klassen und Funktionen nicht auseinander halten kannst? Bitte beschreibe einmal ausfuehrlich, _was genau_ du machen willst, nicht die Mittel, um das zu erreichen.

Kurzerklaerung (ohne Designueberlegungen, die aber vllt das wichtigste bei Klassen sind):
Funktionen haben Argumente, stellen mit diesen etwas an und geben das Ergebnis zurueck.
Klassen haben ein ``__init__`` und das uebernimmt Argumente und gibt eine Instanz/Exemplar der Klasse zurueck, dass dann eigenstaendig "lebt". Von Klassen kann man erben, d.h. alle Namen der Basisklasse sind in der erbenden Klasse vorhanden (bis sie evtl ueberschrieben werden).

Verfasst: Dienstag 1. September 2009, 09:49
von J.D.
Erst mal ein Dankeschön für die Hilfe von euch.

Anscheinend waren meine Finger wieder schneller als die Augen.
Der Code von /me trifft es schon relativ genau. So sieht das bei mir aus:

Code: Alles auswählen


class foo(object):
    """Validierung mit Hilfe von Validatorklasse. bar() ist z.B. eine Validatorklasse"""
    value1 = validator.(ValidatorKlasse1())
    value2 = validator.(ValidatorKlasse2())
    value3 = validator.(bar()) # Hier tritt das Problem auf, die Subklasse kann nicht aufgerufen werden, weil Python sie angeblich nicht kennt.

class bar(foo):
    """ von foo abgeleitet. überschreibt value3 um endlose rekursion zu vermeiden. """
    value3 = None
So sieht das ungefähr aus. Eventuell gibt es auch eine elegantere Möglichkeit (z.B. mit echter Rekursion) aber mich interessiert jetzt auch mal speziell dieses Problem. Eure Beispiele oben funktionieren, das glaube ich. Aber anscheinend sieht die Sache anders aus, sobald Vererbung ins Spiel kommt. Bin offen für Vorschläge.

Verfasst: Dienstag 1. September 2009, 09:55
von cofi
Das Problem taucht aber nicht erst in ``bar`` auf, sondern schon in ``foo``, wenn du ``bar`` wirklich in ``foo`` brauchst, solltest du das dem Aufruf von ``foo`` als Argument uebergeben.

Ist das eigentlich Absicht, dass das alles Klassenvariablen sind?

Code: Alles auswählen

class foo(object):
    def __init__(ValidatorKlasse1, ValidatorKlasse2, ValidatorKlasse3):
    value1 = validator.(ValidatorKlasse1())
    value2 = validator.(ValidatorKlasse2())
    value3 = validator.(ValidatorKlasse3())

Verfasst: Dienstag 1. September 2009, 09:57
von nemomuk
Ich verstehe nicht ganz, was du da wirklich machen willst, aber wo kommt da jetzt "validator" her?

@cofi: was soll dein Beispiel da jetzt bewirken - so wird das bestimmt nicht funktionieren? Vllt. verstehe ich da auch was falsch...^^

Verfasst: Dienstag 1. September 2009, 10:04
von cofi
SchneiderWeisse hat geschrieben:@cofi: was soll dein Beispiel da jetzt bewirken - so wird das bestimmt nicht funktionieren?
Erm ja ... danke. Ich bin noch vor dem ersten Kaffee.

Code: Alles auswählen

class foo(object):
    def __init__(ValidatorKlasse1, ValidatorKlasse2, ValidatorKlasse3):
        value1 = ValidatorKlasse1()
        value2 = ValidatorKlasse2()
        value3 = ValidatorKlasse3()
@J.D.: Kanns sein, dass ``validator`` eine Funktion sein soll?

Verfasst: Dienstag 1. September 2009, 13:51
von J.D.
Das mit den Validatoren war nur beispielhaft (weil mein Code zur Validierung von Objekten dienen soll) und verwirrt hier vielleicht mehr, als dass es hilft. Zudem habe ich mich offenbar schon wieder vertippt (Schande über mich): Der Punkt bei validators ist natürlich Unsinn.

Ich möchte nicht ausschließen, dass sich der Code noch stark optimieren lässt, da ich selbst blutiger Anfänger in Python bin (komme aus der Java-Ecke).

In Java wäre mein Beispiel kein Problem, da Klassen zumeist in eigenen Dateien zu finden sind und nicht in Modulen zusammengefasst werden, in denen dann die Reihenfolge offenbar schon eine Rolle spielt (zumindest wenn man mit Vererbung arbeitet.

Das Problem an meinem Beispiel ist einfach: Klasse B muss Klasse A kennen (da B Kind von A ist), aber Klasse A muss auch B kennen (da B in A verwendet wird). Meiner Meinung nach geht das nicht, falls nicht eine der beiden Klassen explizit der anderen übergeben wird (was in meinem Beispiel nicht der Fall ist).

Das mit den Validatoren ist eher nebensächlich, sorry dafür. Aber ja: validators() ist eine Funktion, der ich eine Klasse übergebe.

Verfasst: Dienstag 1. September 2009, 22:12
von BlackJack
@J.D.: Kannst Du das eventuell mit Sinn füllen? Warum willst Du so eine komische Beziehung haben? Ich fände das auch unter Java eigenartig. Kannst Du keine Gemeinsamkeiten in eine dritte Klasse auslagern?

Verfasst: Mittwoch 2. September 2009, 00:03
von Defnull

Code: Alles auswählen

>>> class A(object):
...   def __init__(self):
...     self.b = B()
... 
>>> class B(A):
...   def __init__(self):
...     pass
... 
>>> a = A()
>>> b = B()
>>> a
<__main__.A object at 0xb7e5b84c>
>>> a.b
<__main__.B object at 0xb7e5b62c>
B erbt von A und A verwendet B. Und wo ist jetzt das Problem?

Verfasst: Mittwoch 2. September 2009, 06:26
von BlackJack
@Defnull: Zumindest den Beispielen nach soll `B` anscheinend bei `A` schon bei der Klassendefinition von `A` als bzw. für ein Klassenattribut verwendet werden.