Der Erwerb und die Freigabe von Sperren sind sowohl implizit als auch vollständig der JVM überlassen, um uns beim Betrieb zu helfen Der erste Wissenspunkt, den es zu lernen gilt, ist die Objektstruktur von Java, da synchronisierte Sperren in Java-Objekten gespeichert sind. Die Java-Objektstruktur ist wie in der folgenden Abbildung dargestellt:
Sie können deutlich erkennen, dass Java-Objekte aus drei Objekten bestehen Teile sind Objekt-Header, Instanzdaten und Fülldaten. Als nächstes führen wir eine einfache Analyse der Objektstruktur durch:
mark-down: Das Objektmarkierungsfeld belegt 8 Bytes Wird zum Speichern von Informationen über die Flag-Bits der Sperre und andere Informationen verwendet. Wie aus der Abbildung ersichtlich ist, gibt es Hash-Werte, Lightweight-Lock-Flag-Bits, Bias-Lock-Flag-Bits usw.
Klassenzeiger: Der Typzeiger des Klassenobjekts, der der Zeiger ist, zu der das aktuelle Objekt gehört. Standardmäßig belegt jdk1.8 nach dem Einschalten komprimierter Zeiger 4 Bytes und nach dem Ausschalten 8 Bytes komprimierte Zeiger.
Tatsächliche Daten des Objekts: Dieser Teil umfasst alle Mitgliedsvariablen des Objekts. Die Größe wird durch jede Mitgliedsvariable bestimmt. Beispielsweise belegt Byte 1 Byte, Int belegt 4 Byte.
Füllen Sie es aus: Dieser Teil des Inhalts dient nur der Vervollständigung des Speicherplatzes und dient als Platzhalter, da das Speicherverwaltungssystem der virtuellen HotSpot-Maschine erfordert, dass die Startadresse des Objekts ein ganzzahliges Vielfaches von 8 Bytes sein muss. Wenn die Objektinstanz nicht ausgerichtet ist, muss sie daher gefüllt werden, um sie zu ergänzen.
In der Mark-Down-Sperrtypmarkierung können Sie sehen, dass es insgesamt fünf Typen gibt, nämlich sperrfrei, vorgespannte Sperre, leichte Sperre, schwere Sperre und GC-Markierung. Wenn Sie also nur 2-Bit verwenden mark Es kann nicht vollständig ausgedrückt werden, daher wird ein Bias-Lock-Tag eingeführt, das heißt, 001 bedeutet keine Sperre und 101 bedeutet Bias-Lock.
Die Objektstruktur wird oben vorgestellt. Sie können sehen, dass in Mark-down unterschiedliche Sperrinformationen gespeichert werden. Wenn der Sperrstatus Schwergewichtssperre (10) ist, wird in Mark-Down ein Zeiger auf Monitor gespeichert. down Zeiger auf das Objekt. Dieses Monitorobjekt wird auch als Monitorsperre bezeichnet.
Der synchronisierte Betriebsmechanismus besteht darin, dass die JVM automatisch zu einer geeigneten Sperrimplementierung wechselt, wenn sie unterschiedliche Wettbewerbssituationen in gemeinsam genutzten Objekten erkennt. (Viele Orte sagen, dass Sperren nur aktualisiert und nicht herabgestuft werden können. Tatsächlich ist diese Aussage falsch. Im Buch „The Art of Java Concurrent Programming“ heißt es, dass voreingenommene Sperren auf eine Sperre herabgestuft werden können. Freier Zustand, und es wird auch voreingenommener Sperrenwiderruf genannt). Derzeit gibt es drei verschiedene Monitor-Implementierungen, nämlich voreingenommene Sperren, leichte Sperren und schwere Sperren. Wenn ein Thread einen Monitor hält, erhält er die Sperre. Monitor in Java wird basierend auf ObjectMonitor von C++ implementiert. Zu seinen Hauptmitgliedern gehören:public static void add() { synchronized (Demo.class) { counter++; } }Dekompilieren Sie sie und sehen Sie sich den Code an: javap -v -p Demo
public static void add(); descriptor: ()V flags: ACC_PUBLIC, ACC_STATIC , ACC_SYNCHRONIZED Code: stack=2, locals=2, args_size=0 0: ldc #12 // class 2: dup 3: astore_0 4: monitorenter 5: getstatic #10 // Field counter:I 8: iconst_1 9: iadd 10: putstatic #10 // Field counter:I 13: aload_0 14: monitorexit 15: goto 23 18: astore_1 19: aload_0 20: monitorexit 21: aload_1 22: athrow 23: return Exception table:Sie können sehen, dass es offensichtlich zwei Anweisungen gibt, die sich auf den Monitor beziehen:
monitorenter: Nachdem festgestellt wurde, dass es das Synchronisationsflag ACC_SYNCHRONIZED hat, hat der Thread, der diese Methode zuerst aufruft, zuerst den Besitzer des Monitors. Zu diesem Zeitpunkt ist der Zähler +1
monitorexit: Wenn die Ausführung abgeschlossen ist und beendet, der Zähler -1, kehrt zu 0 zurück und wird durch andere ersetzt. Der eingehende Thread erhält
Sichtbarkeit bedeutet, dass, wenn mehrere Threads auf dieselbe Variable zugreifen und ein Thread den Wert dieser Variablen ändert, andere Threads kann es sofort spüren und den veränderten Wert sehen. Die Thread-Sichtbarkeit hängt eng mit JMM zusammen. Im nächsten Artikel werden wir untersuchen, wie man das Schlüsselwort volatile verwendet, um Sichtbarkeit zu erreichen
Und Synchronized hat Sichtbarkeit, weil es die folgende Semantik zum Sperren und Freigeben von Sperren hat:
Bevor ein Thread gesperrt wird, muss er gelöscht werden Wert der gemeinsam genutzten Variablen im Arbeitsspeicher und liest dabei den neuesten Wert der gemeinsam genutzten Variablen aus dem Hauptspeicher.
Wenn der Thread die Sperre aufhebt, muss der Wert der gemeinsam genutzten Variablen im Hauptspeicher aktualisiert werden.
Die Sichtbarkeit der Synchronisierung hängt von der Mutex-Implementierung des Betriebssystemkerns ab, was dem Sperren und Entsperren in der JVM entspricht. Beim Verlassen des Codeblocks müssen die gemeinsam genutzten Variablen im Hauptspeicher aktualisiert werden das Schlüsselwort volatile. Die Sichtbarkeit von Schlüsselwörtern hängt von Speicherbarrieren (auch Memory Fences genannt) ab.
as-if-serial soll sicherstellen, dass unabhängig davon, wie der Compiler und der Prozessor die Anweisungen zur Leistungsoptimierung neu anordnen, die Richtigkeit der laufenden Ergebnisse unter Single-Thread sichergestellt werden muss. Das heißt: Wenn Sie innerhalb dieses Threads beobachten, sind alle Vorgänge in Ordnung, Wenn Sie einen anderen Thread in einem Thread beobachten, sind alle Vorgänge nicht in Ordnung.
Beachten Sie, dass sich die Reihenfolge hier von der flüchtigen Reihenfolge unterscheidet. Es ist nicht flüchtig, was eine Neuordnung der Anweisungen verhindert.
Das Konzept einer Wiedereintrittssperre ist sehr einfach, das heißt, ein Thread kann die Objektsperre, die er hält, mehrmals erwerben. Bei dieser Art von Sperre muss die gleiche Anzahl von Sperren freigegeben werden die gleiche Sperre. Im synchronisierten Sperrobjekt gibt es einen Zähler, der die Häufigkeit des Erwerbs der Sperre aufzeichnet, also die Anzahl der Wiedereintritte.
synchronisierte Schlösser haben vier alternative Upgrade-Zustände: kein Schloss, voreingenommenes Schloss, leichtes Schloss und schweres Schloss. Diese Zustände eskalieren allmählich mit der Wettbewerbssituation. Ein vollständiges Schloss-Upgrade-Diagramm wird später hinzugefügt.
Das obige ist der detaillierte Inhalt vonWas ist das Prinzip von Synchronized in Java?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!