Regulärer Ausdruck für Gleitkommazahlen


115

Ich habe die Aufgabe, Gleitkommazahlen abzugleichen. Ich habe den folgenden regulären Ausdruck dafür geschrieben:

[-+]?[0-9]*\.?[0-9]*

Es wird jedoch ein Fehler zurückgegeben:

Invalid escape sequence (valid ones are  \b  \t  \n  \f  \r  \"  \'  \\ )

Nach meinem Wissen müssen wir auch für das einen Escape-Charakter verwenden .. Bitte korrigieren Sie mich, wo ich falsch liege.


10
In welcher Sprache wird dieser reguläre Ausdruck verwendet?
CaffGeek

3
@JDB - Warum verschenken Sie 100 Punkte für eine Zahl / Float-Regex? Der Standard war (?:\d+(?:\.\d*)?|\.\d+)und ist immer ad infinitum auf SO ...


1
[-+]?([0-9]*[.])?[0-9]+([eE][-+]?\d+)?Wenn Sie auch die Exponentialnotation abfangen möchten, z. B. 3.023e-23
wcochran

In einigen Sprachen wie Java oder C ++ muss der Backslash maskiert werden. Um den regulären Ausdruck "\." Zu erhalten, verwenden Sie die Zeichenfolge "\\.". Python umgeht dies, indem es rohe Zeichenfolgen verwendet.
HackerBoss

Antworten:


255

TL; DR

Verwenden Sie [.]anstelle von \.und [0-9]anstelle von \d, um zu vermeiden, dass Probleme in einigen Sprachen (wie Java) vermieden werden.

Vielen Dank an den Namenlosen , der dies ursprünglich erkannt hat.

Ein relativ einfaches Muster zum Abgleichen einer Gleitkommazahl ist

[+-]?([0-9]*[.])?[0-9]+

Dies wird übereinstimmen:

  • 123
  • 123.456
  • .456

Siehe ein Arbeitsbeispiel

Wenn Sie auch übereinstimmen möchten 123.(ein Punkt ohne Dezimalteil), benötigen Sie einen etwas längeren Ausdruck:

[+-]?([0-9]+([.][0-9]*)?|[.][0-9]+)

Siehe pkeller Antwort für eine ausführlichere Erklärung dieses Musters

Wenn Sie nicht dezimale Zahlen wie Hex und Oktal einfügen möchten, lesen Sie meine Antwort auf Wie identifiziere ich, ob eine Zeichenfolge eine Zahl ist? .

Wenn Sie möchten bestätigen , dass eine Eingabe eine Zahl ( und nicht eine Zahl innerhalb des Eingangs zu finden), dann sollten Sie das Muster umgeben mit ^und $, etwa so:

^[+-]?([0-9]+([.][0-9]*)?|[.][0-9]+)$

Unregelmäßige reguläre Ausdrücke

"Reguläre Ausdrücke", wie sie in den meisten modernen Sprachen, APIs, Frameworks, Bibliotheken usw. implementiert sind, basieren auf einem in der formalen Sprachtheorie entwickelten Konzept . Softwareentwickler haben jedoch viele Erweiterungen hinzugefügt, die diese Implementierungen weit über die formale Definition hinausführen. Während sich die meisten Engines für reguläre Ausdrücke ähneln, gibt es eigentlich keinen Standard. Aus diesem Grund hängt vieles davon ab, welche Sprache, API, welches Framework oder welche Bibliothek Sie verwenden.

(Um Verwirrung zu vermeiden , haben viele übrigens " Regex " oder " Regexp " verwendet, um diese erweiterten Übereinstimmungssprachen zu beschreiben. Weitere Informationen finden Sie unter Ist ein Regex mit einem regulären Ausdruck identisch ? Auf RexEgg.com.)

Das heißt, die meisten Regex-Motoren (eigentlich alle, soweit ich weiß) würden akzeptieren \.. Höchstwahrscheinlich gibt es ein Problem mit der Flucht.

Das Problem mit der Flucht

Einige Sprachen bieten integrierte Unterstützung für reguläre Ausdrücke, z. B. JavaScript . Für diejenigen Sprachen, die dies nicht tun, kann das Entkommen ein Problem sein.

Dies liegt daran, dass Sie grundsätzlich in einer Sprache innerhalb einer Sprache codieren. Java wird beispielsweise \als Escape-Zeichen in seinen Zeichenfolgen verwendet. Wenn Sie also ein wörtliches Backslash-Zeichen in eine Zeichenfolge einfügen möchten, müssen Sie es maskieren:

// creates a single character string: "\"
String x = "\\";

Allerdings Regexes auch die Verwendung \Charakter zu entkommen, wenn Sie also eine wörtliche übereinstimmen soll \Charakter, müssen Sie es für die regexe Motor entweichen, und es dann wieder entkommen für Java:

// Creates a two-character string: "\\"
// When used as a regex pattern, will match a single character: "\"
String regexPattern = "\\\\";

In Ihrem Fall sind Sie dem Backslash-Zeichen in der Sprache, in der Sie programmieren, wahrscheinlich nicht entkommen:

// will most likely result in an "Illegal escape character" error
String wrongPattern = "\.";
// will result in the string "\."
String correctPattern = "\\.";

All diese Flucht kann sehr verwirrend werden. Wenn die Sprache, mit der Sie arbeiten, unformatierte Zeichenfolgen unterstützt , sollten Sie diese verwenden, um die Anzahl der Backslashes zu verringern, aber nicht alle Sprachen (insbesondere Java). Glücklicherweise gibt es eine Alternative, die manchmal funktioniert:

String correctPattern = "[.]";

Für eine Regex-Engine \.und [.]genau das Gleiche bedeuten. Beachten Sie, dass dies nicht in jedem Fall funktioniert, wie z. B. newline ( \\n), offene eckige Klammer ( \\[) und Backslash ( \\\\oder [\\]).

Ein Hinweis zu übereinstimmenden Zahlen

(Hinweis: Es ist schwieriger als Sie denken)

Das Abgleichen einer Zahl ist eines der Dinge, von denen Sie denken, dass sie mit Regex recht einfach sind, aber es ist tatsächlich ziemlich schwierig. Schauen wir uns Stück für Stück Ihren Ansatz an:

[-+]?

Passen Sie eine optionale -oder+

[0-9]*

Entspricht 0 oder mehr aufeinander folgenden Ziffern

\.?

Passen Sie eine optionale .

[0-9]*

Entspricht 0 oder mehr aufeinander folgenden Ziffern

Erstens können wir diesen Ausdruck ein wenig bereinigen, indem wir eine Zeichenklassen-Kurzform für die Ziffern verwenden (beachten Sie, dass dies auch für das oben erwähnte Escape-Problem anfällig ist):

[0-9] = \d

Ich werde \dunten verwenden, aber denken Sie daran, dass es das gleiche bedeutet wie [0-9]. (Nun, tatsächlich \dstimmen in einigen Engines die Ziffern aller Skripte überein, sodass mehr als [0-9]die Ziffern übereinstimmen , aber das ist in Ihrem Fall wahrscheinlich nicht von Bedeutung.)

Wenn Sie sich das genau ansehen, werden Sie feststellen, dass jeder einzelne Teil Ihres Musters optional ist . Dieses Muster kann mit einer Zeichenfolge mit einer Länge von 0 übereinstimmen. eine Zeichenfolge, die nur aus +oder besteht -; oder eine Zeichenfolge, die nur aus a besteht .. Dies ist wahrscheinlich nicht das, was Sie beabsichtigt haben.

Um dies zu beheben, ist es hilfreich, zunächst Ihren regulären Ausdruck mit der minimal erforderlichen Zeichenfolge zu "verankern", wahrscheinlich mit einer einzelnen Ziffer:

\d+

Jetzt wollen wir den Dezimalteil hinzufügen, aber er geht nicht dahin, wo Sie denken, dass es sein könnte:

\d+\.?\d* /* This isn't quite correct. */

Dies entspricht weiterhin Werten wie 123.. Schlimmer noch, es hat einen Hauch von Bösem . Der Zeitraum ist optional, dh Sie haben zwei wiederholte Klassen nebeneinander ( \d+und \d*). Dies kann tatsächlich gefährlich sein, wenn es falsch verwendet wird und Ihr System für DoS-Angriffe geöffnet wird.

Um dies zu beheben, müssen wir den Punkt nicht als optional behandeln, sondern nach Bedarf behandeln (um die wiederholten Zeichenklassen zu trennen) und stattdessen den gesamten Dezimalteil optional machen:

\d+(\.\d+)? /* Better. But... */

Das sieht jetzt besser aus. Wir benötigen einen Zeitraum zwischen der ersten und der zweiten Ziffernfolge, aber es gibt einen schwerwiegenden Fehler: Wir können nicht übereinstimmen, .123da jetzt eine führende Ziffer erforderlich ist.

Dies ist eigentlich ziemlich einfach zu beheben. Anstatt den "dezimalen" Teil der Zahl optional zu machen, müssen wir ihn als eine Folge von Zeichen betrachten: 1 oder mehr Zahlen, denen ein Präfix vorangestellt werden .kann, denen 0 oder mehr Zahlen vorangestellt werden können:

(\d*\.)?\d+

Jetzt fügen wir einfach das Zeichen hinzu:

[+-]?(\d*\.)?\d+

Natürlich sind diese Schrägstriche in Java ziemlich ärgerlich, daher können wir sie in unseren Langform-Zeichenklassen ersetzen:

[+-]?([0-9]*[.])?[0-9]+

Matching versus Validating

Dies ist in den Kommentaren ein paar Mal aufgetaucht, daher füge ich einen Nachtrag zum Matching versus Validieren hinzu.

Das Ziel des Matchings ist es, einen Inhalt innerhalb der Eingabe zu finden (die "Nadel im Heuhaufen"). Das Ziel der Validierung besteht darin, sicherzustellen, dass die Eingabe in einem erwarteten Format vorliegt.

Regexes stimmen naturgemäß nur mit Text überein . Bei einigen Eingaben finden sie entweder passenden Text oder nicht. Durch "Einrasten" eines Ausdrucks am Anfang und Ende der Eingabe mit Ankertags ( ^und $) können wir jedoch sicherstellen, dass keine Übereinstimmung gefunden wird, es sei denn, die gesamte Eingabe stimmt mit dem Ausdruck überein, wobei zur Validierung Regexes verwendet werden .

Der oben beschriebene reguläre Ausdruck ( [+-]?([0-9]*[.])?[0-9]+) stimmt mit einer oder mehreren Zahlen innerhalb einer Zielzeichenfolge überein . Also gegeben die Eingabe:

apple 1.34 pear 7.98 version 1.2.3.4

Die Regex paßt auf 1.34, 7.98, 1.2, .3und .4.

Um zu überprüfen, ob eine bestimmte Eingabe eine Zahl und nichts anderes als eine Zahl ist, "fangen" Sie den Ausdruck am Anfang und Ende der Eingabe, indem Sie ihn in Ankertags einschließen:

^[+-]?([0-9]*[.])?[0-9]+$

Dies findet nur dann eine Übereinstimmung, wenn die gesamte Eingabe eine Gleitkommazahl ist, und findet keine Übereinstimmung, wenn die Eingabe zusätzliche Zeichen enthält. Bei der Eingabe 1.2wird also eine Übereinstimmung gefunden, es werden jedoch apple 1.2 pearkeine Übereinstimmungen gefunden.

Beachten Sie, dass einige regex Motoren haben eine validate, isMatchoder eine ähnliche Funktion, die im Wesentlichen das tut , was ich automatisch beschrieben habe, zurückkehrt , truewenn eine Übereinstimmung gefunden wird und falsewenn keine Übereinstimmung gefunden wird. Beachten Sie auch, dass Sie mit einigen Engines Flags setzen können, die die Definition von ^und ändern und $mit dem Anfang / Ende einer Zeile und nicht mit dem Anfang / Ende der gesamten Eingabe übereinstimmen. Dies ist normalerweise nicht die Standardeinstellung, aber halten Sie Ausschau nach diesen Flags.


2
JDB, danke und ich hoffe du bist noch da! Ich lese Ihren Beitrag in Zukunft :) Ihre Antwort kümmert sich sicherlich um 0,24 und 2,2 und verbietet 4.2.44 korrekt. Alle mit regex101.com getesteten Daten verbieten jedoch 123. Was , wie Sie sagen, akzeptabel sein kann (und ich denke es) ist!). Ich kann dies beheben, indem ich Ihren Ausdruck in [- +]? (\ D * [.])? \ D * ändere (beachten Sie * am Ende anstelle von +), aber dann verrückte Dinge wie. (Ihr zweites Beispiel) sind erlaubt. Wie auch immer, um meinen Kuchen zu haben und ihn auch zu essen?
Dave

2
@ Dave -\d+(\.\d*)?|\.\d+
JDB erinnert sich noch an Monica

/[-+]?(\d*[.])?\d+/.test("1.bc") // returns true
Yeouuu

1
@yeouuu ja, weil 1.passt. Fügen Sie ^und $am Anfang und Ende des regulären Ausdrucks hinzu, wenn Sie nur dann übereinstimmen möchten, wenn die gesamte Eingabe übereinstimmt.
JDB erinnert sich noch an Monica

5
Floats können Exponenten haben oder NaN / Inf sein, daher würde ich dies verwenden : [-+]?(([0-9]*[.]?[0-9]+([ed][-+]?[0-9]+)?)|(inf)|(nan)), e / d für Float / Float mit doppelter Genauigkeit. Vergessen Sie nicht eine Faltschachtel zum Regex
Markus Schmassmann

23

Ich denke nicht, dass eine der Antworten auf dieser Seite zum Zeitpunkt des Schreibens richtig ist (auch viele andere Vorschläge an anderer Stelle auf SO sind ebenfalls falsch). Die Komplikation besteht darin, dass Sie alle folgenden Möglichkeiten erfüllen müssen:

  • Kein Dezimalpunkt (dh ein ganzzahliger Wert)
  • Ziffern sowohl vor als auch nach dem Dezimalpunkt (z 0.35, 22.165)
  • Stellen vor dem Komma nur (zB 0., 1234.)
  • Stellen nach dem Komma nur (zB .0, .5678)

Gleichzeitig müssen Sie sicherstellen, dass irgendwo mindestens eine Ziffer vorhanden ist, dh Folgendes ist nicht zulässig:

  • ein Dezimalpunkt für sich
  • ein vorzeichenbehafteter Dezimalpunkt ohne Ziffern (dh +.oder -.)
  • +oder -alleine
  • eine leere Zeichenfolge

Dies scheint zunächst schwierig zu sein, aber eine Möglichkeit, Inspiration zu finden, besteht darin, in der OpenJDK-Quelle nach der java.lang.Double.valueOf(String)Methode zu suchen (beginnen Sie unter http://hg.openjdk.java.net/jdk8/jdk8/jdk , klicken Sie auf "Durchsuchen" und navigieren Sie nach unten /src/share/classes/java/lang/und finde die DoubleKlasse). Der lange reguläre Ausdruck, den diese Klasse enthält, bietet verschiedene Möglichkeiten, die das OP wahrscheinlich nicht in Betracht gezogen hat, ignoriert jedoch der Einfachheit halber die Teile, die sich mit NaN, Unendlichkeit, hexadezimaler Notation und Exponenten befassen, und verwendet \danstelle der POSIX-Notation Mit einer einzelnen Ziffer kann ich die wichtigen Teile der Regex für eine vorzeichenbehaftete Gleitkommazahl ohne Exponenten reduzieren auf:

[+-]?((\d+\.?\d*)|(\.\d+))

Ich glaube nicht, dass es eine Möglichkeit gibt, die (...)|(...)Konstruktion zu vermeiden, ohne etwas zuzulassen, das keine Ziffern enthält, oder eine der Möglichkeiten zu verbieten, die keine Ziffern vor dem Dezimalpunkt oder keine Ziffern danach haben.

In der Praxis müssen Sie natürlich auf nachgestellte oder vorangestellte Leerzeichen achten, entweder im regulären Ausdruck selbst oder im Code, der ihn verwendet.


Wenn Sie die Anforderung hinzufügen, Zahlen wie zu entsprechen 123., dann ja ... der Schalter oder ist die einzige Lösung, wie ich in einem Kommentar zu meinem ursprünglichen Beitrag hervorgehoben habe.
JDB erinnert sich noch an Monica

1
Diese und alle / die meisten anderen Antworten ignorieren, dass ein Float einen Exponenten haben kann.
NateS

1
@NateS Das stimmt, ich habe geschrieben, "der Einfachheit halber habe ich die Teile ignoriert, die sich mit NaN, Unendlichkeit, hexadezimaler Notation und Exponenten befassen", weil dies dem Umfang der OP-Frage zu entsprechen scheint. Es gibt vollständigere Implementierungen, einschließlich der, die ich im JDK-Quellcode gefunden habe.
pkeller

1
Kann der Regex [+-]?((?=\.?\d)\d*\.?\d*)verwendet werden, um den Wechsel zu vermeiden? Es verwendet einen Lookahead ...
4esn0k

1
@ 4esn0k Netter Regex! Ich habe damit herumgespielt und es funktioniert. Ich habe zwei Vorbehalte: (1) Nicht alle Regex-Engines unterstützen Behauptungen mit einer Breite von Null (obwohl die meisten modernen AFAIK-Assists), und (2) der Ausblick ist nur eine Abwechslung mit einem anderen Namen: Der Motor muss noch etwas ausprobieren und zurückverfolgen, wenn es nicht funktioniert. Haben Sie trotzdem eine Gegenstimme für eine sehr nette Idee.
pkeller

7

Was Sie brauchen ist:

[\-\+]?[0-9]*(\.[0-9]+)?

Ich bin dem "+" und "-" Zeichen entkommen und habe die Dezimalstelle mit den folgenden Ziffern gruppiert, da so etwas wie "1". ist keine gültige Nummer.

Mit den Änderungen können Sie Ganzzahlen und Gleitkommazahlen abgleichen. beispielsweise:

0
+1
-2.0
2.23442

Das Problem mit diesem Ausdruck ist, dass .1dies nicht zulässig wäre, obwohl eine solche Eingabe allgemein als korrekt anerkannt wird.
JDB erinnert sich noch an Monica

Dies akzeptiert jetzt Zeichenfolgen mit der Länge Null -und +, die keine Zahlen sind. Regex ist schwierig! :)
JDB erinnert sich noch an Monica

Dies beantwortet auch nicht die eigentliche Frage des OP, die \.nicht funktioniert.
JDB erinnert sich noch an Monica

7

Ich möchte übereinstimmen, was die meisten Sprachen als gültige Zahlen betrachten (Ganzzahl und Gleitkommazahlen):

  • '5' / '-5'

  • '1.0' / '1.' / '.1' / '-1.' / '-.1'

  • '0.45326e+04', '666999e-05', '0.2e-3', '-33.e-1'

Anmerkungen:

  • preceding sign of number ('-' or '+') is optional

  • '-1.' and '-.1' are valid but '.' and '-.' are invalid

  • '.1e3' is valid, but '.e3' and 'e3' are invalid

Um beide '1' zu unterstützen. und '.1' benötigen wir einen ODER-Operator ('|'), um sicherzustellen, dass wir '.' vom Matching.

[+-]?+/- singen ist optional, da ?0 oder 1 Übereinstimmungen bedeuten

( Da wir 2 Unterausdrücke haben, müssen wir sie in Klammern setzen

\d+([.]\d*)?(e[+-]?\d+)? Dies gilt für Zahlen, die mit einer Ziffer beginnen

| trennt Unterausdrücke

[.]\d+(e[+-]?\d+)? Dies gilt für Zahlen, die mit '.' beginnen.

) Ende der Ausdrücke

  • Für Zahlen, die mit '.' Beginnen.

[.] Das erste Zeichen ist ein Punkt (in Klammern oder ein Platzhalterzeichen).

\d+ eine oder mehrere Ziffern

(e[+-]?\d+)? Dies ist eine optionale wissenschaftliche Notation (0 oder 1 Übereinstimmungen aufgrund der Endung '?')

  • Für Zahlen, die mit einer Ziffer beginnen

\d+ eine oder mehrere Ziffern

([.]\d*)? Optional können wir ein Punktzeichen mit null oder mehr Ziffern danach haben

(e[+-]?\d+)? Dies ist eine optionale wissenschaftliche Notation

  • Wissenschaftliche Schreibweise

e Literal, das den Exponenten angibt

[+-]? optionales Exponentenzeichen

\d+ eine oder mehrere Ziffern

Alle zusammen:

[+-]?(\d+([.]\d*)?(e[+-]?\d+)?|[.]\d+(e[+-]?\d+)?)

Auch zu akzeptieren E:

[+-]?(\d+([.]\d*)?([eE][+-]?\d+)?|[.]\d+([eE][+-]?\d+)?)

( Testfälle )


4

Dies ist einfach: Sie haben Java verwendet und sollten \\.stattdessen verwenden \.(Suche nach Zeichen, die in Java entkommen).


Sie haben wahrscheinlich Recht ... Die Fehlermeldung sieht eher aus wie ein Programmiersprachen-Syntaxfehler als wie ein Regex-Parser-Fehler.
JDB erinnert sich noch an Monica

3

Dieser hat für mich gearbeitet:

(?P<value>[-+]*\d+\.\d+|[-+]*\d+)

Sie können auch diesen verwenden (ohne benannten Parameter):

([-+]*\d+\.\d+|[-+]*\d+)

Verwenden Sie zum Testen einen Online-Regex-Tester (z. B. Regex101).


2
^[+]?([0-9]{1,2})*[.,]([0-9]{1,1})?$

Dies wird übereinstimmen:

  1. 1.2
  2. 12.3
  3. 1,2
  4. 12,3

Während dieser Code - Schnipsel zu begrüßen ist, und etwas Hilfe bieten kann, würde es erheblich verbessert , wenn es eine Erklärung enthalten von , wie und warum Dies löst das Problem. Denken Sie daran, dass Sie in Zukunft die Frage für die Leser beantworten, nicht nur für die Person, die jetzt fragt! Bitte bearbeiten Sie Ihre Antwort, um eine Erklärung hinzuzufügen, und geben Sie an, welche Einschränkungen und Annahmen gelten.
Toby Speight

oh thnks, ich bin lokking dafür
Serg Burlaka

0
[+-]?(([1-9][0-9]*)|(0))([.,][0-9]+)?

[+-]? - optionales Leitzeichen

(([1-9][0-9]*)|(0)) - Ganzzahl ohne führende Null, einschließlich einzelner Null

([.,][0-9]+)? - optionaler Bruchteil


1
Geben Sie weitere Informationen - für Leute, die die Regexps nicht kennen, sind es Hyerogliphs. Für Leute, die sie kennen, brauchen sie es nicht.
Peter - Wiedereinsetzung Monica

0

In C ++ mit der Regex-Bibliothek

Die Antwort würde so aussehen:

[0-9]?([0-9]*[.])?[0-9]+

Beachten Sie, dass ich das Zeichensymbol nicht nehme. Wenn Sie es mit dem Zeichensymbol möchten, geht es folgendermaßen vor:

[+-]?([0-9]*[.])?[0-9]+

Dies trennt auch eine reguläre Zahl oder eine Dezimalzahl.


0

In c-Notation kann die Float-Nummer in folgenden Formen auftreten:

  1. 123
  2. 123.
  3. 123,24
  4. .24
  5. 2e-2 = 2 · 10 pow-2 = 2 · 0,1
  6. 4E + 4 = 4 · 10 pow 4 = 4 · 10 000

Um einen regulären Float-Ausdruck zu erstellen, erstelle ich zuerst eine "int-reguläre Ausdrucksvariable":

(([1-9][0-9]*)|0) will be int

Jetzt werde ich kleine Stücke von Float Regular Expresion schreiben - die Lösung besteht darin, diese Stücke mit "|" zu verknüpfen oder zu simbolen.

Brocken:

- (([+-]?{int}) satysfies case 1
- (([+-]?{int})"."[0-9]*)  satysfies cases 2 and 3
- ("."[0-9]*) satysfies case 4
- ([+-]?{int}[eE][+-]?{int}) satysfies cases 5 and 6

Endlösung (kleine Stücke zusammenfassen):

(([+-]?{int})|(([+-]?{int})"."[0-9]*)|("."[0-9]*)|([+-]?{int}[eE][+-]?{int})


-1

für Javascript

const test = new RegExp('^[+]?([0-9]{0,})*[.]?([0-9]{0,2})?$','g');

Welches würde für 1.23 1234,22 0 0,12 12 funktionieren

Sie können die Teile in ändern {}, um unterschiedliche Ergebnisse in Bezug auf die Dezimallänge und die Vorderseite der Dezimalstelle zu erhalten. Dies wird in Eingaben verwendet, um eine Zahl einzugeben und jede Eingabe zu überprüfen, während Sie nur eingeben, was passiert.

Durch die Nutzung unserer Website bestätigen Sie, dass Sie unsere Cookie-Richtlinie und Datenschutzrichtlinie gelesen und verstanden haben.
Licensed under cc by-sa 3.0 with attribution required.