<?xml version="1.0"?>
<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="de">
	<id>https://wiki-de.moshellshocker.dns64.de/index.php?action=history&amp;feed=atom&amp;title=Message_Passing_Interface</id>
	<title>Message Passing Interface - Versionsgeschichte</title>
	<link rel="self" type="application/atom+xml" href="https://wiki-de.moshellshocker.dns64.de/index.php?action=history&amp;feed=atom&amp;title=Message_Passing_Interface"/>
	<link rel="alternate" type="text/html" href="https://wiki-de.moshellshocker.dns64.de/index.php?title=Message_Passing_Interface&amp;action=history"/>
	<updated>2026-05-19T10:36:21Z</updated>
	<subtitle>Versionsgeschichte dieser Seite in Wikipedia (Deutsch) – Lokale Kopie</subtitle>
	<generator>MediaWiki 1.43.8</generator>
	<entry>
		<id>https://wiki-de.moshellshocker.dns64.de/index.php?title=Message_Passing_Interface&amp;diff=210501&amp;oldid=prev</id>
		<title>~2026-25734-62: Kommafehler</title>
		<link rel="alternate" type="text/html" href="https://wiki-de.moshellshocker.dns64.de/index.php?title=Message_Passing_Interface&amp;diff=210501&amp;oldid=prev"/>
		<updated>2026-04-27T13:21:49Z</updated>

		<summary type="html">&lt;p&gt;Kommafehler&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Neue Seite&lt;/b&gt;&lt;/p&gt;&lt;div&gt;{{Infobox Software&lt;br /&gt;
| Name = MPI&lt;br /&gt;
| Logo = [[Datei:MPI-logo-green.gif|Logo]]&lt;br /&gt;
| Screenshot = &lt;br /&gt;
| Beschreibung = MPI Logo&lt;br /&gt;
| Maintainer = &lt;br /&gt;
| Hersteller = [http://www.mpi-forum.org/ MPI Forum]&lt;br /&gt;
| Erscheinungsjahr = &lt;br /&gt;
| AktuelleVersion = [https://www.mpi-forum.org/docs/mpi-4.1/mpi41-report.pdf Version 4.1] (PDF; 5,3&amp;amp;nbsp;MB)&lt;br /&gt;
| AktuelleVersionFreigabeDatum = 2. November 2023&lt;br /&gt;
| AktuelleVorabVersion = &lt;br /&gt;
| AktuelleVorabVersionFreigabeDatum = &lt;br /&gt;
| Betriebssystem = [[Linux]], [[Unix]], [[Microsoft Windows NT]], [[macOS]]&lt;br /&gt;
| Programmiersprache = [[C++]], [[C (Programmiersprache)|C]], [[Fortran]], [[Python (Programmiersprache)|Python]], [[Java (Programmiersprache)|Java]]&lt;br /&gt;
| Kategorie = [[Programmierschnittstelle|API]]&lt;br /&gt;
| Lizenz = &lt;br /&gt;
| Deutsch = nein&lt;br /&gt;
| Website = [http://www.mpi-forum.org/ www.mpi-forum.org]&lt;br /&gt;
| Dateien = &lt;br /&gt;
}}&lt;br /&gt;
&lt;br /&gt;
&amp;#039;&amp;#039;&amp;#039;Message Passing Interface&amp;#039;&amp;#039;&amp;#039; (&amp;#039;&amp;#039;&amp;#039;MPI&amp;#039;&amp;#039;&amp;#039;) ist ein Standard, der den [[Nachrichtenaustausch]] bei parallelen Berechnungen auf verteilten Computersystemen beschreibt. Er legt dabei eine Sammlung von Operationen und ihre Semantik, also eine [[Programmierschnittstelle]], fest, aber keine [[Implementierung]].&lt;br /&gt;
&lt;br /&gt;
Eine MPI-Applikation besteht in der Regel aus mehreren miteinander kommunizierenden [[Prozess (Informatik)|Prozessen]], die alle zu Beginn der Programmausführung parallel gestartet werden. Alle diese Prozesse arbeiten dann gemeinsam an einem Problem und nutzen zum Datenaustausch Nachrichten, welche explizit von einem zum anderen Prozess geschickt werden. Ein Vorteil dieses Prinzips ist es, dass der Nachrichtenaustausch auch über Rechnergrenzen hinweg funktioniert. Parallele MPI-Programme sind somit sowohl auf PC-[[Computercluster|Clustern]] (hier funktioniert der Austausch der Nachrichten z.&amp;amp;nbsp;B. über [[Transmission Control Protocol|TCP]]), als auch auf dedizierten [[Parallelrechner]]n ausführbar (hier läuft der Nachrichtenaustausch über ein Hochgeschwindigkeitsnetz wie [[InfiniBand]] oder [[Myrinet]] oder über den [[Shared Memory|gemeinsamen]] [[Hauptspeicher]]).&lt;br /&gt;
&lt;br /&gt;
== Geschichte ==&lt;br /&gt;
1992 begann die Entwicklung des MPI 1.0 Standards mit Entwürfen (November 1992, Februar 1993, November 1993). Ausgangspunkt waren ältere Kommunikationsbibliotheken wie PVM, PARMACS, P4, Chameleon und Zipcode. Der Standard erschien am 5. Mai 1994 mit&lt;br /&gt;
* Punkt-zu-Punkt-Kommunikation&lt;br /&gt;
* globale Kommunikation&lt;br /&gt;
* Gruppen, Kontext und Kommunikatoren&lt;br /&gt;
* Umgebung&lt;br /&gt;
* Profiling-Schnittstelle&lt;br /&gt;
* Spracheinbindung für C und [[Fortran]] 77&lt;br /&gt;
&lt;br /&gt;
Im Juni 1995 erfolgten Fehlerkorrekturen mit MPI 1.1.&lt;br /&gt;
&lt;br /&gt;
Am 18. Juli 1997 wurde die stabile Version MPI 1.2 veröffentlicht, die neben weiteren Fehlerkorrekturen eine Versionidentifikation erlaubt. Sie wird auch als MPI-1 bezeichnet.&lt;br /&gt;
&lt;br /&gt;
Am 30. Mai 2008 erschien MPI 1.3 mit weiteren Fehlerkorrekturen und Klarstellungen.&lt;br /&gt;
&lt;br /&gt;
Zeitgleich zur Version 1.2 wurde am 18. Juli 1997 auch der MPI 2.0 Standard verabschiedet. Dieser wird auch als MPI-2 bezeichnet und enthält unter anderem folgende Erweiterungen:&lt;br /&gt;
* parallele Datei-Ein-/Ausgabe&lt;br /&gt;
* dynamische Prozessverwaltung&lt;br /&gt;
* Zugriff auf Speicher anderer Prozesse&lt;br /&gt;
* zusätzliche Spracheinbindung von [[C++]] und Fortran 90&lt;br /&gt;
&lt;br /&gt;
Am 23. Juni 2008 wurden die bisher separaten Teile MPI-1 und MPI-2 zu einem gemeinsamen Dokument vereint und als MPI 2.1 veröffentlicht. MPI Standard Version 2.2 ist vom 4. September 2009 und enthält weitere Verbesserungen und kleinere Erweiterungen.&lt;br /&gt;
&lt;br /&gt;
Am 21. September 2012 hat das MPI Forum MPI-3 veröffentlicht,&amp;lt;ref&amp;gt;{{Webarchiv |url=http://www.mpi-forum.org/docs/docs.html |text=MPI Documents |wayback=20061106152359 |archiv-bot=2019-05-01 05:48:36 InternetArchiveBot}}&amp;lt;/ref&amp;gt; das neue Funktionalität einarbeitet wie beispielsweise nicht blockierende Kollektive, ein verbessertes einseitiges Kommunikationsmodell (RMA, Remote Memory Access), ein neues Fortran-Interface, topographiebezogene Kommunikation und nicht blockierende parallele Ein- und Ausgabe.&amp;lt;ref&amp;gt;[http://www.hpcadvisorycouncil.com/events/2012/Spain-Workshop/pres/3_OSU.pdf &amp;#039;&amp;#039;The Future of MPI&amp;#039;&amp;#039;.] (PDF; 2,4&amp;amp;nbsp;MB)&amp;lt;/ref&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Am 9. Juni 2021 wurde MPI-4 veröffentlicht. Wesentliche Neuerungen sind Funktionsinterfaces, die Parameter mit einem größeren Wertebereich unterstützen. Zuvor waren durch den vorgegebenen [[Integer (Datentyp)#Häufige Speicherformen|32-bit Datentyp]] wesentliche Größen, wie z.&amp;amp;nbsp;B. die Anzahl zu kommunizierender Datenelemente, auf etwas mehr als zwei Milliarden beschränkt. Persistente Kollektive eröffnen Möglichkeiten zur Optimierung wiederholt ausgeführter Kommunikation und sind zudem im Gegensatz zu den schon vorhandenen kollektiven Operationen in beliebiger Reihenfolge ausführbar. Zudem wurde die Fehlerbehandlung in vielen Punkten verbessert und ein neues Session-Modell zur dynamischen Nutzung der von MPI verwalteten Ressourcen eingeführt.&amp;lt;ref&amp;gt;[https://www.mpi-forum.org/docs/mpi-4.0/mpi40-report.pdf#subsection.B.1.2 &amp;#039;&amp;#039;Changes in MPI-4.0&amp;#039;&amp;#039;]&amp;lt;/ref&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Zu den Hauptentwicklern gehört [[Bill Gropp]].&lt;br /&gt;
&lt;br /&gt;
== Punkt-zu-Punkt Kommunikation ==&lt;br /&gt;
Die grundlegendste Art der Kommunikation findet zwischen zwei Prozessen statt: ein Sendeprozess übermittelt dabei Informationen an einen Empfangsprozess. In MPI werden diese Informationen in sogenannte Nachrichten verpackt, die mit den Parametern &amp;lt;code&amp;gt;buffer&amp;lt;/code&amp;gt;, &amp;lt;code&amp;gt;count&amp;lt;/code&amp;gt;, und &amp;lt;code&amp;gt;datatype&amp;lt;/code&amp;gt; beschrieben werden. Zu jeder Sendeoperation muss eine passende Empfangsoperation existieren. Da in parallelen Anwendungen die bloße Reihenfolge der Abarbeitung von Operationen nicht immer ausreichend ist, bietet MPI zusätzlich den &amp;lt;code&amp;gt;tag&amp;lt;/code&amp;gt;-Parameter an – nur wenn dieser Wert bei Sende- und Empfangsoperation identisch ist, dann passen beide zusammen.&lt;br /&gt;
&lt;br /&gt;
=== Blockierendes Senden und Empfangen ===&lt;br /&gt;
Die einfachsten Operationen für eine Punkt-zu-Punkt Kommunikation sind &amp;#039;&amp;#039;senden&amp;#039;&amp;#039; und &amp;#039;&amp;#039;empfangen&amp;#039;&amp;#039;:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;C&amp;quot;&amp;gt;&lt;br /&gt;
int MPI_Send (void* buf, int count, MPI_Datatype datatype, int dest, int tag, MPI_Comm comm)&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
* &amp;#039;&amp;#039;buf&amp;#039;&amp;#039;: [[Zeiger (Informatik)|Zeiger]] auf den Sendepuffer&lt;br /&gt;
* &amp;#039;&amp;#039;count&amp;#039;&amp;#039;: Zahl der Elemente im Sendepuffer&lt;br /&gt;
* &amp;#039;&amp;#039;datatype&amp;#039;&amp;#039;: [[Datentyp]] der Elemente im Sendepuffer&lt;br /&gt;
* &amp;#039;&amp;#039;dest&amp;#039;&amp;#039;: Rang des Zielprozesses&lt;br /&gt;
* &amp;#039;&amp;#039;tag&amp;#039;&amp;#039;: Markierung der Nachricht&lt;br /&gt;
* &amp;#039;&amp;#039;comm&amp;#039;&amp;#039;: Kommunikator der Prozessgruppe&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;C&amp;quot;&amp;gt;&lt;br /&gt;
int MPI_Recv (void* buf, int count, MPI_Datatype datatype, int source, int tag, MPI_Comm comm, MPI_Status* status)&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
* &amp;lt;code&amp;gt;buf&amp;lt;/code&amp;gt;: Zeiger auf einen Empfangspuffer ausreichender Größe&lt;br /&gt;
* &amp;lt;code&amp;gt;count&amp;lt;/code&amp;gt;: Zahl der Elemente im Empfangspuffer&lt;br /&gt;
* &amp;lt;code&amp;gt;datatype&amp;lt;/code&amp;gt;: Datentyp der Elemente im Empfangspuffer&lt;br /&gt;
* &amp;lt;code&amp;gt;source&amp;lt;/code&amp;gt;: Rang des Quellprozesses (mit &amp;lt;code&amp;gt;source=MPI_ANY_SOURCE&amp;lt;/code&amp;gt; wird von einem beliebigen Prozess empfangen)&lt;br /&gt;
* &amp;lt;code&amp;gt;tag&amp;lt;/code&amp;gt;: erwartete Markierung der Nachricht (mit &amp;lt;code&amp;gt;tag=MPI_ANY_TAG&amp;lt;/code&amp;gt; wird jede Nachricht empfangen)&lt;br /&gt;
* &amp;lt;code&amp;gt;comm&amp;lt;/code&amp;gt;: Kommunikator der Prozessgruppe&lt;br /&gt;
* &amp;lt;code&amp;gt;status&amp;lt;/code&amp;gt;: Zeiger auf eine Statusstruktur, in der Informationen über die empfangene Nachricht abgelegt werden sollen&lt;br /&gt;
&lt;br /&gt;
Die beiden Operationen sind &amp;#039;&amp;#039;blockierend&amp;#039;&amp;#039; und &amp;#039;&amp;#039;asynchron&amp;#039;&amp;#039;. Das bedeutet:&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_Recv&amp;lt;/code&amp;gt; kann ausgeführt werden, bevor das zugehörige &amp;lt;code&amp;gt;MPI_Send&amp;lt;/code&amp;gt; gestartet wurde&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_Recv&amp;lt;/code&amp;gt; blockiert, bis die Nachricht vollständig empfangen wurde&lt;br /&gt;
Analog gilt:&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_Send&amp;lt;/code&amp;gt; kann ausgeführt werden, bevor das zugehörige &amp;lt;code&amp;gt;MPI_Recv&amp;lt;/code&amp;gt; gestartet wurde&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_Send&amp;lt;/code&amp;gt; blockiert, bis der Sendepuffer wiederverwendet werden kann (d.&amp;amp;nbsp;h. die Nachricht vollständig übermittelt oder zwischengepuffert wurde)&lt;br /&gt;
&lt;br /&gt;
==== Programmbeispiel ====&lt;br /&gt;
Die Verwendung von &amp;lt;code&amp;gt;MPI_Send&amp;lt;/code&amp;gt; und &amp;lt;code&amp;gt;MPI_Recv&amp;lt;/code&amp;gt; wird im folgenden [[ANSI-C]]-Beispiel für 2 MPI-Prozesse veranschaulicht:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;C&amp;quot;&amp;gt;&lt;br /&gt;
#include &amp;quot;mpi.h&amp;quot;&lt;br /&gt;
#include &amp;lt;stdio.h&amp;gt;&lt;br /&gt;
#include &amp;lt;string.h&amp;gt;&lt;br /&gt;
&lt;br /&gt;
int main(int argc, char *argv[])&lt;br /&gt;
{&lt;br /&gt;
    int myrank, message_size=50, tag=42;&lt;br /&gt;
    char message[message_size];&lt;br /&gt;
    MPI_Status status;&lt;br /&gt;
&lt;br /&gt;
    MPI_Init(&amp;amp;argc, &amp;amp;argv);&lt;br /&gt;
    MPI_Comm_rank(MPI_COMM_WORLD, &amp;amp;myrank);&lt;br /&gt;
&lt;br /&gt;
    if (myrank == 0) {&lt;br /&gt;
        MPI_Recv(message, message_size, MPI_CHAR, 1, tag, MPI_COMM_WORLD, &amp;amp;status);&lt;br /&gt;
        printf(&amp;quot;received \&amp;quot;%s\&amp;quot;\n&amp;quot;, message);&lt;br /&gt;
    }&lt;br /&gt;
    else {&lt;br /&gt;
        strcpy(message, &amp;quot;Hello, there&amp;quot;);&lt;br /&gt;
        MPI_Send(message, strlen(message)+1, MPI_CHAR, 0, tag, MPI_COMM_WORLD);&lt;br /&gt;
    }&lt;br /&gt;
    MPI_Finalize();&lt;br /&gt;
    return 0;&lt;br /&gt;
}&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
=== Nichtblockierende Kommunikation ===&lt;br /&gt;
Die Effizienz einer parallelen Anwendung kann oftmals gesteigert werden, indem man Kommunikation mit Berechnung überlappt und/oder synchronisationsbedingte Wartezeiten vermeidet. Dazu definiert der MPI-Standard sogenannte nichtblockierende Kommunikation, bei der die Kommunikationsoperation lediglich angestoßen wird. Eine separate Funktion muss dann aufgerufen werden, um solch eine Operation zu beenden. Im Unterschied zur blockierenden Variante wird beim Starten der Operation ein &amp;lt;code&amp;gt;Request&amp;lt;/code&amp;gt;-Objekt erzeugt, mit dessen Hilfe auf die Beendigung dieser Operation geprüft oder gewartet werden kann.&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;C&amp;quot;&amp;gt;&lt;br /&gt;
int MPI_Isend (void* buf, int count, MPI_Datatype datatype, int dest, int tag, MPI_Comm comm, MPI_Request* request)&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
* …&lt;br /&gt;
* &amp;lt;code&amp;gt;request&amp;lt;/code&amp;gt;: Adresse der [[Datenstruktur]], die Informationen zur Operation enthält&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;C&amp;quot;&amp;gt;&lt;br /&gt;
int MPI_Irecv (void* buf, int count, MPI_Datatype datatype, int source, int tag, MPI_Comm comm, MPI_Request* request)&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
* …&lt;br /&gt;
&lt;br /&gt;
==== Fortschritt abfragen ====&lt;br /&gt;
Um den Fortschritt einer dieser Operationen zu erfahren, wird folgende Operation verwendet:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;C&amp;quot;&amp;gt;&lt;br /&gt;
int MPI_Test (MPI_Request* request, int* flag, MPI_Status* status)&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Wobei &amp;lt;code&amp;gt;flag=1&amp;lt;/code&amp;gt; oder &amp;lt;code&amp;gt;0&amp;lt;/code&amp;gt; gesetzt wird, je nachdem, ob die Operation abgeschlossen ist oder noch andauert.&lt;br /&gt;
&lt;br /&gt;
==== Blockierend warten ====&lt;br /&gt;
Um dennoch blockierend auf eine &amp;lt;code&amp;gt;MPI_Isend&amp;lt;/code&amp;gt;- oder &amp;lt;code&amp;gt;MPI_Irecv&amp;lt;/code&amp;gt;-Operation zu warten, wird folgende Operation benutzt:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;C&amp;quot;&amp;gt;&lt;br /&gt;
int MPI_Wait (MPI_Request* request, MPI_Status* status)&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
=== Synchronisierendes Senden ===&lt;br /&gt;
Für die Sendeoperationen werden auch die synchronen Varianten &amp;lt;code&amp;gt;MPI_Ssend&amp;lt;/code&amp;gt; und &amp;lt;code&amp;gt;MPI_Issend&amp;lt;/code&amp;gt; definiert. In diesem Modus wird das Senden erst dann beendet, wenn die zugehörige Empfangsoperation begonnen wurde.&lt;br /&gt;
&lt;br /&gt;
=== Puffernde Varianten ===&lt;br /&gt;
…&lt;br /&gt;
&lt;br /&gt;
== Gruppen und Kommunikatoren ==&lt;br /&gt;
Prozesse lassen sich in &amp;#039;&amp;#039;Gruppen&amp;#039;&amp;#039; zusammenfassen, wobei jedem Prozess eine eindeutige Nummer, der sogenannte &amp;#039;&amp;#039;Rang&amp;#039;&amp;#039; zugeordnet wird. Für den Zugriff auf eine Gruppe wird ein &amp;#039;&amp;#039;Kommunikator&amp;#039;&amp;#039; benötigt. Soll also eine globale Kommunikationsoperation auf eine Gruppe beschränkt werden, so muss der zur Gruppe gehörende Kommunikator angegeben werden. Der Kommunikator für die Menge aller Prozesse heißt &amp;lt;code&amp;gt;MPI_COMM_WORLD&amp;lt;/code&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
Die zum Kommunikator &amp;lt;code&amp;gt;comm&amp;lt;/code&amp;gt; gehörende Gruppe erhält man mit&lt;br /&gt;
&lt;br /&gt;
&amp;lt;syntaxhighlight lang=&amp;quot;C&amp;quot;&amp;gt;&lt;br /&gt;
int MPI_Comm_group (MPI_Comm comm, MPI_Group* group)&lt;br /&gt;
&amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Für Prozessgruppen stehen die üblichen Mengenoperationen zur Verfügung.&lt;br /&gt;
&lt;br /&gt;
=== Vereinigung ===&lt;br /&gt;
Zwei Gruppen &amp;lt;code&amp;gt;group1&amp;lt;/code&amp;gt; und &amp;lt;code&amp;gt;group2&amp;lt;/code&amp;gt; können zu einer neuen Gruppe &amp;lt;code&amp;gt;new_group&amp;lt;/code&amp;gt; vereinigt werden:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;&lt;br /&gt;
int &amp;#039;&amp;#039;&amp;#039;MPI_Group_union&amp;#039;&amp;#039;&amp;#039; (MPI_Group &amp;#039;&amp;#039;group1&amp;#039;&amp;#039;, MPI_Group &amp;#039;&amp;#039;group2&amp;#039;&amp;#039;, MPI_Group* &amp;#039;&amp;#039;new_group&amp;#039;&amp;#039;)&lt;br /&gt;
&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Die Prozesse aus &amp;lt;code&amp;gt;group1&amp;lt;/code&amp;gt; behalten ihre ursprüngliche Nummerierung. Die aus &amp;lt;code&amp;gt;group2&amp;lt;/code&amp;gt;, die nicht bereits in der ersten enthalten sind, werden fortlaufend weiter nummeriert.&lt;br /&gt;
&lt;br /&gt;
=== Schnittmenge ===&lt;br /&gt;
Die Schnittmenge zweier Gruppen erhält man mit&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;&lt;br /&gt;
int &amp;#039;&amp;#039;&amp;#039;MPI_Group_intersection&amp;#039;&amp;#039;&amp;#039; (MPI_Group &amp;#039;&amp;#039;group1&amp;#039;&amp;#039;, MPI_Group &amp;#039;&amp;#039;group2&amp;#039;&amp;#039;, MPI_Group* &amp;#039;&amp;#039;new_group&amp;#039;&amp;#039;)&lt;br /&gt;
&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
=== Differenz ===&lt;br /&gt;
Die Differenz zweier Gruppen erhält man mit&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;&lt;br /&gt;
int &amp;#039;&amp;#039;&amp;#039;MPI_Group_difference&amp;#039;&amp;#039;&amp;#039; (MPI_Group &amp;#039;&amp;#039;group1&amp;#039;&amp;#039;, MPI_Group &amp;#039;&amp;#039;group2&amp;#039;&amp;#039;, MPI_Group* &amp;#039;&amp;#039;new_group&amp;#039;&amp;#039;)&lt;br /&gt;
&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== Globale Kommunikation ==&lt;br /&gt;
In parallelen Anwendungen trifft man häufig spezielle Kommunikationsmuster an, bei denen mehrere oder gar alle MPI-Prozesse gleichzeitig beteiligt sind. Der MPI-Standard hat deswegen für die wichtigsten Muster eigene Operationen definiert. Diese unterscheidet man grob in drei Arten: Synchronisation (Barrier), Kommunikation (z.Bsp. Broadcast, Gather, Alltoall) und Kommunikation gekoppelt mit Berechnung (z.&amp;amp;nbsp;B. Reduce oder Scan). Manche dieser Operationen verwenden einen ausgewählten MPI-Prozess, der eine Sonderrolle einnimmt und der typischerweise mit &amp;lt;code&amp;gt;root&amp;lt;/code&amp;gt; bezeichnet wird. Wo es sinnvoll ist, existieren zusätzlich zu den regulären Kommunikationsoperationen noch vektorbasierte Varianten (z.&amp;amp;nbsp;B. Scatterv), die unterschiedliche Argumente pro Prozess ermöglichen.&lt;br /&gt;
&lt;br /&gt;
=== Broadcast (ausstrahlen) ===&lt;br /&gt;
[[Datei:mpi broadcast.svg|mini|hochkant|Die Broadcast-Operation für drei Prozesse]]&lt;br /&gt;
&lt;br /&gt;
Mit der &amp;#039;&amp;#039;Broadcast&amp;#039;&amp;#039;-Operation schickt ein ausgewählter MPI-Prozess &amp;lt;code&amp;gt;root&amp;lt;/code&amp;gt; allen anderen Prozessen in seiner Gruppe &amp;lt;code&amp;gt;comm&amp;lt;/code&amp;gt; die gleichen Daten.&lt;br /&gt;
Die dafür definierte Funktion ist dabei für alle beteiligten Prozesse identisch:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;&lt;br /&gt;
int &amp;#039;&amp;#039;&amp;#039;MPI_Bcast&amp;#039;&amp;#039;&amp;#039; (void *&amp;#039;&amp;#039;buffer&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;count&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;type&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;root&amp;#039;&amp;#039;, MPI_Comm &amp;#039;&amp;#039;comm&amp;#039;&amp;#039;)&lt;br /&gt;
&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Der MPI-Prozess &amp;lt;code&amp;gt;root&amp;lt;/code&amp;gt; stellt in &amp;lt;code&amp;gt;buffer&amp;lt;/code&amp;gt; seine Daten zur Verfügung, während die anderen Prozesse hier die Adresse ihres Empfangspuffers übergeben. Die restlichen Parameter müssen bei allen Prozessen gleich (bzw. gleichwertig) sein. Nachdem die Funktion zurückkehrt, befinden sich in allen Puffern die Daten, die ursprünglich nur bei &amp;lt;code&amp;gt;root&amp;lt;/code&amp;gt; vorhanden waren.&lt;br /&gt;
&lt;br /&gt;
=== Gather (sammeln) ===&lt;br /&gt;
[[Datei:mpi gather.svg|mini|Die Gather-Operation]]&lt;br /&gt;
&lt;br /&gt;
Mit der &amp;#039;&amp;#039;Gather&amp;#039;&amp;#039;-Operation sammelt der MPI-Prozess &amp;lt;code&amp;gt;root&amp;lt;/code&amp;gt; die Daten aller beteiligten Prozesse ein. Die Daten aller Sendepuffer werden dabei (nach Rang sortiert) hintereinander im Empfangspuffer abgelegt:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;&lt;br /&gt;
int &amp;#039;&amp;#039;&amp;#039;MPI_Gather&amp;#039;&amp;#039;&amp;#039; (void *&amp;#039;&amp;#039;sendbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;sendcount&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;sendtype&amp;#039;&amp;#039;, void *&amp;#039;&amp;#039;recvbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;recvcount&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;recvtype&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;root&amp;#039;&amp;#039;, MPI_Comm &amp;#039;&amp;#039;comm&amp;#039;&amp;#039;)&lt;br /&gt;
&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
==== Vektorbasierte Variante ====&lt;br /&gt;
Die vektorbasierte Variante der Gather-Operation erlaubt eine prozessabhängige Anzahl von Elementen:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;&lt;br /&gt;
int &amp;#039;&amp;#039;&amp;#039;MPI_Gatherv&amp;#039;&amp;#039;&amp;#039; (void *&amp;#039;&amp;#039;sendbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;sendcount&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;sendtype&amp;#039;&amp;#039;, void *&amp;#039;&amp;#039;recvbuf&amp;#039;&amp;#039;, int *&amp;#039;&amp;#039;recvcounts&amp;#039;&amp;#039;, int *&amp;#039;&amp;#039;displs&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;recvtype&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;root&amp;#039;&amp;#039;, MPI_Comm &amp;#039;&amp;#039;comm&amp;#039;&amp;#039;)&lt;br /&gt;
&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
* &amp;lt;code&amp;gt;recvcounts&amp;lt;/code&amp;gt;: Feld, das die Zahl der Elemente enthält, die von den einzelnen Prozessen empfangen werden (nur für &amp;lt;code&amp;gt;root&amp;lt;/code&amp;gt; relevant)&lt;br /&gt;
* &amp;lt;code&amp;gt;displs&amp;lt;/code&amp;gt;: Feld, dessen Eintrag i die Verschiebung im Empfangspuffer festlegt, bei der die Daten von Prozess i abgelegt werden sollen (ebenfalls nur für &amp;lt;code&amp;gt;root&amp;lt;/code&amp;gt; relevant)&lt;br /&gt;
&lt;br /&gt;
Bei den Feldern ist zu beachten, dass im Empfangspuffer zwar Lücken erlaubt sind aber keine Überlappungen.&lt;br /&gt;
Sollen also etwa von 3 Prozessen jeweils 1, 2 und 3 Elemente vom Typ &amp;#039;&amp;#039;Integer&amp;#039;&amp;#039; empfangen werden, so muss &amp;lt;code&amp;gt;recvcounts = {1, 2, 3}&amp;lt;/code&amp;gt; und &amp;lt;code&amp;gt;displs = {0, 1 * sizeof(int), 3 * sizeof(int)}&amp;lt;/code&amp;gt; gesetzt werden.&lt;br /&gt;
&lt;br /&gt;
=== Scatter (streuen) ===&lt;br /&gt;
[[Datei:mpi scatter.svg|mini|hochkant|Die Scatter-Operation]]&lt;br /&gt;
&lt;br /&gt;
Mit einer &amp;#039;&amp;#039;Scatter&amp;#039;&amp;#039;-Operation schickt der MPI Prozess &amp;lt;code&amp;gt;root&amp;lt;/code&amp;gt; jedem beteiligten Prozess ein unterschiedliches, aber gleich großes Datenelement:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;int &amp;#039;&amp;#039;&amp;#039;MPI_Scatter&amp;#039;&amp;#039;&amp;#039; (void *&amp;#039;&amp;#039;sendbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;sendcount&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;sendtype&amp;#039;&amp;#039;, void *&amp;#039;&amp;#039;recvbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;recvcount&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;recvtype&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;root&amp;#039;&amp;#039;, MPI_Comm &amp;#039;&amp;#039;comm&amp;#039;&amp;#039;)&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
==== Vektorbasierte Variante ====&lt;br /&gt;
&amp;lt;code&amp;gt;int &amp;#039;&amp;#039;&amp;#039;MPI_Scatterv&amp;#039;&amp;#039;&amp;#039; (void *&amp;#039;&amp;#039;sendbuf&amp;#039;&amp;#039;, int *&amp;#039;&amp;#039;sendcounts&amp;#039;&amp;#039;, int *&amp;#039;&amp;#039;displs&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;sendtype&amp;#039;&amp;#039;, void *&amp;#039;&amp;#039;recvbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;recvcount&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;recvtype&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;root&amp;#039;&amp;#039;, MPI_Comm &amp;#039;&amp;#039;comm&amp;#039;&amp;#039;)&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
=== Akkumulation ===&lt;br /&gt;
Die Akkumulation ist eine spezielle Form der &amp;#039;&amp;#039;Gather&amp;#039;&amp;#039;-Operation. Hierbei werden ebenfalls die Daten aller beteiligten Prozesse aufgesammelt, aber zusätzlich noch mittels einer festgelegten Reduktionsoperation zu einem Datum reduziert. Sei beispielsweise &amp;lt;math&amp;gt;x_i&amp;lt;/math&amp;gt; der Wert bei dem Prozess mit Rang &amp;lt;math&amp;gt;i&amp;lt;/math&amp;gt;, dann liefert Reduce(+) die Gesamtsumme aller Werte: &amp;lt;math&amp;gt;\sum_{i=0}^{n-1} x_i&amp;lt;/math&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;int &amp;#039;&amp;#039;&amp;#039;MPI_Reduce&amp;#039;&amp;#039;&amp;#039; (void *&amp;#039;&amp;#039;sendbuf&amp;#039;&amp;#039;, void *&amp;#039;&amp;#039;recvbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;count&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;type&amp;#039;&amp;#039;, MPI_Op &amp;#039;&amp;#039;op&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;root&amp;#039;&amp;#039;, MPI_Comm &amp;#039;&amp;#039;comm&amp;#039;&amp;#039;)&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Für den Parameter &amp;lt;code&amp;gt;op&amp;lt;/code&amp;gt; existieren dabei folgende vordefinierte Reduktionsoperationen:&lt;br /&gt;
&lt;br /&gt;
==== Logische Operationen ====&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_LAND&amp;lt;/code&amp;gt;: logische UND-Verknüpfung&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_BAND&amp;lt;/code&amp;gt;: bitweise UND-Verknüpfung&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_LOR&amp;lt;/code&amp;gt;: logische ODER-Verknüpfung&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_BOR&amp;lt;/code&amp;gt;: bitweise ODER-Verknüpfung&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_LXOR&amp;lt;/code&amp;gt;: logische exklusiv-ODER-Verknüpfung&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_BXOR&amp;lt;/code&amp;gt;: bitweise exklusiv-ODER-Verknüpfung&lt;br /&gt;
&lt;br /&gt;
==== Arithmetische Operationen ====&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_MAX&amp;lt;/code&amp;gt;: Maximum&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_MIN&amp;lt;/code&amp;gt;: Minimum&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_SUM&amp;lt;/code&amp;gt;: Summe&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_PROD&amp;lt;/code&amp;gt;: Produkt&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_MINLOC&amp;lt;/code&amp;gt;: Minimum mit Prozess&lt;br /&gt;
* &amp;lt;code&amp;gt;MPI_MAXLOC&amp;lt;/code&amp;gt;: Maximum mit Prozess&lt;br /&gt;
&lt;br /&gt;
Die Operationen &amp;lt;code&amp;gt;MPI_MINLOC&amp;lt;/code&amp;gt; und &amp;lt;code&amp;gt;MPI_MAXLOC&amp;lt;/code&amp;gt; geben zusätzlich den Rang des MPI-Prozesses zurück, der das Ergebnis bestimmte.&lt;br /&gt;
&lt;br /&gt;
==== Benutzerdefinierte Operationen ====&lt;br /&gt;
Zusätzlich zu den vordefinierten Reduktionsoperationen können auch eigene Reduktionsoperationen verwendet werden. Dazu wird eine frei programmierbare binäre Verknüpfungsoperation, die assoziativ sein muss und optional kommutativ sein kann, dem MPI bekanntgegeben:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;int &amp;#039;&amp;#039;&amp;#039;MPI_Op_create&amp;#039;&amp;#039;&amp;#039; (MPI_User_function *&amp;#039;&amp;#039;function&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;commute&amp;#039;&amp;#039;, MPI_Op *&amp;#039;&amp;#039;op&amp;#039;&amp;#039;)&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Die dazugehörige Nutzerfunktion berechnet aus zwei Eingabewerten einen Ausgabewert und macht dies – aus Optimierungsgründen – nicht nur einmal mit Skalaren, sondern elementweise auf Vektoren beliebiger Länge:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;typedef void &amp;#039;&amp;#039;&amp;#039;MPI_User_function&amp;#039;&amp;#039;&amp;#039; (void *invec, void *inoutvec, int *len, MPI_Datatype *datatype)&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
==== Präfixreduzierung ====&lt;br /&gt;
Zusätzlich zur oben genannten Akkumulation, existiert auch eine Allreduce Variante – welche das gleiche Ergebnis allen MPI-Prozessen zur Verfügung stellt und nicht nur einem &amp;lt;code&amp;gt;root&amp;lt;/code&amp;gt; Prozess. Die sogenannte Präfixreduzierung erweitert nun diese Möglichkeit, indem nicht allen Prozessen das gleiche Ergebnis, sondern stattdessen ein prozessspezifisches Teilergebnis berechnet wird. Sei beispielsweise erneut &amp;lt;math&amp;gt;x_i&amp;lt;/math&amp;gt; der Wert bei dem Prozess mit Rang &amp;lt;math&amp;gt;i&amp;lt;/math&amp;gt;, dann liefert Scan(+) die Partialsumme der Werte von Rang &amp;lt;math&amp;gt;0&amp;lt;/math&amp;gt; bis &amp;lt;math&amp;gt;i&amp;lt;/math&amp;gt;: &amp;lt;math&amp;gt;\sum_{j=0}^{i} x_j&amp;lt;/math&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;int &amp;#039;&amp;#039;&amp;#039;MPI_Scan&amp;#039;&amp;#039;&amp;#039; (void *&amp;#039;&amp;#039;sendbuf&amp;#039;&amp;#039;, void *&amp;#039;&amp;#039;recvbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;count&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;type&amp;#039;&amp;#039;, MPI_Op &amp;#039;&amp;#039;op&amp;#039;&amp;#039;, MPI_Comm &amp;#039;&amp;#039;comm&amp;#039;&amp;#039;)&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Soll der eigene Wert nicht mit in die Berechnung eingehen (d.&amp;amp;nbsp;h., ausgeschlossen werden), so kann dies mit der exklusiven Scan Funktion &amp;lt;code&amp;gt;MPI_Exscan&amp;lt;/code&amp;gt; bewerkstelligt werden.&lt;br /&gt;
&lt;br /&gt;
=== Allgather ===&lt;br /&gt;
[[Datei:mpi allgather.svg|mini|Die Allgather-Operation]]&lt;br /&gt;
&lt;br /&gt;
Bei der &amp;#039;&amp;#039;Allgather&amp;#039;&amp;#039;-Operation schickt jeder Prozess an jeden anderen Prozess die gleichen Daten. Es handelt sich also um eine Multi-Broadcast-Operation, bei der es keinen gesonderten MPI-Prozess gibt.&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;int &amp;#039;&amp;#039;&amp;#039;MPI_Allgather&amp;#039;&amp;#039;&amp;#039; (void *&amp;#039;&amp;#039;sendbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;sendcount&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;sendtype&amp;#039;&amp;#039;, void *&amp;#039;&amp;#039;recvbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;recvcount&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;recvtype&amp;#039;&amp;#039;, MPI_Comm &amp;#039;&amp;#039;comm&amp;#039;&amp;#039;)&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
=== All-to-all (Gesamtaustausch) ===&lt;br /&gt;
[[Datei:mpi alltoall.svg|mini|Die Alltoall-Operation]]&lt;br /&gt;
&lt;br /&gt;
Bei der All-to-all-Kommunikation werden – ähnlich wie bei der Allgather-Kommunikation – Daten zwischen allen Prozessen ausgetauscht. Dabei wird jedoch nur der &amp;#039;&amp;#039;i&amp;#039;&amp;#039;-te Teil des Sendebuffers an den &amp;#039;&amp;#039;i&amp;#039;&amp;#039;-ten Prozess gesendet. Daten, die vom Prozess mit dem Rang &amp;#039;&amp;#039;j&amp;#039;&amp;#039; kommen, werden entsprechend an &amp;#039;&amp;#039;j&amp;#039;&amp;#039;-ter Stelle im Empfangsbuffer gespeichert.&lt;br /&gt;
&lt;br /&gt;
&amp;lt;code&amp;gt;int &amp;#039;&amp;#039;&amp;#039;MPI_Alltoall&amp;#039;&amp;#039;&amp;#039; (void *&amp;#039;&amp;#039;sendbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;sendcount&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;sendtype&amp;#039;&amp;#039;, void *&amp;#039;&amp;#039;recvbuf&amp;#039;&amp;#039;, int &amp;#039;&amp;#039;recvcount&amp;#039;&amp;#039;, MPI_Datatype &amp;#039;&amp;#039;recvtype&amp;#039;&amp;#039;, MPI_Comm &amp;#039;&amp;#039;comm&amp;#039;&amp;#039;)&amp;lt;/code&amp;gt;&lt;br /&gt;
&lt;br /&gt;
Des Weiteren gibt es noch die synchronisierende &amp;#039;&amp;#039;MPI_Barrier&amp;#039;&amp;#039;-Operation. Diese Funktion kehrt erst zurück, nachdem alle in der angegebenen Gruppe befindlichen MPI-Prozesse diesen Teil des Programmes erreicht haben.&lt;br /&gt;
&lt;br /&gt;
== MPI-2 ==&lt;br /&gt;
Seit 1997 ist eine zweite Version des MPI-Standards verfügbar, die einige Erweiterungen zu dem weiterhin bestehenden MPI-1.1 Standard hinzufügt. Zu diesen Erweiterungen gehören unter anderem&lt;br /&gt;
* eine dynamische Prozessverwaltung, d.&amp;amp;nbsp;h. Prozesse können nun zur Laufzeit erzeugt und gelöscht werden&lt;br /&gt;
* [paralleler] Zugriff auf das Dateisystem&lt;br /&gt;
&lt;br /&gt;
* einseitige Kommunikation&lt;br /&gt;
* Spezifikation zusätzlicher Sprachschnittstellen (C++, Fortran 90), wobei die Sprachschnittstellen zu C++ seit MPI 2.2 als veraltet markiert sind&lt;br /&gt;
&lt;br /&gt;
Beispiel: Lesen einer nx(n+1)-Matrix mit paralleler Datei-Eingabe und size Prozessen mit den Nummern rank = 0 … size-1. Die Spalte n+1 enthält die rechte Seite des Gleichungssystems A * x = b in Form der erweiterten Matrix [A, b]. Die Zeilen der Matrix werden gleichmäßig auf die Prozessoren verteilt. Die Verteilung erfolgt zyklisch (Jeder Prozessor eine Zeile, nach size Zeilen wird wieder rank=0 bedient) und nicht blockweise (jeder Prozessor bekommt einen zusammenhängenden Block von n/size Zeilen):&lt;br /&gt;
&lt;br /&gt;
    ndims = 1;                /* dimensions          */&lt;br /&gt;
    aosi [0] = size * (n+1);  /* array of sizes      */&lt;br /&gt;
    aoss [0] = n+1;           /* array of subsizes   */&lt;br /&gt;
    aost [0] = rank * (n+1);  /* array of starts     */&lt;br /&gt;
    order = MPI_ORDER_C;      /* row or column order */&lt;br /&gt;
    MPI_Type_create_subarray (ndims, aosi, aoss, aost, order, MPI_DOUBLE, &amp;amp;ft);&lt;br /&gt;
    MPI_Type_commit (&amp;amp;ft);&lt;br /&gt;
&lt;br /&gt;
    MPI_File_open (MPI_COMM_WORLD, fn, MPI_MODE_RDONLY, MPI_INFO_NULL, &amp;amp;fh);&lt;br /&gt;
    MPI_File_set_view (fh, sizeof (int), MPI_DOUBLE, ft, „native“, MPI_INFO_NULL);&lt;br /&gt;
&lt;br /&gt;
    for (i = rank; i &amp;lt; n; i+=size)&lt;br /&gt;
    {   MPI_File_read (fh, rdbuffer, n+1, MPI_DOUBLE, &amp;amp;status);&lt;br /&gt;
        for (j = 0; j &amp;lt; n+1; j++)&lt;br /&gt;
        {   A [i / size] [j] = rdbuffer [j]; /* nur die dem Prozess zugeordneten Zeilen */&lt;br /&gt;
    }   }&lt;br /&gt;
&lt;br /&gt;
    MPI_File_close (&amp;amp;fh);&lt;br /&gt;
&lt;br /&gt;
Die Schnittstelle folgt mit leichten der Parallelität geschuldeten Änderungen dem POSIX-1003.1-Standard. Die Datei wird mit MPI_File_open zum gemeinsamen Lesen eröffnet. Die Blenden (views) für die einzelnen Prozesse werden mit MPI_File_set_view festgelegt. Hier wird die vorher definierte Variable ft (filetype) benötigt, in der in einem Block von size * (n+1) doubles eine Zeile mit n+1 doubles herausgepickt wird, die bei Position rank * (n+1) beginnt. Somit wird vom gesamten Block jedem Prozess sukzessive genau eine Zeile zugewiesen. Dieser Typ wird mit MPI_Type_create_subarray definiert und mit MPI_Type_commit im MPI-System bekanntgemacht. Jeder Prozess liest mit MPI_File_read „seine“ Zeilen mit den Nummern i = rank, rank + size, rank + 2*size, … bis die gesamte Matrix gelesen wurden. Das Argument size of (int) berücksichtigt die Größe der Matrix, die am Anfang der Datei als int gespeichert wird.&lt;br /&gt;
&lt;br /&gt;
Gewinn: In size Prozessoren kann eine Matrix verteilt gespeichert werden, die auf einem einzelnen Prozessor in dessen Speicher keinen Platz mehr hätte. Das rechtfertigt auch die Konvention, als Speicher einer Parallelanlage die Summe des Speichers der Einzelcores und Einzelknoten anzugeben.&lt;br /&gt;
&lt;br /&gt;
Dateiformat:&lt;br /&gt;
 n&lt;br /&gt;
 Zeile 0 (n+1) Zahlen für Prozess rank 0&lt;br /&gt;
 Zeile 1 (n+1) Zahlen für Prozess rank 1&lt;br /&gt;
 …&lt;br /&gt;
 Zeile r (n+1) Zahlen für Prozess rank r&lt;br /&gt;
 …&lt;br /&gt;
 Zeile size-1 (n+1) Zahlen für Prozess rank size-1&lt;br /&gt;
&lt;br /&gt;
 Zeile size (n+1) Zahlen für Prozess rank 0&lt;br /&gt;
 Zeile size+1 (n+1) Zahlen für Prozess rank 1&lt;br /&gt;
 …&lt;br /&gt;
 Zeile size+r (n+1) Zahlen für Prozess rank r&lt;br /&gt;
 …&lt;br /&gt;
 Zeile 2*size-1 (n+1) Zahlen für Prozess rank size-1&lt;br /&gt;
&lt;br /&gt;
 Zeile 2*size (n+1) Zahlen für Prozess rank 0&lt;br /&gt;
 …&lt;br /&gt;
 …&lt;br /&gt;
 es folgen entsprechend der Zeilenzahl der Matrix ausreichend viele solcher Blöcke&lt;br /&gt;
&lt;br /&gt;
Das eigentliche Lesen erfolgt mit MPI_File_read. Jeder Prozess liest sequentiell nur die ihm zugeteilten Zeilen. Die kollektive Operation besteht darin, dass die MPI-Bibliothek das Lesen optimieren und parallelisieren kann. Nach Ende des Lesens muss die Datei wie üblich geschlossen werden. Das geschieht mit MPI_File_close. MPI verfügt für die Operationen über eigene Datentypen MPI_Datatype ft und MPI_File fh. Die Beschreibung des filetypes erfolgt mit normalen C-Variablen: int ndims;&lt;br /&gt;
int aosi [1];&lt;br /&gt;
int aoss [1];&lt;br /&gt;
int aost [1];&lt;br /&gt;
int order;&lt;br /&gt;
&lt;br /&gt;
Weiteres in.&amp;lt;ref name=&amp;quot;MPI-2&amp;quot;&amp;gt;&amp;#039;&amp;#039;{{Webarchiv |url=http://www.mpi-forum.org/docs/mpi-20-html/mpi2-report.html |text=MPI-2: Extensions to the Message-Passing Interface |wayback=20070921212101 |archiv-bot=2019-05-01 05:48:36 InternetArchiveBot}}&amp;#039;&amp;#039;&amp;lt;/ref&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== Implementierungen ==&lt;br /&gt;
=== C++, C und Fortran ===&lt;br /&gt;
Die erste Implementierung des MPI-1.x-Standards war [[Mpich|MPICH]] vom [[Argonne National Laboratory]] und der [[Mississippi State University]]. Mittlerweile ist MPICH2 verfügbar, das den MPI-2.1-Standard implementiert. LAM/MPI vom Ohio Supercomputing Center war eine weitere freie Version, deren Weiterentwicklung inzwischen zugunsten von Open MPI eingestellt wurde.&lt;br /&gt;
&lt;br /&gt;
Ab der Version 1.35 der [[Boost (C++-Bibliothek)|Boost Libraries]] gibt es Boost.MPI, eine C++-freundliche Schnittstelle zu verschiedenen MPI-Implementierungen. Auch andere Projekte, wie z.&amp;amp;nbsp;B. [[TPO++]], bieten diese Möglichkeit und sind in der Lage, STL-Container zu versenden und zu empfangen.&lt;br /&gt;
&lt;br /&gt;
=== C# ===&lt;br /&gt;
* [http://osl.iu.edu/research/mpi.net/ MPI.NET] (MPI1)&lt;br /&gt;
&lt;br /&gt;
=== Python ===&lt;br /&gt;
* [https://mpi4py.github.io MPI for Python] (MPI-1/MPI-2)&lt;br /&gt;
* [http://pympi.sourceforge.net/ pyMPI] (Entwicklung eingestellt)&lt;br /&gt;
* [http://mathema.tician.de/software/boostmpi Boost:MPI Python Bindings] (Entwicklung eingestellt)&lt;br /&gt;
&lt;br /&gt;
=== Java ===&lt;br /&gt;
* [http://mpjexpress.org/ MPJ Express]&lt;br /&gt;
* [http://www.hpjava.org/mpiJava.html mpiJava] (MPI-1; Entwicklung eingestellt)&lt;br /&gt;
&lt;br /&gt;
=== Perl ===&lt;br /&gt;
* [http://search.cpan.org/search?mode=all&amp;amp;query=mpi einige MPI Module]&lt;br /&gt;
&lt;br /&gt;
=== R ===&lt;br /&gt;
* [http://www.stats.uwo.ca/faculty/yu/Rmpi/ Rmpi]&lt;br /&gt;
&lt;br /&gt;
=== Haskell ===&lt;br /&gt;
* [http://www.foldr.org/~michaelw/hmpi/ hMPI]&lt;br /&gt;
* [http://hackage.haskell.org/package/haskell-mpi Haskell-MPI]&lt;br /&gt;
&lt;br /&gt;
== Siehe auch ==&lt;br /&gt;
* [[Parallele Virtuelle Maschine]]&lt;br /&gt;
* [[OpenMP]]&lt;br /&gt;
&lt;br /&gt;
== Literatur ==&lt;br /&gt;
* Heiko Bauke, Stephan Mertens: &amp;#039;&amp;#039;Cluster Computing.&amp;#039;&amp;#039; Springer, 2006, ISBN 3-540-42299-4.&lt;br /&gt;
* William Gropp, Ewing Lusk, Anthony Skjellum: &amp;#039;&amp;#039;MPI – Eine Einführung – Portable parallele Programmierung mit dem Message-Passing Interface&amp;#039;&amp;#039;. München 2007, ISBN 978-3-486-58068-6.&lt;br /&gt;
* M. Firuziaan, O. Nommensen: &amp;#039;&amp;#039;Parallel Processing via MPI &amp;amp; OpenMP&amp;#039;&amp;#039;. Linux Enterprise, 10/2002&lt;br /&gt;
* [[Marc Snir]], Steve Otto, Steven Huss-Lederman, David Walker, [[Jack Dongarra]]: &amp;#039;&amp;#039;MPI – The complete reference&amp;#039;&amp;#039;, Vol 1: The MPI core. 2. Auflage. MIT Press, 1998&lt;br /&gt;
* William Gropp, Steven Huss-Lederman, Andrew Lumsdaine, Ewing Lusk, Bill Nitzberg, William Saphir, Marc Snir: &amp;#039;&amp;#039;MPI-The Complete Reference.&amp;#039;&amp;#039; Vol. 2: The MPI-2 Extensions. The MIT Press, 1998.&lt;br /&gt;
&lt;br /&gt;
== Weblinks ==&lt;br /&gt;
* [http://www-unix.mcs.anl.gov/mpi/ MPI]&lt;br /&gt;
* [http://www.mpi-forum.org/ MPI-Forum]&lt;br /&gt;
* [http://www.eurompi2010.org/ EuroMPI – MPI Users’ Group Meeting (2010 edition)]&lt;br /&gt;
* [http://www.open-mpi.org/ Open MPI – frei verfügbare MPI Implementierung]&lt;br /&gt;
* [http://phase.hpcc.jp/mirrors/mpi/mpich/index.html MPICH – frei verfügbare MPI Implementierung]&lt;br /&gt;
* [http://www-unix.mcs.anl.gov/mpi/mpich2/ MPICH2 – frei verfügbare MPI Implementierung]&lt;br /&gt;
* [http://mvapich.cse.ohio-state.edu/ MVAPICH – MPI Implementierung mit VAPI]&lt;br /&gt;
* [http://www.boost.org/ Boost]&lt;br /&gt;
* [http://tpo.sourceforge.net/ TPO++]&lt;br /&gt;
&lt;br /&gt;
== Einzelnachweise ==&lt;br /&gt;
&amp;lt;references /&amp;gt;&lt;br /&gt;
&lt;br /&gt;
[[Kategorie:Verteiltes System]]&lt;br /&gt;
[[Kategorie:Programmierschnittstelle]]&lt;br /&gt;
[[Kategorie:Parallelverarbeitung]]&lt;/div&gt;</summary>
		<author><name>~2026-25734-62</name></author>
	</entry>
</feed>