Prolog.tex 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. %!TEX root = Programmierparadigmen.tex
  2. \chapter{Prolog}
  3. \index{Prolog|(}
  4. Prolog ist eine Programmiersprache, die das logische Programmierparadigma
  5. befolgt.
  6. Eine interaktive Prolog-Sitzung startet man mit \texttt{swipl}.
  7. In Prolog definiert man Terme.
  8. \section{Erste Schritte}
  9. \subsection{Hello World}
  10. Speichere folgenden Quelltext als \texttt{hello-world.pl}:
  11. \inputminted[linenos, numbersep=5pt, tabsize=4, frame=lines, label=hello-world.hs]{prolog}{scripts/prolog/hello-world.pl}
  12. Kompiliere ihn mit \texttt{gplc hello-world.pl}. Es wird eine
  13. ausführbare Datei erzeugt.
  14. \section{Syntax}
  15. In Prolog gibt es Prädikate, die Werte haben. Prädikate werden immer klein geschrieben.
  16. So kann das Prädikat \texttt{farbe} mit den Werten \texttt{rot}, \texttt{gruen},
  17. \texttt{blau}, \texttt{gelb} - welche auch immer klein geschrieben werden - wie
  18. folgt definiert werden:
  19. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/praedikat-farbe.pl}
  20. \begin{itemize}
  21. \item Terme werden durch \texttt{,} mit einem logischem \textbf{und} verknüpft.
  22. \item Ungleichheit wird durch \texttt{\=} ausgedrückt.
  23. \end{itemize}
  24. So ist folgendes Prädikat \texttt{nachbar(X, Y)} genau dann wahr, wenn $X$
  25. und $Y$ Farben sind und $X \neq Y$ gilt:
  26. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/simple-term.pl}
  27. \subsection{= und ==}\xindex{= (Prolog)}\xindex{== (Prolog)}
  28. In Prolog entspricht \texttt{=} dem Prädikat \texttt{=/2}. Das Prädikat \texttt{<a> = <b>} wird
  29. erfüllt, wenn die beiden Terme \texttt{<a>} und \texttt{<b>} unifiziert werden
  30. können.
  31. Das Prädikat \texttt{<a> == <b>} ist im Gegensatz dazu jedoch nur erfüllt, wenn
  32. die beiden Terme bereits identisch sind.
  33. \begin{beispiel}[= und ==]
  34. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/equal.pl}
  35. \end{beispiel}
  36. Weitere Informationen: \url{http://stackoverflow.com/a/8220315/562769}
  37. \subsection{Arithmetik}
  38. Die Auswertung artihmetischer Ausdrücke muss in Prolog explizit durch \texttt{is}
  39. durchgeführt werden:
  40. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/arithmetik.pl}
  41. Dabei müssen alle Variablen, die im Term rechts von \texttt{is} vorkommen,
  42. istanziiert sein:
  43. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/arithmetik-fail.pl}
  44. Arithmetische Ausdrücke können mit \texttt{=:= , =\textbackslash= , < , <= , > , >=}
  45. verglichen werden.
  46. \begin{beispiel}[Arithmetik in Prolog\footnotemark]
  47. \begin{bspenum}
  48. \item \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/even.pl}\xindex{even (Prolog)@\texttt{even} (Prolog)}
  49. \item \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/fibonacci2.pl}\xindex{fib}
  50. \end{bspenum}
  51. \end{beispiel}
  52. \footnotetext{WS 2013 / 2014, Folie 237f}
  53. \subsection{Listen}
  54. Das Atom \texttt{[]} ist die leere Liste.
  55. Mit der Syntax \texttt{[K|R]} wird eine Liste in den Listekopf \texttt{K} und
  56. den Rest der Liste \texttt{R} gesplitet:
  57. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/liste-basic.pl}
  58. Einen Test \texttt{member(X, Liste)}, der \texttt{True} zurückgibt wenn \texttt{X}
  59. in \texttt{Liste} vorkommt, realisiert man wie folgt:
  60. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/liste-member.pl}\xindex{member}
  61. Eine Regel \texttt{append(A, B, C)}, die die Listen \texttt{A} und \texttt{B}
  62. zusammenfügt und als Liste \texttt{C} speichert, kann
  63. wie folgt erstellt werden:
  64. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/liste-append.pl}
  65. Die erste Regel besagt, dass das Hinzufügen der leeren Liste zu einer Liste
  66. \texttt{L} immer noch die Liste \texttt{L} ist.
  67. Die zweite Regel besagt: Wenn die Liste \texttt{R} und \texttt{L} die Liste \texttt{T}
  68. ergeben, dann ergibt die Liste, deren Kopf \texttt{X} ist und deren Rumpf \texttt{R}
  69. ist zusammen mit der Liste \texttt{L} die Liste mit dem Kopf \texttt{X} und dem
  70. Rumpf \texttt{T}.
  71. \xindex{split}Übergibt man \texttt{append(X,Y,[1,2,3,4,5])}, so werden durch Reerfüllung alle
  72. Möglichkeiten durchgegangen, wie man die Liste \texttt{[1,2,3,4,5]} splitten kann.
  73. Die Länge einer Liste \texttt{L} kann durch folgendes Prädikat ermittelt werden:\xindex{length(?List, ?Int)@\texttt{length(?List, ?Int)}}%
  74. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/list-length.pl}
  75. \underline{Hinweis}: Da es das Prädikat \texttt{length(?List, ?Int)} bereits gibt,
  76. musste dieses Prädikat \texttt{lengthof} genannt werden.
  77. Weitere nützliche Standard-Listenprädikate sind:\xindex{sort(+List, -Sorted)@\texttt{sort(+List, -Sorted)}}
  78. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/standard-list-predicates.pl}
  79. \underline{Hinweis}: \texttt{sort} entfernt Duplikate, \texttt{msort} hingegen nicht.
  80. Eine Liste kann mit \texttt{rev/2}\xindex{rev/2@\texttt{rev/2}} umgedreht werden:
  81. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/reverse-list.pl}
  82. \subsection{Bäume}
  83. Bäume können in Prolog wie folgt erstellt werden:
  84. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/binary-tree-example.pl}
  85. \begin{figure}[htp]
  86. \centering
  87. \input{figures/binary-tree.tex}
  88. \caption{Binärbaum \texttt{T2}}
  89. \label{fig:binary-tree-t2}
  90. \end{figure}
  91. Dabei ist
  92. \begin{itemize}
  93. \item \texttt{T0} der einzelne Knoten \texttt{a},
  94. \item \texttt{T1} der Baum, der \texttt{a} als Wurzel und \texttt{b} und
  95. \texttt{c} als Kinder hat,
  96. \item \texttt{T2} ist in \cref{fig:binary-tree-t2} dargestellt und
  97. \item \texttt{T3} ist der leere Baum.
  98. \end{itemize}
  99. Die folgenden Prädikate stammen von \url{https://sites.google.com/site/prologsite/prolog-problems/4}:
  100. \subsection{Binärbaum-Check}
  101. Das folgende Prädikate \texttt{istree/1} überprüft, ob es sich bei dem Parameter
  102. um einen Binärbaum handelt:
  103. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/istree.pl}
  104. \subsection{Balancierte Binärbaumkonstruktion}
  105. Das folgende Prädikate \texttt{cbal\_tree(n, T)} erstellt einen balancierten
  106. Binärbaum mit \texttt{n} Knoten in \texttt{T}:
  107. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/balancedtreeconstruction.pl}
  108. \section{Beispiele}
  109. \subsection{Humans}
  110. Erstelle folgende Datei:
  111. \inputminted[linenos, numbersep=5pt, tabsize=4, frame=lines, label=human.pro]{prolog}{scripts/prolog/human.pro}
  112. Kompiliere diese mit
  113. \inputminted[numbersep=5pt, tabsize=4]{bash}{scripts/prolog/human.sh}
  114. Dabei wird eine \texttt{a.out} Datei erzeugt, die man wie folgt
  115. nutzen kann:
  116. \inputminted[numbersep=5pt, tabsize=4]{bash}{scripts/prolog/human-2.sh}
  117. \subsection{Splits}
  118. \inputminted[linenos, numbersep=5pt, tabsize=4, frame=lines, label=splits.pl]{prolog}{scripts/prolog/splits.pl}
  119. Dieses skript soll man \texttt{swipl -f test.pl} aufrufen. Dann erhält man:
  120. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/splits.sh}
  121. \subsection{Delete}\xindex{remove}\xindex{delete}%
  122. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/delete.pl}
  123. % \subsection{Zebrarätsel}
  124. % Folgendes Rätsel wurde von \url{https://de.wikipedia.org/w/index.php?title=Zebrar%C3%A4tsel&oldid=126585006}
  125. % entnommen:
  126. % \begin{enumerate}
  127. % \item Es gibt fünf Häuser.
  128. % \item Der Engländer wohnt im roten Haus.
  129. % \item Der Spanier hat einen Hund.
  130. % \item Kaffee wird im grünen Haus getrunken.
  131. % \item Der Ukrainer trinkt Tee.
  132. % \item Das grüne Haus ist direkt rechts vom weißen Haus.
  133. % \item Der Raucher von Altem-Gold-Zigaretten hält Schnecken als Haustiere.
  134. % \item Die Zigaretten der Marke Kools werden im gelben Haus geraucht.
  135. % \item Milch wird im mittleren Haus getrunken.
  136. % \item Der Norweger wohnt im ersten Haus.
  137. % \item Der Mann, der Chesterfields raucht, wohnt neben dem Mann mit dem Fuchs.
  138. % \item Die Marke Kools wird geraucht im Haus neben dem Haus mit dem Pferd.
  139. % \item Der Lucky-Strike-Raucher trinkt am liebsten Orangensaft.
  140. % \item Der Japaner raucht Zigaretten der Marke Parliaments.
  141. % \item Der Norweger wohnt neben dem blauen Haus.
  142. % \end{enumerate}
  143. % Wer trinkt Wasser? Wem gehört das Zebra?
  144. % \inputminted[linenos, numbersep=5pt, tabsize=4, frame=lines, label=zebraraetsel.pro]{prolog}{scripts/prolog/zebraraetsel.pro}
  145. % TODO
  146. \subsection{Zahlen generieren}
  147. Folgendes Skript generiert durch reerfüllung die Zahlen $1, \dots, 10$:
  148. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/zahlen-bis-10.pl}
  149. \subsection{Reguläre ausdrücke}
  150. Folgendes Beispiel stammt aus der Programmierparadigmenklausur vom WS 2013/2014
  151. bei Prof. Dr. Snelting:
  152. \inputminted[numbersep=5pt, tabsize=4]{prolog}{scripts/prolog/regex.pl}
  153. \section{Weitere Informationen}
  154. \begin{itemize}
  155. \item \href{http://wiki.ubuntuusers.de/Prolog}{\path{wiki.ubuntuusers.de/Prolog}}: Hinweise zur Installation von Prolog unter Ubuntu
  156. \item \url{http://www.swi-prolog.org/}
  157. \end{itemize}
  158. \index{Prolog|)}