Zum Inhalt springen

„Model View Controller“ – Versionsunterschied

aus Wikipedia, der freien Enzyklopädie
[ungesichtete Version][gesichtete Version]
Inhalt gelöscht Inhalt hinzugefügt
Quellen und Weblinks überarbeitet (fast alle sind inzwischen offline => Archivlinks)
 
(561 dazwischenliegende Versionen von mehr als 100 Benutzern, die nicht angezeigt werden)
Zeile 1: Zeile 1:
[[Datei:ModelViewControllerDiagram2.svg|mini|Ein Model-View-Controller-Konzept. Eine durchgezogene Linie symbolisiert hier eine direkte [[Assoziation (UML)|Assoziation]], eine gestrichelte eine indirekte Assoziation (zum Beispiel über einen [[Beobachter (Entwurfsmuster)|Beobachter]]).]]
Beim '''Model-View-Controller-Modell''', kurz '''MVC-Modell''', handelt es sich um ein [[Architekturmuster]] zur Trennung eines [[Computerprogramm|Programms]] in die drei Einheiten Datenmodell (engl. ''Model''), Präsentation (engl. ''View'') und Programmsteuerung (engl. ''Controller''). Ziel des Modells ist ein flexibles Programmdesign, um u.a. eine spätere Änderung oder Erweiterung einfach zu halten und die Wiederverwendbarkeit der einzelnen Komponenten zu ermöglichen. Außerdem sorgt das Modell bei großen Anwendungen für eine gewisse Übersicht und Ordnung durch Reduzierung der Komplexität. Gleichzeitig bringt die Trennung auch eine Rollenverteilung mit sich. Fachkräfte können so optimal, ihrer Fähigkeit entsprechend, eingesetzt werden: Ein [[Gestalter]] oder [[GUI]]-Designer erstellt das grafische Erscheinungsbild, der [[Programmierer]] erstellt die nötige Geschäftslogik, [[Datenbank]]experten kümmern sich um die optimale Datenverwaltung usw.


'''{{lang|en|Model View Controller}}''' ('''MVC''', {{enS}} für ''Modell-Ansicht-Steuerung'') ist ein [[Entwurfsmuster]] zur Unterteilung einer [[Software]] in die drei Komponenten ''Datenmodell'' (englisch ''{{lang|en|model}}''), ''Ansicht'' (englisch ''{{lang|en|view}}'') und ''Programmsteuerung'' (englisch ''{{lang|en|controller}}''). Das Muster kann sowohl als [[Architekturmuster]] als auch als Entwurfsmuster eingesetzt werden.<ref>{{Internetquelle |autor=Kamal Wickramanayake |url=https://www.swview.org/blog/mvc-design-pattern-or-architectural-pattern |titel=Is MVC a design pattern or an architectural pattern? |werk=Software View |datum=2010-07-17 |sprache=en |abruf=2016-12-16}}</ref> Ziel des Musters ist ein flexibler Programmentwurf, der eine spätere Änderung oder Erweiterung erleichtert und eine Wiederverwendbarkeit der einzelnen Komponenten ermöglicht. Es ist dann zum Beispiel möglich, eine Anwendung zu schreiben, die dasselbe Modell nutzt und es dann für Windows, Mac, Linux oder für das Internet zugänglich macht. Die Umsetzungen nutzen dasselbe Modell, nur Controller und View müssen dabei jeweils neu implementiert werden.
Eine Variante des MVC-Modells ist das [[#MVC_2|MVC2-Modell]]. Es ist eine für Webanwendungen optimierte zustandslose Variante des objektorientierten MVC-Modells.


Das MVC-Konzept wurde 1979 zunächst für Benutzeroberflächen in [[Smalltalk (Programmiersprache)|Smalltalk]] durch [[Trygve Reenskaug]] beschrieben (Seeheim-Modell), der damals an Smalltalk im [[Xerox PARC]] arbeitete. Es gilt mittlerweile aber als De-facto-Standard für den Grobentwurf vieler komplexer Softwaresysteme, teils mit Differenzierungen und oftmals mehreren jeweils nach dem MVC-Muster aufgeteilten Modulen.
== Schichten ==
=== Model ===
Das Datenmodell enthält die dauerhaften (persistenten) Daten der Anwendung. Das Model hat lesenden Zugriff auf diverse Backend-Speicher wie zum Beispiel Datenbanken. Das Model kennt weder die View noch den Controller, es weiß also gar nicht, wie, ob und wie oft es dargestellt und verändert wird. Änderungen im Model werden allerdings über einen Update-Mechanismus bekannt gegeben, indem ein [[Ereignis_(Programmierung)|Event]] ausgelöst wird. Dazu muß sich zumindest die View als abhängiges Objekt am Model registrieren, um über Datenänderungen informiert zu werden (Vergleiche [[Beobachter (Entwurfsmuster)]]).


== Klassisches Architekturmuster ==
=== View ===
Die drei Komponenten hängen je nach Umsetzung unterschiedlich stark voneinander ab:
Die Darstellungsschicht präsentiert die Daten in der Regel - jedoch nicht notwendigerweise - zwecks Anzeige. Die Programmlogik sollte aus dem View entfernt werden. Der View kennt das Model und ist dort registriert, um sich selbständig aktualisieren zu können.


=== Controller ===
=== Modell ''({{lang|en|model}})'' ===
Das Modell enthält Daten, die von der Ansicht dargestellt werden. Es ist von Ansicht und Steuerung unabhängig. Die Änderungen der Daten werden der Ansicht durch das [[Beobachter (Entwurfsmuster)|Entwurfsmuster „Beobachter“]] bekanntgegeben. In manchen Umsetzungen des MVC-Musters enthält das Modell eine [[Geschäftslogik]], die für die Änderung der Daten zuständig ist.
Der Controller verwaltet die Sichten, nimmt von ihnen Benutzeraktionen entgegen, wertet diese aus und hat schreibenden Zugriff auf das Modell. Er enthält die Intelligenz und steuert den Ablauf (engl. Workflow) der Anwendung.


=== Ansicht ''({{lang|en|view}})'' ===
Das MVC-Muster trifft keine Aussage über die Positionierung der Geschäftslogik innerhalb der MVC-Klassen. Diese kann je nach Anwendungsfall besser im Controller aufgehoben sein oder besser in das Modell verlagert werden (z.B. wenn es mehrere Controller gibt).
Die Ansicht ist für die Darstellung der Daten des Modells und die Realisierung der Benutzerinteraktionen zuständig. Sie kennt das Modell, dessen Daten sie präsentiert, ist aber nicht für die Verarbeitung dieser Daten zuständig. Des Weiteren ist sie von der Steuerung unabhängig. Die Bekanntgabe von Benutzerinteraktionen an die Steuerung geschieht nach dem Entwurfsmuster „Beobachter“. Die Ansicht wird über Änderungen der Daten im Modell mithilfe des Entwurfsmuster „Beobachter“ unterrichtet und kann daraufhin die Darstellung aktualisieren. Die Ansicht verwendet oft das [[Kompositum (Entwurfsmuster)|Entwurfsmuster „Kompositum“]].


=== Steuerung ''({{lang|en|controller}})'' ===
==Analyse==
Die Steuerung verwaltet die Ansicht und das Modell. Sie wird von der Ansicht über Benutzerinteraktionen (mithilfe des [[Beobachter (Entwurfsmuster)|Entwurfsmusters „Beobachter“]]) informiert, wertet diese aus und nimmt daraufhin Anpassungen an der Ansicht sowie Änderungen an den Daten im Modell vor. In einigen modernen Implementierungen des MVC-Musters aktualisiert die Steuerung die Daten im Modell nicht mehr direkt, stattdessen aktualisiert sie die Daten indirekt, indem sie auf die im Modell implementierte Geschäftslogik zugreift. In einem Spezialfall des MVC-Musters kann die Steuerung auch mehrere Ansichten oder mehrere Modelle gleichzeitig verwalten.
Üblicherweise spielen eine View und ein Controller zusammen, welche sich gegenseitig kennen und sich auf ein Model beziehen. Ein Model kann aber von mehreren View-Controller-Paaren gesteuert werden. Beispielsweise kann ein Zahlenwert sowohl über ein Eingabefeld als auch über einen Schieberegler steuerbar sein. Ebenfalls ist denkbar, dass Datenmaterial aus dem Model durch verschiedene Views dargestellt werden kann (z.B. in Form von Tabellen und Diagrammen). Verändert einer der dazugehörigen Controller den Wert, so sendet das Model einen [[Ereignis (Programmierung)|Event]], der dazu führt, dass sich beide grafischen Controls parallel aktualisieren.


=== Nicht festgelegte Funktionalitäten ===
Das MVC-Konzept wurde zunächst für [[Grafische Benutzungsoberfläche|Benutzungsoberflächen]] in [[Smalltalk (Programmiersprache)|Smalltalk]] durch [[Trygve Reenskaug]] beschrieben ([[Seeheim-Modell]]), gilt mittlerweile aber als defacto Standard für den Grobentwurf aller komplexen Softwaresysteme (mit diversen Verfeinerungen und oftmals mehreren jeweils nach MVC pattern aufgeteilten Modulen).
==== Geschäftslogik ====
Da das MVC-Muster in verschiedenen Programmiersprachen unterschiedlich realisiert werden muss, gibt es keine allgemeingültige Definition, wo die [[Geschäftslogik]] innerhalb der MVC-Klassen angesiedelt werden sollte. Sie wird – historisch bedingt – oft noch im Controller programmiert, aber heute zunehmend im Modell implementiert. So enthält das Modell alle [[Geschäftsobjekt]]e mit allen ihren Daten und Funktionen und kann deshalb isoliert, schnell, vollständig und automatisiert getestet werden. Einige MVC-[[Framework]]s schreiben strikt vor, wo die Geschäftslogik zu implementieren ist, andere überlassen diese Entscheidung den Softwareentwicklern.


==== Validierung von Benutzereingaben ====
Die Bedeutung des MVC Entwurfsmuster wird noch klarer, wenn man sich
In ähnlicher Weise ist der Ort für die [[Datenvalidierung|Validierung]] der Benutzereingaben nicht definiert. Einfache Formatvalidierungen können bereits im View realisiert werden. Validierungen, welche stärker die [[Geschäftslogik]] berücksichtigen müssen, werden eher im Modell oder in der Steuerung implementiert.
in die Lage der Entwickler von GUI-Frameworks versetzt. Hier besteht die Herausforderung darin, dass zum Entwicklungszeitpunkt der [[Widget (GUI)|GUI Widgets]] (View) nicht feststeht, welche fachlichen Daten und Datenstrukturen (Modell) repräsentiert und welche fachlichen Abläufe (Controller) realisiert werden sollen. Damit besteht die Aufgabe der Entwickler eines GUI-Frameworks auch darin, eine Abstraktion für das Modell in Form von Schnittstellen bereitzustellen. Darüber hinaus muss die Kommunikation zwischen den einzelnen Komponenten in beide Richtungen gewährleistet sein. Zur Vermeidung von zirkularen Abhängigkeiten zwischen View, Modell und Controller wird diese dann vorzugsweise mittels des [[Beobachter (Entwurfsmuster)|Listener-Entwurfsmuster]] realisiert, auch diese Infrastruktur wird von GUI-Frameworks mitgeliefert.


==== Daten-Formatierung und Internationalisierung ====
Das MVC Entwurfsmuster definiert damit den Rahmen für die Entwickler von GUI-Frameworks, das fertige GUI-Framework beinhaltet:
Auch für die Formatierung der Rohdaten und die [[Internationalisierung (Softwareentwicklung)|Internationalisierung]] ist nicht definiert, wo diese erfolgen. Aus Gründen der Entwicklungseffizienz bietet es sich oft an, diese im Modell zu integrieren, so dass man sich beim View auf die Erstellung von Widgets oder Templates beschränken kann. Andererseits werden dadurch Aspekte der Darstellung in das Modell verlagert, was zur Grundidee durchaus im Widerspruch steht. Als Variante bietet es sich daher auch an, hierfür eigenständige Funktionsbereiche vorzusehen, die man dann weder Model, View noch Controller zurechnen muss.
# einen View in Form ausimplementierter [[Widget (GUI)|GUI Widgets]],
# den Vertrag für das Modell in Form von Schnittstellen,
# den Vertrag für Benachrichtigungen aufgrund von Nutzerinteraktion in Form von Schnittstellen und ausimplementierten Klassen, sowie
# den Vertrag für Benachrichtigungen aufgrund von Modelländerungen in Form von Schnittstellen und ausimplementierten Klassen.


== Heutige Umsetzungen ==
Umstritten ist, ob es sich hierbei um ein [[Entwurfsmuster]] handelt. Nach strenger Auslegung ist das nicht der Fall, da es sich nicht um ein allgemeines, beliebig verwendbares Muster handelt, sondern um einen ganz konkreten Anwendungsfall, nämlich um den der [[Graphical User Interface|Grafischen Oberfläche]].
Die Begriffe des ursprünglichen MVC-Musters werden heute oft entlehnt, um Systeme begreiflich zu machen, die weitaus komplexer sind als die damalige Software. Dabei kommt es auf die Perspektive des betrachteten Teilsystems an, welche Elemente damit bezeichnet werden. So könnte ein [[Webbrowser]] als View eines größeren Gesamtsystems verstanden werden, während schon ein einzelnes Formularelement im Browser wiederum aus einem kleinen Datenmodell, der zugehörigen Darstellung und seiner Steuerung besteht. Die Grundidee der Trennung von Model, View und Controller hat sich erhalten, wird aber feiner granuliert und verschachtelt eingesetzt.


Während sich viele Projekte als Model-View-Controller-Architektur definieren, wird der Begriff sehr vielfältig benutzt. Es etablieren sich neue Begriffe, wie das [[Model-View-Presenter]]-, das [[Model View ViewModel|Model-View-ViewModel]]- oder das [[Model-View-Adapter]]-Muster, die versuchen, die Varianten präziser zu beschreiben.
==Beispiel==
[[Bild:Mvc-modell.png]] Abb. 1


=== Widget-Bibliotheken für Desktop-Applikationen ===
Abbildung 1 zeigt das MVC-Modell für eine einfache Web-Registrierung. Der Benutzer (Client) fragt als erstes die Seite ''register.jsp'' an. Er bekommt eine Seite mit einem [[HTML]]-Formular als Antwort. Als „Action“ ist im Formular die ''validate.jsp'' angegeben. Also schickt der [[Browser]] nach dem Ausfüllen des Formulars die eingegebenen Daten an die ''validate.jsp''. ''Validate.jsp'', welches in diesem Fall der Controller ist, prüft die eingegebenen Werte. Es ist nur für die Prüfung und Verarbeitung der Daten zuständig. Selbst gibt ''validate.jsp'' dem Benutzer kein Feedback. Der Controller gibt dazu die Kontrolle an die entsprechenden Views weiter. In diesem Fall entweder an ''register.jsp'', wenn die Eingaben ungültig waren, sonst an die ''ok.jsp''. Wird die Kontrolle wieder zurück an die ''register.jsp'' übergeben, zeigt ''register.jsp'' dem User erneut das Formular mit z.B. einem Fehler-Hinweis an. Der Browser schickt die korrigierten Daten wieder an die ''validate.jsp''. Sind die Eingaben korrekt, werden die Daten zur Speicherung an die UsersBean übergeben. Die Kontrolle wird daraufhin an die ''ok.jsp'' abgegeben. Diese zeigt dem User beispielsweise eine Erfolgsbestätigung.
Als [[Widget]]s werden die einzelnen Komponenten [[Grafische Benutzeroberfläche|grafischer Benutzeroberflächen]] (GUI) bezeichnet, wie [[Menü (Computer)|Menüpunkte]] oder [[Textfeld]]er. Widgets zeichnen sich dadurch aus, dass sie neben der Ansicht auch typische Merkmale des klassischen Controllers in einer Komponente vereinen, wie das Event-Handling. Einige Widgets, wie z.&nbsp;B. Auswahllisten, können sogar über ein eigenes internes Modell verfügen, wobei dieses dann mit dem eigentlichen Modell synchronisiert werden muss.


Obwohl die Widgets die feste Dreiteilung durchbrechen, spricht man trotzdem noch von einer Model-View-Controller-Architektur. Es kommen auch Komponenten wie Filter zur Sortierung oder Bestätigungsdialoge vor, die sich nicht eindeutig in die klassische Dreiteilung einordnen lassen.
Dies ist natürlich nur eine von vielen Möglichkeiten für ein MVC-Modell. Aber es zeigt die grundlegende Funktionsweise des MVC-Modells. An der Abbildung lässt sich auch gut erkennen, dass unproblematisch einzelne Teile, wie die Datenspeicherung oder das Aussehen, ausgetauscht werden können.


Bei der Anwendung der Widget-[[Programmbibliothek|Bibliotheken]] überlässt der Controller damit einen Teil der klassischen Controller-Funktion den Widgets und beschränkt sich auf die Steuerung des Models und gegebenenfalls anderer Komponenten des Views.
Durch den einfachen Austausch entsteht ein beachtlicher Vorteil des Modells. Die Anwendung kann mit neuer oder anderer Technologie an veränderte Anforderungen relativ einfach angepasst werden. Das MVC-Modell ermöglicht es also, eine anfangs „kleine“ [[Webanwendung]] im Nachhinein bei Bedarf mit mächtiger Technologie nachzurüsten.


Die Bedeutung des MVC-Entwurfsmusters wird noch klarer, wenn man sich in die Lage der Entwickler von GUI-[[Framework]]s versetzt. Hier besteht die Herausforderung darin, dass zum Entwicklungszeitpunkt der [[Steuerelement|GUI Widgets]] ''({{lang|en|View}})'' nicht feststeht, welche fachlichen Daten und Datenstrukturen (Modell) präsentiert und welche fachlichen Abläufe ''({{lang|en|Control}})'' realisiert werden sollen. Damit besteht die Aufgabe der Entwickler eines GUI-Frameworks auch darin, eine Abstraktion für das Modell in Form von Schnittstellen bereitzustellen. An der Abbildung lässt sich gut erkennen, dass einzelne Teile, wie die Datenspeicherung oder das Aussehen, problemlos ausgetauscht werden können.
== Implementierungen ==
[[1979]] wurde MVC das erste mal von [[Trygve Reenskaug]] beschrieben, der dann an [[Smalltalk (Programmiersprache)|Smalltalk]] beim [[PARC]] arbeitete. Diese ursprüngliche Implementierung inspirierte viele andere GUI-Frameworks wie:


Das MVC-Entwurfsmuster definiert auch den Rahmen für die Entwickler von GUI-Frameworks. Ein fertiges GUI-Framework beinhaltet:
*[[NeXTSTEP]]/[[OPENSTEP]] und die darauf beruhenden [[Cocoa]] frameworks
# eine Ansicht ''({{lang|en|view}})'' in Form ausimplementierter GUI-Widgets,
*[[Microsoft Foundation Classes]]
# die Vereinbarung eines zugrundeliegenden Datenmodells in Form von Schnittstellen,
*[[Swing (Java)]]
# die Vereinbarung von [[Ereignis (Programmierung)|Ereignissen]] (englisch ''events'') auf Grund von Benutzerinteraktionen in Form von [[Schnittstelle (Objektorientierung)|Schnittstellen]] und ausimplementierten [[Klasse (Objektorientierung)|Klassen]] sowie
*[[Qt (Toolkit)]]
# die Vereinbarung von Ereignissen auf Grund von Modelländerungen in Form von Schnittstellen und ausimplementierten Klassen.


=== Zusammenspiel von Server und Browser bei Webanwendungen ===
<!--
Im weiteren Sinne verteilt sich das MVC-Muster bei Webanwendungen über Server und Browser und ist damit komplexer als das klassische MVC-Muster. Abstrakt betrachtet übernimmt der Browser dabei die sichtbare Darstellung und unmittelbaren Nutzereingaben sowie die nicht seitenspezifischen Funktionen von Controller und View. Der Server kümmert sich um spezifische Steuerung des Browsers, indem er mit diesem über [[HTTP]] kommuniziert.
More recently there have been attempts to apply MVC architectures for web-based interfaces. In the design of web applications, MVC is also known as a "Model 2" architecture in Sun parlance. Complex web applications continue to be more difficult to design than traditional applications, and MVC is being pushed as a potential solution to these difficulties.


Im engeren Sinne versteht man darunter aber nur das serverseitige Programm. Dabei kann man noch einmal zwischen dem Webserver für statische Webseiten oder dessen Delegation an spezielle Zusatzprogramme unterscheiden. Der Begriff MVC findet insbesondere im Rahmen solcher Zusatzprogramme zum Webserver Verwendung.
* JavaServer Faces , Jakarta Struts and Webwork2 are currently the most popular web oriented MVC implementations. Struts works at a page level; JSF works at a component level.
* The WebObjects development/deployment environment is strongly based on MVC.
* Fusebox
* Mach-II
* Maypole
* Mojavi
* Ruby on Rails
* Tapestry
* ZNF
* Apache Cocoon
-->


== MVC 2 ==
==== Model ====
Für den Browser ist die [[Webseite|HTML-Seite]] der Datenkern seines Models. Aus der Perspektive des Gesamtsystems ist sie nur eine Sicht auf das Gesamtmodel, welches auf dem Server lokalisiert ist.
'''MVC 2''' steht für '''MVC Version 2''' bzw. '''MVC Model 2''' und ist eine für [[Webanwendung|Webanwendungen]] spezialisierte Variante des Architekturmusters Model View Controller (MVC, siehe oben).


==== View ====
Im Gegensatz zu allgemeinen Client/Server-Anwendungen, dient MVC2 in Webanwendungen der Trennung eines Controllers, der [[Hypertext Transfer Protocol|HTTP]]-Requests verarbeitet, von einer View, die [[Hypertext Markup Language|HTML]]-Responses erzeugt. Das einmalige Senden der Response geschieht durch einen Updatekontrollfluss vom Controller zur View.
Der Browser kümmert sich um die allgemeinen Funktionen, wie die Darstellung von Text, Formularelementen und eingebetteten Objekten. Die Darstellung wird dabei im Speziellen durch den View-Programmteil des Servers per HTTP-Response gesteuert, deren Hauptteil der Darstellungsanweisung aus der HTML-Seite besteht.


== Siehe auch ==
==== Controller ====
Der Browser akzeptiert Formulareingaben und sendet diese ab oder nimmt das [[Anklicken]] eines Links entgegen. In beiden Fällen sendet er einen HTTP-Request an den Server. Der Controller-Programmteil verarbeitet die Daten der HTTP-Requests und stößt schließlich die Erstellung eines neuen Views an.
*[[Ajax (Programmierung)]]

*[[Struts]]
==== JavaScript ====
*[[Servlet]]
Die Webseite kann Programmcode enthalten, normalerweise [[JavaScript]], z.&nbsp;B. für die browserseitige Validierung von Formulareingaben oder Steuerungslogiken zum Nachladen von Inhalten. Dieser Programmcode lässt sich wiederum nach dem MVC-Muster gliedern und so als Teil des Gesamtsystems betrachten. Zu beachten ist, dass der Einsatz von [[Client|clientseitiger Logik]], welche nach dem MVC-Muster strukturiert ist, von einem serverseitig verwendeten MVC zu differenzieren ist. Client und Server stellen getrennte Teilsysteme dar. Solche [[Webanwendung]]en werden häufig nach dem [[Single-Page-Webanwendung|Single-Page-Paradigma]] umgesetzt.
*[[JavaServer Pages]]

*[[Catalyst]]
==== Verzicht auf das Observer-Muster ====
*[[JavaServer Faces]]
Bei klassischen Webanwendungen kann der Browser nicht nach dem klassischen [[Beobachter (Entwurfsmuster)|Observer-Muster]] unmittelbar auf Änderungen des Models auf dem Server reagieren. Jede Antwort (HTTP-Response) an den Browser setzt eine Anfrage (HTTP-Request) voraus. Man spricht vom Request-Response-Cycle. Daraus folgt, dass das Observer-Muster auch auf Seiten des Servers seine Vorteile nicht ausspielen kann. Weil es somit einen Mehraufwand bedeutet hätte, kam es typischerweise nicht zum Einsatz. Stattdessen tritt meist der Controller als aktiver Vermittler zwischen Model und View im Rahmen eines Request-Response-Cycles auf.
*[[Three-Tier-Architektur]]

Neuere [[Webanwendung]]en erlauben bereits die Implementierung eines Observer-Musters. Die hierbei verwendete [[Push-Medien|Push-Technologie]] (Server-Push) erlaubt dem Server, Ereignisse direkt und ohne Anfrage an die Clients zu übermitteln. Viele Implementierungen nutzen hierbei das sogenannte [[Long Polling]] (Request mit verzögerter Antwort, bis ein Ereignis eintritt) oder die neueren [[WebSocket|Websockets]]. Einige [[JavaScript-Framework]]s abstrahieren hierbei das Push-Verfahren und nutzen „das Beste“ vom jeweiligen Browser bzw. der Serveranwendung zur Verfügung gestellte Verfahren. Somit ist es auch möglich, das Observer-Muster in Webanwendungen einzuführen.

==== Die besonderen Herausforderungen des Hyperlinks und der Form-Action ====
Der [[Hyperlink]] ist ein herausragendes Merkmal von [[Webanwendung|Webapplikationen]]. In einer klassischen GUI-Applikation würde hier im View ein [[Schaltfläche|Button]] erzeugt, dessen Klickevent anhand seiner ID im Controller mit dem Wechsel der Ansicht verknüpft wird. Der Hyperlink enthält zwar auch eine ID, es ist aber nicht seine eigene, sondern die Zieladresse der neuen Ansicht. Gleiches gilt für die Action-Adresse eines [[Webformular|HTML-Formulars]].

Beides sind für den Benutzer eines Browsers Controller-Elemente, deren funktionales Ziel allerdings in der Webseite codiert ist. Hiermit stellt sich die Herausforderung, bei der Erzeugung der Webseite die reine Ansicht von der Funktionalität der [[Uniform Resource Locator|URL]] zu trennen. Der Entwickler des Views soll sich keine Gedanken über die oft komplexe Controller-Funktionalität der URL machen müssen.

Die Aufgabenteilung von View und Controller kann mittels einer ID einfach erreicht werden. In Analogie zur GUI-Applikation wird die ID im Controller mit der Zieladresse verknüpft. Im View kann die URL über eine Schnittstelle anhand der ID abgerufen werden oder die ID tritt als Platzhalter für die URL ein, zum Beispiel innerhalb eines Templates oder in Form von Link-Objekten im Rahmen eines Objektbaums.

==== JavaScript-Bibliotheken und AJAX-Anbindung ====
Hier wird ein Teil der Programme der Model-View-Controller-Architektur clientseitig im Browser eingesetzt, während ein anderer Teil, insbesondere das Model, auf dem Server verbleibt. JavaScript-Bibliotheken stellen vielfältige Widgets zur Verfügung. Diese Anwendungen nehmen eine Zwischenstellung zwischen Webanwendungen und desktopartigen Widget-Bibliotheken ein.

=== Serverseitige Webanwendungen ===
Der serverseitige Controller wertet in der Regel die eintreffenden Daten ''(Request)'' des Browsers aus. Meist tritt er dabei als Moderator zwischen Model und View auf. Serverseitig werden unter dem View diejenigen Programmteile verstanden, die den [[Hypertext Markup Language|HTML]]-Code für die Antwort ''(Response)'' erzeugen. Häufig arbeitet der View mit HTML-Templates, deren Platzhalter mit den Daten des Models ersetzt werden.

==== Der Controller als Mittler zwischen Model, View und Webserver ====
Ein typischer Funktionsablauf (ohne HTTP-Redirect):

Browser -&gt; HTTP-Request -&gt; Webserver -&gt; Controller
&lt;=&gt; (beliebig häufig) Model oder View
Browser &lt;- HTTP-Response &lt;- Webserver &lt;- Controller

Der Aufgabenumfang des Controllers kann sehr variabel sein. Im einfachsten Fall ordnet er nur Model und View zu. Er kann aber auch vielfältige weitere Aufgaben übernehmen. Das hängt davon ab, wie aktiv oder passiv sich Model und View jeweils verhalten in Bezug auf die Validierung, die Internationalisierung, die Geschäftslogik, die Iterationen über die Daten beim Einfügen in den View und in Bezug auf zahlreiche andere Aspekte.

Die Praxis variiert in Abhängigkeit vom persönlichen Programmierstil, Webservern, Programmiersprachen, Frameworks, dem Einsatz von Unit-Tests und den Projektanforderungen. Im Fall von PHP-Programmen steht zwischen Webserver und Controller z.&nbsp;B. noch der Programm-Interpreter, der bereits die Daten des HTTP-Requests aufbereitet und damit seinerseits Teilfunktionen des klassischen Controllers übernimmt.

==== HTTP-Redirect ====

Nach dem Ändern des Models ({{lang|en|create}} oder {{lang|en|update}}) empfiehlt sich ein HTTP-Redirect. Mit dieser Technik wird das irrtümliche mehrfache Absenden durch einen Seitenreload verhindert. Außerdem wird dadurch das Schreiben des vorherigen Datensatzes vom Lesen des nachfolgenden Datensatzes getrennt, so dass sich die Controller sinnvoller organisieren lassen.

Bei erfolgreicher Validierung greift die erste Anfrage, die noch den vorherigen Datensatz behandelt, schreibend auf das Model zu. Danach erfolgt ein Redirect. Die zweite Anfrage greift lesend zu und präsentiert bereits den nächsten Datensatz zur Bearbeitung. Für den Benutzer des Browsers fühlt sich das wie ein einziger Aufruf an.

Browser -&gt; HTTP-Request -&gt; Webserver -&gt; Controller -&gt; positive Validierung -&gt; Model (speichern der Daten nach Validierung)
Browser &lt;- HTTP-Redirect &lt;- Webserver &lt;- Controller (danach Redirect durch den Controller)
Browser -&gt; HTTP-Request -&gt; Webserver -&gt; Controller -&gt; Model -&gt; View (führt zu neuer Formularanfrage ohne Nutzeraktion)
Browser &lt;- HTTP-Response &lt;- Webserver &lt;- Controller (reguläre Response)

Bei einer gescheiterten Validierung wird dagegen der empfangene Datensatz direkt wieder im gleichen Formular zur Korrektur präsentiert. Ein Redirect entfällt. In der Regel entfällt auch eine erneute Abfrage des Models.

Browser -&gt; HTTP-Request -&gt; Webserver -&gt; Controller -&gt; negative Validierung -&gt; View (Formular zur Überarbeitung der Eingaben)
Browser &lt;- HTTP-Response &lt;- Webserver &lt;- Controller

==== Controller-Kaskade ====

Um dem Umfang der Funktionalität professioneller Webauftritte gerecht zu werden, muss der Controller strukturiert werden. Häufig wird der Controller kaskadenartig strukturiert. Entweder wird auf der untersten Ebene der Kaskade ein Controller angesteuert oder die Controller verzweigen im Ablauf der Kaskade baumartig und führen zu einer Verschachtelung des resultierenden Views.

'''Front-Controller''', '''Controller''' und '''Actions''' sind häufig verwendete Benennungen für eine 3-schichtige Controller-Struktur. Diese Benennungen spiegeln den Aufbau von Datenmodellen und die zugehörigen Datenoperationen wider. Der ''Front-Controller'' verzweigt dabei zu einer Sicht auf das Datenmodell, die im Fokus steht und durch einen ''Controller'' gesteuert wird. Die ''Actions'' als unterste Controller-Ebene führen die Datenoperationen für diese Sicht aus, die man mit Create, Read, Update und Delete ([[CRUD]]) zusammenfassen kann.

Ein Beispiel für eine Controller-Schachtelung wäre eine Website, bei welcher der oberste Controller die Anzeige der Seiten steuert. In einer Seite können wiederum spezifisch mehrere MVC-Blöcke gleichzeitig eingesetzt werden, z.&nbsp;B. für einen zentralen Artikel und für unterschiedliche Kontext-Informationen.

==== Serverseitige Controller mit dem Remote Presentation Model Muster ====
Eine Weiterentwicklung des MVC Musters stellt das [[Remote Presentation Model]] Muster da. Hierbei findet bereits in der Definition des Musters eine größere Trennung zwischen View und Controller statt, wodurch es in vielen Bereichen einfacher ist den Controller auf den Server auszulagern.

== Beispiel: MVC realisiert mit JavaServer Pages ==
[[Datei:Mvc-modell.svg|zentriert|MVC-Modell für eine einfache Web-Registrierung]]

Die obige Abbildung zeigt das MVC-Modell für eine einfache Web-Registrierung. Der Benutzer (Client) fragt als erstes die Seite <code>register.jsp</code> an. Er bekommt eine Seite mit einem HTML-Formular als Antwort. Als ''{{lang|en|Action}}'' ist im Formular die <code>validate.jsp</code> angegeben. Also schickt der [[Webbrowser|Browser]] nach dem Ausfüllen des Formulars die eingegebenen Daten an das <code>validate.jsp</code>, das in diesem Fall das ''{{lang|en|Control}}''-Modul ist und die eingegebenen Werte prüft. Es ist nur für die Prüfung und Verarbeitung der Daten zuständig. Selbst gibt <code>validate.jsp</code> dem Benutzer kein Feedback. Das ''{{lang|en|Control}}''-Modul gibt dazu die Kontrolle an die entsprechenden ''{{lang|en|Views}}'' weiter. In diesem Fall entweder an <code>register.jsp</code>, wenn die Eingaben ungültig waren, sonst an die <code>ok.jsp</code>. Wird die Kontrolle wieder zurück an die <code>register.jsp</code> übergeben, zeigt <code>register.jsp</code> dem Anwender erneut das Formular mit z.&nbsp;B. einem Fehlerhinweis an. Der Browser schickt die korrigierten Daten wieder an die <code>validate.jsp</code>. Sind die Eingaben korrekt, werden die Daten zur Speicherung an die <code>UsersBean</code> übergeben. Die Kontrolle wird daraufhin an die <code>ok.jsp</code> abgegeben. Diese zeigt dem Anwender beispielsweise eine Erfolgsbestätigung.

== Siehe auch ==
* [[Data Context Interaction]] ist eine Weiterentwicklung und ein eher komplementäres Architekturmuster.
* [[Smart UI]]

== Literatur ==
* {{Literatur |Autor=Erich Gamma, Richard Helm, Ralph Johnson |Titel=Entwurfsmuster. Elemente wiederverwendbarer objektorientierter Software |Auflage=2 |Verlag=Addison-Wesley |Sprache=de |ISBN=978-3-8273-1862-6}}


== Weblinks ==
== Weblinks ==
* {{Internetquelle |autor=Steffen Düsel |url=https://www.methodpark.de/blog/model-view-controller-mvc/ |titel=Model-View-Controller (MVC) |sprache=de |archiv-url=https://web.archive.org/web/20210417082732/https://www.methodpark.de/blog/model-view-controller-mvc/ |archiv-datum=2021-04-17 |abruf=2021-06-17 |abruf-verborgen=ja}}
*[http://heim.ifi.uio.no/~trygver/themes/mvc/mvc-index.html Trygve Reenskaug - MVC]
* {{Internetquelle |url=http://c2.com/cgi-bin/wiki?ModelViewController |titel=Model View Controller |werk=Portland Pattern Repository |sprache=en |archiv-url=https://web.archive.org/web/20160806075025/http://c2.com/cgi-bin/wiki?ModelViewController |archiv-datum=2016-08-06 |abruf=2007-12-04 |abruf-verborgen=ja}}
*[http://www.phpwact.org/pattern/model_view_controller MVC] (englisch)
* {{Internetquelle |autor=Jeff Moore |url=http://www.phpwact.org/pattern/model_view_controller |titel=Model View Controller |sprache=en |archiv-url=https://web.archive.org/web/20160608102231/http://www.phpwact.org/pattern/model_view_controller |archiv-datum=2016-06-08 |abruf=2007-01-30 |abruf-verborgen=ja}}
* {{Internetquelle |autor=Martin Fowler |url=https://martinfowler.com/eaaDev/uiArchs.html |titel=GUI Architectures |sprache=en |abruf=2011-05-21 |abruf-verborgen=ja}}

== Einzelnachweise ==
<references />

{{Navigationsleiste Entwurfsmuster}}


[[Kategorie:Entwurfsmuster]]
[[Kategorie:Entwurfsmuster]]
[[Kategorie:Softwarearchitektur]]
[[Kategorie:Softwarearchitektur]]

[[cs:Model-view-controller]]
[[en:Model-view-controller]]
[[eo:MVC]]
[[es:Modelo Vista Controlador]]
[[fr:Modèle-Vue-Contrôleur]]
[[he:ארכיטקטורת MVC]]
[[it:Modello-vista-controllore]]
[[ja:MVC]]
[[nl:Model-View-Controller-model]]
[[pt:MVC]]
[[sv:Model-view-controller]]

Aktuelle Version vom 22. September 2024, 23:03 Uhr

Ein Model-View-Controller-Konzept. Eine durchgezogene Linie symbolisiert hier eine direkte Assoziation, eine gestrichelte eine indirekte Assoziation (zum Beispiel über einen Beobachter).

Model View Controller (MVC, englisch für Modell-Ansicht-Steuerung) ist ein Entwurfsmuster zur Unterteilung einer Software in die drei Komponenten Datenmodell (englisch model), Ansicht (englisch view) und Programmsteuerung (englisch controller). Das Muster kann sowohl als Architekturmuster als auch als Entwurfsmuster eingesetzt werden.[1] Ziel des Musters ist ein flexibler Programmentwurf, der eine spätere Änderung oder Erweiterung erleichtert und eine Wiederverwendbarkeit der einzelnen Komponenten ermöglicht. Es ist dann zum Beispiel möglich, eine Anwendung zu schreiben, die dasselbe Modell nutzt und es dann für Windows, Mac, Linux oder für das Internet zugänglich macht. Die Umsetzungen nutzen dasselbe Modell, nur Controller und View müssen dabei jeweils neu implementiert werden.

Das MVC-Konzept wurde 1979 zunächst für Benutzeroberflächen in Smalltalk durch Trygve Reenskaug beschrieben (Seeheim-Modell), der damals an Smalltalk im Xerox PARC arbeitete. Es gilt mittlerweile aber als De-facto-Standard für den Grobentwurf vieler komplexer Softwaresysteme, teils mit Differenzierungen und oftmals mehreren jeweils nach dem MVC-Muster aufgeteilten Modulen.

Klassisches Architekturmuster

[Bearbeiten | Quelltext bearbeiten]

Die drei Komponenten hängen je nach Umsetzung unterschiedlich stark voneinander ab:

Das Modell enthält Daten, die von der Ansicht dargestellt werden. Es ist von Ansicht und Steuerung unabhängig. Die Änderungen der Daten werden der Ansicht durch das Entwurfsmuster „Beobachter“ bekanntgegeben. In manchen Umsetzungen des MVC-Musters enthält das Modell eine Geschäftslogik, die für die Änderung der Daten zuständig ist.

Die Ansicht ist für die Darstellung der Daten des Modells und die Realisierung der Benutzerinteraktionen zuständig. Sie kennt das Modell, dessen Daten sie präsentiert, ist aber nicht für die Verarbeitung dieser Daten zuständig. Des Weiteren ist sie von der Steuerung unabhängig. Die Bekanntgabe von Benutzerinteraktionen an die Steuerung geschieht nach dem Entwurfsmuster „Beobachter“. Die Ansicht wird über Änderungen der Daten im Modell mithilfe des Entwurfsmuster „Beobachter“ unterrichtet und kann daraufhin die Darstellung aktualisieren. Die Ansicht verwendet oft das Entwurfsmuster „Kompositum“.

Steuerung (controller)

[Bearbeiten | Quelltext bearbeiten]

Die Steuerung verwaltet die Ansicht und das Modell. Sie wird von der Ansicht über Benutzerinteraktionen (mithilfe des Entwurfsmusters „Beobachter“) informiert, wertet diese aus und nimmt daraufhin Anpassungen an der Ansicht sowie Änderungen an den Daten im Modell vor. In einigen modernen Implementierungen des MVC-Musters aktualisiert die Steuerung die Daten im Modell nicht mehr direkt, stattdessen aktualisiert sie die Daten indirekt, indem sie auf die im Modell implementierte Geschäftslogik zugreift. In einem Spezialfall des MVC-Musters kann die Steuerung auch mehrere Ansichten oder mehrere Modelle gleichzeitig verwalten.

Nicht festgelegte Funktionalitäten

[Bearbeiten | Quelltext bearbeiten]

Geschäftslogik

[Bearbeiten | Quelltext bearbeiten]

Da das MVC-Muster in verschiedenen Programmiersprachen unterschiedlich realisiert werden muss, gibt es keine allgemeingültige Definition, wo die Geschäftslogik innerhalb der MVC-Klassen angesiedelt werden sollte. Sie wird – historisch bedingt – oft noch im Controller programmiert, aber heute zunehmend im Modell implementiert. So enthält das Modell alle Geschäftsobjekte mit allen ihren Daten und Funktionen und kann deshalb isoliert, schnell, vollständig und automatisiert getestet werden. Einige MVC-Frameworks schreiben strikt vor, wo die Geschäftslogik zu implementieren ist, andere überlassen diese Entscheidung den Softwareentwicklern.

Validierung von Benutzereingaben

[Bearbeiten | Quelltext bearbeiten]

In ähnlicher Weise ist der Ort für die Validierung der Benutzereingaben nicht definiert. Einfache Formatvalidierungen können bereits im View realisiert werden. Validierungen, welche stärker die Geschäftslogik berücksichtigen müssen, werden eher im Modell oder in der Steuerung implementiert.

Daten-Formatierung und Internationalisierung

[Bearbeiten | Quelltext bearbeiten]

Auch für die Formatierung der Rohdaten und die Internationalisierung ist nicht definiert, wo diese erfolgen. Aus Gründen der Entwicklungseffizienz bietet es sich oft an, diese im Modell zu integrieren, so dass man sich beim View auf die Erstellung von Widgets oder Templates beschränken kann. Andererseits werden dadurch Aspekte der Darstellung in das Modell verlagert, was zur Grundidee durchaus im Widerspruch steht. Als Variante bietet es sich daher auch an, hierfür eigenständige Funktionsbereiche vorzusehen, die man dann weder Model, View noch Controller zurechnen muss.

Heutige Umsetzungen

[Bearbeiten | Quelltext bearbeiten]

Die Begriffe des ursprünglichen MVC-Musters werden heute oft entlehnt, um Systeme begreiflich zu machen, die weitaus komplexer sind als die damalige Software. Dabei kommt es auf die Perspektive des betrachteten Teilsystems an, welche Elemente damit bezeichnet werden. So könnte ein Webbrowser als View eines größeren Gesamtsystems verstanden werden, während schon ein einzelnes Formularelement im Browser wiederum aus einem kleinen Datenmodell, der zugehörigen Darstellung und seiner Steuerung besteht. Die Grundidee der Trennung von Model, View und Controller hat sich erhalten, wird aber feiner granuliert und verschachtelt eingesetzt.

Während sich viele Projekte als Model-View-Controller-Architektur definieren, wird der Begriff sehr vielfältig benutzt. Es etablieren sich neue Begriffe, wie das Model-View-Presenter-, das Model-View-ViewModel- oder das Model-View-Adapter-Muster, die versuchen, die Varianten präziser zu beschreiben.

Widget-Bibliotheken für Desktop-Applikationen

[Bearbeiten | Quelltext bearbeiten]

Als Widgets werden die einzelnen Komponenten grafischer Benutzeroberflächen (GUI) bezeichnet, wie Menüpunkte oder Textfelder. Widgets zeichnen sich dadurch aus, dass sie neben der Ansicht auch typische Merkmale des klassischen Controllers in einer Komponente vereinen, wie das Event-Handling. Einige Widgets, wie z. B. Auswahllisten, können sogar über ein eigenes internes Modell verfügen, wobei dieses dann mit dem eigentlichen Modell synchronisiert werden muss.

Obwohl die Widgets die feste Dreiteilung durchbrechen, spricht man trotzdem noch von einer Model-View-Controller-Architektur. Es kommen auch Komponenten wie Filter zur Sortierung oder Bestätigungsdialoge vor, die sich nicht eindeutig in die klassische Dreiteilung einordnen lassen.

Bei der Anwendung der Widget-Bibliotheken überlässt der Controller damit einen Teil der klassischen Controller-Funktion den Widgets und beschränkt sich auf die Steuerung des Models und gegebenenfalls anderer Komponenten des Views.

Die Bedeutung des MVC-Entwurfsmusters wird noch klarer, wenn man sich in die Lage der Entwickler von GUI-Frameworks versetzt. Hier besteht die Herausforderung darin, dass zum Entwicklungszeitpunkt der GUI Widgets (View) nicht feststeht, welche fachlichen Daten und Datenstrukturen (Modell) präsentiert und welche fachlichen Abläufe (Control) realisiert werden sollen. Damit besteht die Aufgabe der Entwickler eines GUI-Frameworks auch darin, eine Abstraktion für das Modell in Form von Schnittstellen bereitzustellen. An der Abbildung lässt sich gut erkennen, dass einzelne Teile, wie die Datenspeicherung oder das Aussehen, problemlos ausgetauscht werden können.

Das MVC-Entwurfsmuster definiert auch den Rahmen für die Entwickler von GUI-Frameworks. Ein fertiges GUI-Framework beinhaltet:

  1. eine Ansicht (view) in Form ausimplementierter GUI-Widgets,
  2. die Vereinbarung eines zugrundeliegenden Datenmodells in Form von Schnittstellen,
  3. die Vereinbarung von Ereignissen (englisch events) auf Grund von Benutzerinteraktionen in Form von Schnittstellen und ausimplementierten Klassen sowie
  4. die Vereinbarung von Ereignissen auf Grund von Modelländerungen in Form von Schnittstellen und ausimplementierten Klassen.

Zusammenspiel von Server und Browser bei Webanwendungen

[Bearbeiten | Quelltext bearbeiten]

Im weiteren Sinne verteilt sich das MVC-Muster bei Webanwendungen über Server und Browser und ist damit komplexer als das klassische MVC-Muster. Abstrakt betrachtet übernimmt der Browser dabei die sichtbare Darstellung und unmittelbaren Nutzereingaben sowie die nicht seitenspezifischen Funktionen von Controller und View. Der Server kümmert sich um spezifische Steuerung des Browsers, indem er mit diesem über HTTP kommuniziert.

Im engeren Sinne versteht man darunter aber nur das serverseitige Programm. Dabei kann man noch einmal zwischen dem Webserver für statische Webseiten oder dessen Delegation an spezielle Zusatzprogramme unterscheiden. Der Begriff MVC findet insbesondere im Rahmen solcher Zusatzprogramme zum Webserver Verwendung.

Für den Browser ist die HTML-Seite der Datenkern seines Models. Aus der Perspektive des Gesamtsystems ist sie nur eine Sicht auf das Gesamtmodel, welches auf dem Server lokalisiert ist.

Der Browser kümmert sich um die allgemeinen Funktionen, wie die Darstellung von Text, Formularelementen und eingebetteten Objekten. Die Darstellung wird dabei im Speziellen durch den View-Programmteil des Servers per HTTP-Response gesteuert, deren Hauptteil der Darstellungsanweisung aus der HTML-Seite besteht.

Der Browser akzeptiert Formulareingaben und sendet diese ab oder nimmt das Anklicken eines Links entgegen. In beiden Fällen sendet er einen HTTP-Request an den Server. Der Controller-Programmteil verarbeitet die Daten der HTTP-Requests und stößt schließlich die Erstellung eines neuen Views an.

Die Webseite kann Programmcode enthalten, normalerweise JavaScript, z. B. für die browserseitige Validierung von Formulareingaben oder Steuerungslogiken zum Nachladen von Inhalten. Dieser Programmcode lässt sich wiederum nach dem MVC-Muster gliedern und so als Teil des Gesamtsystems betrachten. Zu beachten ist, dass der Einsatz von clientseitiger Logik, welche nach dem MVC-Muster strukturiert ist, von einem serverseitig verwendeten MVC zu differenzieren ist. Client und Server stellen getrennte Teilsysteme dar. Solche Webanwendungen werden häufig nach dem Single-Page-Paradigma umgesetzt.

Verzicht auf das Observer-Muster

[Bearbeiten | Quelltext bearbeiten]

Bei klassischen Webanwendungen kann der Browser nicht nach dem klassischen Observer-Muster unmittelbar auf Änderungen des Models auf dem Server reagieren. Jede Antwort (HTTP-Response) an den Browser setzt eine Anfrage (HTTP-Request) voraus. Man spricht vom Request-Response-Cycle. Daraus folgt, dass das Observer-Muster auch auf Seiten des Servers seine Vorteile nicht ausspielen kann. Weil es somit einen Mehraufwand bedeutet hätte, kam es typischerweise nicht zum Einsatz. Stattdessen tritt meist der Controller als aktiver Vermittler zwischen Model und View im Rahmen eines Request-Response-Cycles auf.

Neuere Webanwendungen erlauben bereits die Implementierung eines Observer-Musters. Die hierbei verwendete Push-Technologie (Server-Push) erlaubt dem Server, Ereignisse direkt und ohne Anfrage an die Clients zu übermitteln. Viele Implementierungen nutzen hierbei das sogenannte Long Polling (Request mit verzögerter Antwort, bis ein Ereignis eintritt) oder die neueren Websockets. Einige JavaScript-Frameworks abstrahieren hierbei das Push-Verfahren und nutzen „das Beste“ vom jeweiligen Browser bzw. der Serveranwendung zur Verfügung gestellte Verfahren. Somit ist es auch möglich, das Observer-Muster in Webanwendungen einzuführen.

[Bearbeiten | Quelltext bearbeiten]

Der Hyperlink ist ein herausragendes Merkmal von Webapplikationen. In einer klassischen GUI-Applikation würde hier im View ein Button erzeugt, dessen Klickevent anhand seiner ID im Controller mit dem Wechsel der Ansicht verknüpft wird. Der Hyperlink enthält zwar auch eine ID, es ist aber nicht seine eigene, sondern die Zieladresse der neuen Ansicht. Gleiches gilt für die Action-Adresse eines HTML-Formulars.

Beides sind für den Benutzer eines Browsers Controller-Elemente, deren funktionales Ziel allerdings in der Webseite codiert ist. Hiermit stellt sich die Herausforderung, bei der Erzeugung der Webseite die reine Ansicht von der Funktionalität der URL zu trennen. Der Entwickler des Views soll sich keine Gedanken über die oft komplexe Controller-Funktionalität der URL machen müssen.

Die Aufgabenteilung von View und Controller kann mittels einer ID einfach erreicht werden. In Analogie zur GUI-Applikation wird die ID im Controller mit der Zieladresse verknüpft. Im View kann die URL über eine Schnittstelle anhand der ID abgerufen werden oder die ID tritt als Platzhalter für die URL ein, zum Beispiel innerhalb eines Templates oder in Form von Link-Objekten im Rahmen eines Objektbaums.

JavaScript-Bibliotheken und AJAX-Anbindung

[Bearbeiten | Quelltext bearbeiten]

Hier wird ein Teil der Programme der Model-View-Controller-Architektur clientseitig im Browser eingesetzt, während ein anderer Teil, insbesondere das Model, auf dem Server verbleibt. JavaScript-Bibliotheken stellen vielfältige Widgets zur Verfügung. Diese Anwendungen nehmen eine Zwischenstellung zwischen Webanwendungen und desktopartigen Widget-Bibliotheken ein.

Serverseitige Webanwendungen

[Bearbeiten | Quelltext bearbeiten]

Der serverseitige Controller wertet in der Regel die eintreffenden Daten (Request) des Browsers aus. Meist tritt er dabei als Moderator zwischen Model und View auf. Serverseitig werden unter dem View diejenigen Programmteile verstanden, die den HTML-Code für die Antwort (Response) erzeugen. Häufig arbeitet der View mit HTML-Templates, deren Platzhalter mit den Daten des Models ersetzt werden.

Der Controller als Mittler zwischen Model, View und Webserver

[Bearbeiten | Quelltext bearbeiten]

Ein typischer Funktionsablauf (ohne HTTP-Redirect):

 Browser  -> HTTP-Request ->  Webserver -> Controller
                                                       <=> (beliebig häufig) Model oder View
 Browser  <- HTTP-Response <- Webserver <- Controller

Der Aufgabenumfang des Controllers kann sehr variabel sein. Im einfachsten Fall ordnet er nur Model und View zu. Er kann aber auch vielfältige weitere Aufgaben übernehmen. Das hängt davon ab, wie aktiv oder passiv sich Model und View jeweils verhalten in Bezug auf die Validierung, die Internationalisierung, die Geschäftslogik, die Iterationen über die Daten beim Einfügen in den View und in Bezug auf zahlreiche andere Aspekte.

Die Praxis variiert in Abhängigkeit vom persönlichen Programmierstil, Webservern, Programmiersprachen, Frameworks, dem Einsatz von Unit-Tests und den Projektanforderungen. Im Fall von PHP-Programmen steht zwischen Webserver und Controller z. B. noch der Programm-Interpreter, der bereits die Daten des HTTP-Requests aufbereitet und damit seinerseits Teilfunktionen des klassischen Controllers übernimmt.

Nach dem Ändern des Models (create oder update) empfiehlt sich ein HTTP-Redirect. Mit dieser Technik wird das irrtümliche mehrfache Absenden durch einen Seitenreload verhindert. Außerdem wird dadurch das Schreiben des vorherigen Datensatzes vom Lesen des nachfolgenden Datensatzes getrennt, so dass sich die Controller sinnvoller organisieren lassen.

Bei erfolgreicher Validierung greift die erste Anfrage, die noch den vorherigen Datensatz behandelt, schreibend auf das Model zu. Danach erfolgt ein Redirect. Die zweite Anfrage greift lesend zu und präsentiert bereits den nächsten Datensatz zur Bearbeitung. Für den Benutzer des Browsers fühlt sich das wie ein einziger Aufruf an.

 Browser  -> HTTP-Request  -> Webserver -> Controller -> positive Validierung -> Model (speichern der Daten nach Validierung)
 Browser  <- HTTP-Redirect <- Webserver <- Controller                                  (danach Redirect durch den Controller)
 Browser  -> HTTP-Request  -> Webserver -> Controller -> Model -> View                 (führt zu neuer Formularanfrage ohne Nutzeraktion)
 Browser  <- HTTP-Response <- Webserver <- Controller                                  (reguläre Response)

Bei einer gescheiterten Validierung wird dagegen der empfangene Datensatz direkt wieder im gleichen Formular zur Korrektur präsentiert. Ein Redirect entfällt. In der Regel entfällt auch eine erneute Abfrage des Models.

 Browser  -> HTTP-Request ->  Webserver -> Controller -> negative Validierung -> View (Formular zur Überarbeitung der Eingaben)
 Browser  <- HTTP-Response <- Webserver <- Controller

Controller-Kaskade

[Bearbeiten | Quelltext bearbeiten]

Um dem Umfang der Funktionalität professioneller Webauftritte gerecht zu werden, muss der Controller strukturiert werden. Häufig wird der Controller kaskadenartig strukturiert. Entweder wird auf der untersten Ebene der Kaskade ein Controller angesteuert oder die Controller verzweigen im Ablauf der Kaskade baumartig und führen zu einer Verschachtelung des resultierenden Views.

Front-Controller, Controller und Actions sind häufig verwendete Benennungen für eine 3-schichtige Controller-Struktur. Diese Benennungen spiegeln den Aufbau von Datenmodellen und die zugehörigen Datenoperationen wider. Der Front-Controller verzweigt dabei zu einer Sicht auf das Datenmodell, die im Fokus steht und durch einen Controller gesteuert wird. Die Actions als unterste Controller-Ebene führen die Datenoperationen für diese Sicht aus, die man mit Create, Read, Update und Delete (CRUD) zusammenfassen kann.

Ein Beispiel für eine Controller-Schachtelung wäre eine Website, bei welcher der oberste Controller die Anzeige der Seiten steuert. In einer Seite können wiederum spezifisch mehrere MVC-Blöcke gleichzeitig eingesetzt werden, z. B. für einen zentralen Artikel und für unterschiedliche Kontext-Informationen.

Serverseitige Controller mit dem Remote Presentation Model Muster

[Bearbeiten | Quelltext bearbeiten]

Eine Weiterentwicklung des MVC Musters stellt das Remote Presentation Model Muster da. Hierbei findet bereits in der Definition des Musters eine größere Trennung zwischen View und Controller statt, wodurch es in vielen Bereichen einfacher ist den Controller auf den Server auszulagern.

Beispiel: MVC realisiert mit JavaServer Pages

[Bearbeiten | Quelltext bearbeiten]
MVC-Modell für eine einfache Web-Registrierung
MVC-Modell für eine einfache Web-Registrierung

Die obige Abbildung zeigt das MVC-Modell für eine einfache Web-Registrierung. Der Benutzer (Client) fragt als erstes die Seite register.jsp an. Er bekommt eine Seite mit einem HTML-Formular als Antwort. Als Action ist im Formular die validate.jsp angegeben. Also schickt der Browser nach dem Ausfüllen des Formulars die eingegebenen Daten an das validate.jsp, das in diesem Fall das Control-Modul ist und die eingegebenen Werte prüft. Es ist nur für die Prüfung und Verarbeitung der Daten zuständig. Selbst gibt validate.jsp dem Benutzer kein Feedback. Das Control-Modul gibt dazu die Kontrolle an die entsprechenden Views weiter. In diesem Fall entweder an register.jsp, wenn die Eingaben ungültig waren, sonst an die ok.jsp. Wird die Kontrolle wieder zurück an die register.jsp übergeben, zeigt register.jsp dem Anwender erneut das Formular mit z. B. einem Fehlerhinweis an. Der Browser schickt die korrigierten Daten wieder an die validate.jsp. Sind die Eingaben korrekt, werden die Daten zur Speicherung an die UsersBean übergeben. Die Kontrolle wird daraufhin an die ok.jsp abgegeben. Diese zeigt dem Anwender beispielsweise eine Erfolgsbestätigung.

  • Erich Gamma, Richard Helm, Ralph Johnson: Entwurfsmuster. Elemente wiederverwendbarer objektorientierter Software. 2. Auflage. Addison-Wesley, ISBN 978-3-8273-1862-6.
  • Steffen Düsel: Model-View-Controller (MVC). Archiviert vom Original am 17. April 2021;.
  • Model View Controller. In: Portland Pattern Repository. Archiviert vom Original am 6. August 2016; (englisch).
  • Jeff Moore: Model View Controller. Archiviert vom Original am 8. Juni 2016; (englisch).
  • Martin Fowler: GUI Architectures. (englisch).

Einzelnachweise

[Bearbeiten | Quelltext bearbeiten]
  1. Kamal Wickramanayake: Is MVC a design pattern or an architectural pattern? In: Software View. 17. Juli 2010, abgerufen am 16. Dezember 2016 (englisch).