Funktionale Programmierung mit Haskell



  • Diese Aussage zeigt mir, dass du noch nie richtig mit Lisp programmiert hast.

    Was heißt schon Lisp programmiert. Lisp stellt ja mehr eine Sprachfamilie dar.
    Ich hab mich mehrere Jahre mit dylan [1] beschäftigt. Allerdings kann ich dir
    auch kurz ziegen, was ich meine.

    (setf *x* 42.1)
    

    Hier definier ich eine Variable und weise dieser einen Wert zu. Das Konzept einer
    Variable existiert in der funktionalen Programmierung allerdings nicht. Daher
    meine Aussage, dass Lisp funktionale Elemente enthält.

    [1] http://de.wikipedia.org/wiki/Dylan_(Programmiersprache)

    --- edit ---
    Ok, auf deinen Edit hin kann ich dir nur voll zustimmen 🙂



  • frosch03 schrieb:

    Allerdings kann ich dir
    auch kurz ziegen, was ich meine.

    (setf *x* 42.1)
    

    Hier definier ich eine Variable und weise dieser einen Wert zu. Das Konzept einer
    Variable existiert in der funktionalen Programmierung allerdings nicht. Daher
    eine Aussage, dass Lisp funktionale Elemente enthält.

    Ich habe meinem Beitrag oben noch zwei Absätze hinzugefügt, wahrscheinlich gerade als du deine Antwort geschrieben hast.

    Ich möchte nicht sagen, dass Lisp im reinen Sinne "funktional" ist; wobei die Definition, wo "funktional" beginnt und wo es aufhört, im Grunde etwas schwammig ist. Man könnte auch sagen, dass eine Sprache, die Funktionen als first-class objects hat, "funktional" ist. Und genau das ist eine Eigenschaft, die Lisp von Sprachen wie Java, C# und VB.net unterscheidet.

    edit: Ich hätte meinen Beitrag nicht mehr so spät editieren sollen. Sorry für die Verwirrung. 🙂



  • frosch03 schrieb:

    (setf *x* 42.1)
    

    Hier definier ich eine Variable und weise dieser einen Wert zu. Das Konzept einer
    Variable existiert in der funktionalen Programmierung allerdings nicht.

    In einer bestimmten Auffassung des Begriffes. Allgemeingültig ist das nicht, oder willst du SML den Status einer funktionalen Programmiersprache absprechen? Wenn "funktional" automatisch "streng ohne Zuweisung" heißen würde, bräuchte man den Begriff "rein-funktional" nicht.



  • frosch03 schrieb:

    Dadurch das Haskell streng und statisch getypt
    ist, sowie ein sehr mächtiges Typensystem besitzt ist es unmöglich einen
    Typfehler zur Laufzeit zu bekommen.

    Die von Dir aufgezählten Argumente zu Gunsten von FP mögen alle richtig sein, nur hat FP den großen Nachteil, daß sie der menschlichen Intuition von "Algorithmus" zuwiderläuft, denn das natürliche algorithmische Denken des Menschen ist prozedural. Niemand schreibt von Natur aus seinen Einkaufszettel als funktionales Programm. Versteht man "Programmieren" als abstrakte, mathematische Tätigkeit, dann ist FP sicherlich ein überlegenes Konzept.

    Dennoch kann man alles, was man mit FP programmieren kann, prinzipiell auch prozedural oder mit OOP programmieren, dafür sorgt die Church-Turing-Äquivalenz.



  • u_ser-l schrieb:

    Niemand schreibt von Natur aus seinen Einkaufszettel als funktionales Programm.

    Wie würde denn das aussehen?

    Imperativ wäre das ja so:

    1. A
    2. B
    3. C
    


  • u_ser-l schrieb:

    Die von Dir aufgezählten Argumente zu Gunsten von FP mögen alle richtig sein, nur hat FP den großen Nachteil, daß sie der menschlichen Intuition von "Algorithmus" zuwiderläuft, denn das natürliche algorithmische Denken des Menschen ist prozedural.

    Meinst du nicht, dass die vielen FP-Enthusiasten es als natürlicher empfinden, funktional zu programmieren? Oder sind das alles Masochisten? Oder keine Menschen?

    Niemand schreibt von Natur aus seinen Einkaufszettel als funktionales Programm.

    Ein Einkaufszettel ist aber auch nicht imperativ, da er keine feste Reihenfolge vorschreibt. Man ordnet das dynamisch um, idealerweise so, dass die Laufwege möglichst kurz sind. :p

    Dennoch kann man alles, was man mit FP programmieren kann, prinzipiell auch prozedural oder mit OOP programmieren, dafür sorgt die Church-Turing-Äquivalenz.

    Ist das ein ernstgemeintes Argument? 😕 "Prinzipiell" kann man auch direkt in Maschinencode programmieren.



  • Taktiker schrieb:

    Wie würde denn das aussehen?

    eben. In Haskell müßte man Monaden bemühen, um einen Einkaufszettel "1. .. 2. .. 3. .." zu implementieren, also eine Nachbildung sequentiellen Verhaltens durch kategorientheoretische Tricks - nicht gerade ein Formalismus, der natürlichem algorithmischem Denken angepaßt wäre. Probleme aus der realen Welt lassen sich oft am einfachsten durch Zustände, Übergänge und sequentielle Abläufe modellieren, also genau das, was mit FP kompliziert ist und mit prozeduraler oder OO-Prog'rung einfach.

    Bei zukünftig weiter zunehmender Parallelisierung der Hardware auf allen Ebenen könnte die große Zeit von Sprachen wie Haskell in der Tat bevorstehen.



  • u_ser-l schrieb:

    Taktiker schrieb:

    Wie würde denn das aussehen?

    eben. In Haskell müßte man Monaden bemühen, um einen Einkaufszettel "1. .. 2. .. 3. .." zu implementieren, also eine Nachbildung sequentiellen Verhaltens durch kategorientheoretische Tricks - nicht gerade ein Formalismus, der natürlichem algorithmischem Denken angepaßt wäre.

    Du hast interessante Vorstellungen über Haskell. Eine Liste ist in Haskell einfach das: Eine Liste.
    In deinem Beispiel: [1, 2, 3].

    Klar kann man das kategorientheoretisch interpretieren. Aber muss man das machen, um die Liste [1, 2, 3] zu verstehen? 😕



  • u_ser-l schrieb:

    Die von Dir aufgezählten Argumente zu Gunsten von FP mögen alle richtig sein, nur hat FP den großen Nachteil, daß sie der menschlichen Intuition von "Algorithmus" zuwiderläuft, denn das natürliche algorithmische Denken des Menschen ist prozedural.

    Das ist absoluter Blödsinn. Da ich im Moment in der Lehre tätig bin und Menschen sehe die zum ersten mal programmieren, sehe ich auch, wie schwierig die prozedurale Denkweise zu erlernen ist. Bloß weil du dich an sie gewöhnt hast, ist sie noch lange nicht natürlich.

    u_ser-l schrieb:

    Niemand schreibt von Natur aus seinen Einkaufszettel als funktionales Programm.

    Niemand schreibt seinen Einkaufs-Zettel als prozedurales Programm. Denn die Sachen auf dem Zettel sind ja kein Programmablauf. Der Zettel stellt ja lediglich die Menge der Teile da, die du vor hast zu erwerben. Du wirst selten die Dinge in der Reihenfolge in den Einkaufskorb legen, wie sie auf dem Zettel stehen. Von daher: Blödes Beispiel.

    Der Mensch denkt rekursiv und nicht prozedural. Beispiel: Das Bilden der Summe von Zahlen in einer Liste. So würde ein Mensch vorgehen:

    Die Summe von zwei Zahlen kann ich. Die Summe von drei Zahlen ist die Summe von zwei Zahlen addiert mit einer weiteren Zahl... (spätestens beim nächsten mal merkt man) Die Summe von n Zahlen ist die Summe von (n-1) Zahlen plus eine weitere Zahl.

    Und genau diese rekursive Denkweise setzen die Funktionalen Programmiersprachen um.

    u_ser-l schrieb:

    Dennoch kann man alles, was man mit FP programmieren kann, prinzipiell auch prozedural oder mit OOP programmieren, dafür sorgt die Church-Turing-Äquivalenz.

    Ja richtig. Die frage ist nur, wie viel schwerer und umständlicher es ist.



  • ProgChild schrieb:

    Da ich im Moment in der Lehre tätig bin und Menschen sehe die zum ersten mal programmieren, sehe ich auch, wie schwierig die prozedurale Denkweise zu erlernen ist. Bloß weil du dich an sie gewöhnt hast, ist sie noch lange nicht natürlich.

    Mein Lieblingsbeispiel ist ja immer noch das folgende. Jemand wundert sich, dass dieses Programm nicht funktioniert:

    int x;
    int y = x*x;
    cout << "Bitte x eingeben:";
    cin >> x;
    cout << "Das Quadrat von x ist: " << y << endl;
    

    Das Beispiel ist fiktiv, aber die Situation an sich hab ich schon zu oft erlebt, um es für einen Zufall zu halten. Denkt der Mensch am Ende deklarativ? (Oder hatten diejenigen Vorerfahrung mit Tabellenkalkulationen?;) )



  • Bashar schrieb:

    Das Beispiel ist fiktiv, aber die Situation an sich hab ich schon zu oft erlebt, um es für einen Zufall zu halten. Denkt der Mensch am Ende deklarativ? (Oder hatten diejenigen Vorerfahrung mit Tabellenkalkulationen?;) )

    Keine Ahnung. Aber die Situation, wie du sie beschreibst, hatte ich auch schon so ähnlich. Da sieht man wirklich, welche Dinge man alle als selbstverständlich annimmt, die es aber nicht sind.



  • ProgChild schrieb:

    u_ser-l schrieb:

    Dennoch kann man alles, was man mit FP programmieren kann, prinzipiell auch prozedural oder mit OOP programmieren, dafür sorgt die Church-Turing-Äquivalenz.

    Ja richtig. Die frage ist nur, wie viel schwerer und umständlicher es ist.

    Wenn du uns jetzt noch verrätst, warum es hier kein Unterforum zu einer funktionalen Sprache gibt..



  • ProgChild schrieb:

    Da ich im Moment in der Lehre tätig bin und Menschen sehe die zum ersten mal programmieren, sehe ich auch, wie schwierig die prozedurale Denkweise zu erlernen ist. Bloß weil du dich an sie gewöhnt hast, ist sie noch lange nicht natürlich.

    So ein Käse. Die prozedurale Denkweise muß nicht erlernt werden, sie ist dem Menschen von Natur aus mitgegeben, weil evolutionär entstanden.

    Der Urmensch mußte auf der Jagd etliche "wenn ... dann"-Entscheidungen treffen und prozedural Jagdstrategien planen, um zu überleben. Das hat die natürliche Denkweise des Menschen maßgeblich geprägt.

    Das Gehirn ist geschaffen, um zu überleben, und nicht, um elegante Formalismen wie FP zu denken. Und zum Überleben muß es seit jeher prozedural denken. Niemand erbeutet ein Mammut mit funktionalen Überlegungen und Monadentheorie.

    ProgChild schrieb:

    Niemand schreibt seinen Einkaufs-Zettel als prozedurales Programm.

    So ein Käse. Jeder macht das. "1. Supermarkt Broccoli. 2. Bücherei 3. Blabla ..." Ich denke, das Prinzip ist klar (!)

    ProgChild schrieb:

    Der Mensch denkt rekursiv und nicht prozedural.

    wenn FP der natürlichen Denkweise entspräche, hätte sie sich längst auf breiter Basis durchgesetzt, wahrscheinlich wäre FP vor Fortran und Cobol erfunden worden, und die Turingmaschine wäre für uns dann ebenso exotisch wie für die prozedural denkende Mehrheit der Lambda-Kalkül 🙂



  • Bashar schrieb:

    int x;
    int y = x*x;
    cout << "Bitte x eingeben:";
    cin >> x;
    cout << "Das Quadrat von x ist: " << y << endl;
    

    Das Beispiel ist fiktiv, aber die Situation an sich hab ich schon zu oft erlebt, um es für einen Zufall zu halten. Denkt der Mensch am Ende deklarativ?

    der mensch denkt, du hast mit der zeile int y = x*x; eine funktion definiert, in der er nur x einsetzen muss. hat er im matheunterricht so gelernt. er rechnet ja nicht damit, dass im computer immer alles von oben nach unten ausgeführt wird und die reihenfolge eine entscheidende rolle spielt.



  • u_ser-l schrieb:

    wahrscheinlich wäre FP vor Fortran und Cobol erfunden worden

    Der Lambda-Kalkül ist meines Wissens nach deutlich älter als Fortran und Cobol.



  • Zoom schrieb:

    Bashar schrieb:

    int x;
    int y = x*x;
    cout << "Bitte x eingeben:";
    cin >> x;
    cout << "Das Quadrat von x ist: " << y << endl;
    

    Das Beispiel ist fiktiv, aber die Situation an sich hab ich schon zu oft erlebt, um es für einen Zufall zu halten. Denkt der Mensch am Ende deklarativ?

    der mensch denkt, du hast mit der zeile int y = x*x; eine funktion definiert, in der er nur x einsetzen muss. hat er im matheunterricht so gelernt. er rechnet ja nicht damit, dass im computer immer alles von oben nach unten ausgeführt wird und die reihenfolge eine entscheidende rolle spielt.

    Und genau das ist das schöne an Haskell: Man kann etwas hinschreiben und es funktioniert genau so, wie man es aus dem Mathe-Unterricht erwartet. 🙂



  • Christoph schrieb:

    Und genau das ist das schöne an Haskell: Man kann etwas hinschreiben und es funktioniert genau so, wie man es aus dem Mathe-Unterricht erwartet.

    kommt drauf an, was man will. entweder computer verstehen, oder computer benutzen.



  • u_ser-l schrieb:

    So ein Käse. Die prozedurale Denkweise muß nicht erlernt werden, sie ist dem Menschen von Natur aus mitgegeben, weil evolutionär entstanden.

    Der Urmensch mußte auf der Jagd etliche "wenn ... dann"-Entscheidungen treffen und prozedural Jagdstrategien planen, um zu überleben. Das hat die natürliche Denkweise des Menschen maßgeblich geprägt.

    Nein. Du beschreibst Dinge die nacheinander ablaufen und behauptest, weil in der Welt die Dinge immer nacheinander ablaufen, so muss der Mensch auch genau so denken. Warum bitte sollte das so sein?

    Sagen wir ein Mensch steht vor einer Mammut-Herde und will ein Mammut erlegen. Folgendes denkt er bestimmt nicht:

    1. Trenne ein Mammut von der Herde
    2. Erlege das Mammut

    Er denkt folgendes: Ein einzelnes Mammut haben wir schonmal erlegt. Wir müssen also irgendwie ein Mammut von der Herde trennen.

    Du merkst die natürliche Denkweise ist genau anders herum.

    u_ser-l schrieb:

    So ein Käse. Jeder macht das. "1. Supermarkt Broccoli. 2. Bücherei 3. Blabla ..." Ich denke, das Prinzip ist klar (!)

    Eine Liste in Haskell bleibt ja auch weiterhin eine Liste. Die Sache ist doch wie du mit der Liste arbeitest.

    u_ser-l schrieb:

    wenn FP der natürlichen Denkweise entspräche, hätte sie sich längst auf breiter Basis durchgesetzt, wahrscheinlich wäre FP vor Fortran und Cobol erfunden worden, und die Turingmaschine wäre für uns dann ebenso exotisch wie für die prozedural denkende Mehrheit der Lambda-Kalkül 🙂

    Badestrand schrieb:

    Wenn du uns jetzt noch verrätst, warum es hier kein Unterforum zu einer funktionalen Sprache gibt..

    Diese Frage lässt sich ganz einfach beantworten. Die Art wie die prozeduralen Programmiersprachen funktionieren ist die Art, wie die Computer funktionieren. Man hat mit Assembler angefangen und der Rest ist halt evolution. Es ist Tatsache, dass viele Dinge in C so sind, wie sie sind, weil man sie recht einfach in Maschienen-Befehle umsetzen kann.

    Das hat aber rein garnichts zu tun, mit der Art und Weise, wie Menschen denken.



  • Zoom schrieb:

    Christoph schrieb:

    Und genau das ist das schöne an Haskell: Man kann etwas hinschreiben und es funktioniert genau so, wie man es aus dem Mathe-Unterricht erwartet.

    kommt drauf an, was man will. entweder computer verstehen, oder computer benutzen.

    Warum schließt du das aus? Man kann mit Haskell vielleicht nicht verstehen, wie ein Computer intern funktioniert. Das kann man mit C++ aber auch nicht. Wer das möchte, muss Assembler oder sogar Verilog (und Logik-Schaltungen) lernen.

    Aber nachdem man die "Assembler-Phase" hinter sich gebracht und verstanden hat, wie das funktioniert, wird IMHO so gut wie niemand mit Assembler ein richtiges Programm schreiben.

    Es interessiert in den meisten Fällen einfach nicht mehr, zu welchen Maschinencodes eine for-Schleife nun genau umgewandelt wird. Genauso kann man sagen, dass es in Haskell einfach nicht mehr interessant ist, in welcher Reihenfolge nun bestimmte Terme ausgewertet werden, solange das Ergebnis stimmt.

    Warum sollte man Zeit investieren in etwas, was der Computer selbst herausfinden kann, wie z.B. die Auswertungs-Reihenfolge? [1]

    [1] Von Ausnahmen natürlich abgesehen. Wenn man höchste Performance oder absolut deterministisches Verhalten möchte, führt eben kaum ein Weg an Assembler oder C vorbei.



  • ProgChild schrieb:

    Er denkt folgendes: Ein einzelnes Mammut haben wir schonmal erlegt. Wir müssen also irgendwie ein Mammut von der Herde trennen.

    Genau, und aus dieser Überlegung heraus wird er einen Plan fassen:
    1. Trenne ein Mammut von der Herde
    2. Erlege das Mammut

    Zerlegst Du eigentlich Deinen Alltag in lauter Teilprobleme, die Du dann völlig unabhängig voneinander in beliebiger Reihenfolge abarbeitest? Finde ich spannend. Ich persönlich nehme mir immer grobe Aufgaben vor, überlege mir aus welche Teilaufgaben zu bearbeiten sind und erledige dann eine nach der anderen.


Anmelden zum Antworten