You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

095 Sockets.ipynb 5.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. {
  2. "cells": [
  3. {
  4. "cell_type": "markdown",
  5. "metadata": {
  6. "pycharm": {
  7. "name": "#%% md\n"
  8. },
  9. "slideshow": {
  10. "slide_type": "slide"
  11. }
  12. },
  13. "source": [
  14. "# Software Entwicklung \n",
  15. "\n",
  16. "## Kapitel 9: IO\n",
  17. "\n",
  18. "### 9.5 Sockets\n",
  19. "\n",
  20. "So wie der Begriff *Datei* eine griffige Bezeichnung für eine Ansammlung dauerhaft gespeicherter Bytes ist, wird\n",
  21. "mit *Socket* ein Endpunkt für Netzwerkverbindungen bezeichnet.\n",
  22. "Die Kommunikation findet zwischen zwei Sockets *bidirektional* statt, d.h. über jedes Socket können Bytes versendet\n",
  23. "und empfangen werden."
  24. ]
  25. },
  26. {
  27. "cell_type": "markdown",
  28. "metadata": {
  29. "pycharm": {
  30. "name": "#%% md\n"
  31. },
  32. "slideshow": {
  33. "slide_type": "slide"
  34. }
  35. },
  36. "source": [
  37. "Sockets stehen in Python durch das Standard-Modul <code>sockets</code> zur Verfügung. Wie schon Dateien belegen auch\n",
  38. "Sockets Betriebssystem-Ressourcen, so dass die Verwendung eines Kontextmanagers ratsam ist."
  39. ]
  40. },
  41. {
  42. "cell_type": "code",
  43. "execution_count": null,
  44. "metadata": {
  45. "pycharm": {
  46. "name": "#%%\n"
  47. }
  48. },
  49. "outputs": [],
  50. "source": [
  51. "import socket\n",
  52. "\n",
  53. "with socket.socket() as s:\n",
  54. " pass"
  55. ]
  56. },
  57. {
  58. "cell_type": "markdown",
  59. "metadata": {
  60. "slideshow": {
  61. "slide_type": "slide"
  62. }
  63. },
  64. "source": [
  65. "#### 9.5.1 Client-Server-Modell\n",
  66. "\n",
  67. "Zwar sind bei einer bestehenden Kommunikationsverbindung zwischen zwei Sockets die beiden Sockets prinzipiell\n",
  68. "gleichberechtigt, jedoch gibt es Unterschiede beim Kommunikationsaufbau.\n",
  69. "\n",
  70. "* Ein *Server-Socket* verhält sich passiv, d.h. es wartet auf eingehende Verbindungsanfragen. Trifft eine solche\n",
  71. " ein, generiert es ein weiteres Socket, das fest dieser Verbindung zugeordnet wird und über das die\n",
  72. " anschließende Kommunikation abgewickelt wird. Das *Server-Socket* selbst verbleibt im Wartezustand.\n",
  73. "\n",
  74. "* Ein *Client-Socket* übernimmt den aktiven Part. Es kontaktiert das *Server-Socket* und geht selbst die\n",
  75. " Verbindung mit dem daraufhin bereitgestellten Socket ein."
  76. ]
  77. },
  78. {
  79. "cell_type": "markdown",
  80. "metadata": {
  81. "slideshow": {
  82. "slide_type": "slide"
  83. }
  84. },
  85. "source": [
  86. "Die Verbindungsaufnahme folgt einem vorgegebenen Ablauf aus definierten Schritten (auch: *Kommunikationsprimitive*).\n",
  87. "\n",
  88. "![Verbindungsaufnahme mit Kommunkationsprimitiven](../img/Primitive.png \"Kommunikationsprimitive\")"
  89. ]
  90. },
  91. {
  92. "cell_type": "markdown",
  93. "metadata": {
  94. "pycharm": {
  95. "name": "#%% md\n"
  96. },
  97. "slideshow": {
  98. "slide_type": "slide"
  99. }
  100. },
  101. "source": [
  102. "#### 9.5.2 Server-Socket\n",
  103. "\n",
  104. "Ein *Server-Socket* ist eindeutig festgelegt durch das Protokoll (z.B. TCP oder UDP), den Rechner\n",
  105. "(z.B. durch dessen IP-Adresse) und eine Portnummer. Letztere dient dazu, dass mehrere Server-Sockets gleichzeitig\n",
  106. "auf einem Rechner existieren können, ohne sich gegenseitig zu beeinflussen.\n",
  107. "\n",
  108. "So gibt es z.B. auf dem Server <code>medinf.efi.th-nuernberg.de</code> mehrere Server-Sockets,\n",
  109. "die permanent angesprochen werden können:\n",
  110. "\n",
  111. "* Port 22: Nimmt Secure-Shell-Anfragen entgegen\n",
  112. "* Port 80: Nimmt HTTP-Anfragen entgegen\n",
  113. "* Port 443: Nimmt HTTPS-Anfragen entgegen"
  114. ]
  115. },
  116. {
  117. "cell_type": "markdown",
  118. "metadata": {
  119. "pycharm": {
  120. "name": "#%% md\n"
  121. },
  122. "slideshow": {
  123. "slide_type": "slide"
  124. }
  125. },
  126. "source": [
  127. "Das *Server-Socket* wird mit <code>bind</code> an einen bestimmten Port und eine\n",
  128. "IP-Adresse (ggf. hat ein Rechner mehrere IP-Adressen) gebunden. Das Binden schlägt fehl, wenn die\n",
  129. "Kombination bereits von einem anderen Socket belegt ist.\n",
  130. "\n",
  131. "Anschließend beginnt das *Server-Socket* mit <code>listen</code> zu lauschen. Das Primitiv <code>accept</code>\n",
  132. "blockiert den Server, bis ein *Client-Socket* Verbindung aufnimmt. Damit entsteht ein weiteres Socket mit\n",
  133. "einer vom System vergebenen Portnummer, das dediziert nur für die Verbindung mit diesem Client verwendet wird.\n",
  134. "\n",
  135. "Über dieses neue Socket folgen danach Primitive zum Senden\n",
  136. "<code>send</code> und Empfangen <code>receive</code> von Nachrichten. Anzahl und Abfolge sind\n",
  137. "für Sockets nicht festgelegt, müssen aber zwischen Client und Server abgestimmt sein. So legt z.B.\n",
  138. "das auf Sockets aufsetzende Protokoll HTTP fest, dass zunächst eine Nachricht vom Client\n",
  139. "an den Server (*HTTP-Request*) und anschließend eine Nachricht vom Server an den Client\n",
  140. "(*HTTP-Response*) gesendet wird. Während in der ersten Version von HTTP danach die\n",
  141. "Verbindung wieder abgebaut wurde, ist es aktuellen Versionen möglich, dieses Abfolge\n",
  142. "wiederholt über eine bestehende Verbindung abzuwickeln."
  143. ]
  144. },
  145. {
  146. "cell_type": "markdown",
  147. "metadata": {
  148. "pycharm": {
  149. "name": "#%% md\n"
  150. },
  151. "slideshow": {
  152. "slide_type": "slide"
  153. }
  154. },
  155. "source": [
  156. "Eine etablierte Verbindung ist mit <code>close</code> zu schließen. Bei Verwendung eines\n",
  157. "Kontextmanagers erfolgt dies automatisch.\n",
  158. "\n",
  159. "Auch das *Server-Socket* muss geschlossen werden, um das Binden an einen Port zu beenden\n",
  160. "und den Port freizugeben."
  161. ]
  162. }
  163. ],
  164. "metadata": {
  165. "celltoolbar": "Slideshow",
  166. "kernelspec": {
  167. "display_name": "Python 3 (ipykernel)",
  168. "language": "python",
  169. "name": "python3"
  170. },
  171. "language_info": {
  172. "codemirror_mode": {
  173. "name": "ipython",
  174. "version": 3
  175. },
  176. "file_extension": ".py",
  177. "mimetype": "text/x-python",
  178. "name": "python",
  179. "nbconvert_exporter": "python",
  180. "pygments_lexer": "ipython3",
  181. "version": "3.9.9"
  182. }
  183. },
  184. "nbformat": 4,
  185. "nbformat_minor": 1
  186. }