Exemple d'analyse du problème de visibilité des variables Java
Question : La synchronisation et la mise en veille peuvent-elles également atteindre l'objectif de visibilité volatile des threads ? La description générale du problème est la suivante :
package com.test;
importer java.util.concurrent.TimeUnit;
test de classe publique1 {
le booléen statique privé est = vrai ;
public static void main (String[] args) {
nouveau fil de discussion (nouveau Runnable() {
@Override
public void run() {
int je = 0;
while(test1.is){
je++;
1 //synchronisé (ceci) { } actualisera de force la valeur variable de la mémoire principale dans la pile de threads ?
2 //System.out.println("1"); println est synchronisé et actualisera de force la valeur variable de la mémoire principale dans la pile de threads ?
3 //le sommeil rechargera la valeur de la mémoire principale ?
// essaye {
// TimeUnit.MICROSECONDS.sleep(1);
// }catch (InterruptedException e) {
// e.printStackTrace();
// }
}
}
}).start();
essayez {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
nouveau fil de discussion (nouveau Runnable() {
@Override
public void run() {
is = false; //Set est sur false pour que le fil ci-dessus termine la boucle while
}
}).start();
}
}
Q : Pourquoi le programme dans son intégralité ne se termine-t-il pas ? Pourquoi le programme se termine-t-il lorsque vous décommentez l'un des blocs de code (1, 2, 3) ? La synchronisation actualisera-t-elle de force les valeurs des variables de mémoire dans la pile de threads ? Que fera sleep ?
Implique une explication des connaissances
volatile : Ce mot-clé garantit la visibilité des variables dans les threads. Tous les threads qui accèdent aux variables modifiées par volatile doivent les lire dans la mémoire principale avant de les utiliser, et les réécrire dans la mémoire principale immédiatement après la modification de la mémoire de travail, garantissant que les autres threads Visibilité, le mot-clé avec le même effet est définitif.
synchronisé : toutes les opérations de synchronisation doivent garantir 1. Atomicité 2. Visibilité, de sorte que les modifications qui se produisent dans le bloc synchronisé seront immédiatement réécrites dans la mémoire principale
sleep : Cette méthode n'abandonnera que le temps d'exécution du processeur et ne libérera pas le verrou.
Analyse du problème
Q1 : Pourquoi le programme ne se termine-t-il pas après avoir commenté le code ?
A1 : étant donné que la valeur variable de boolean is=true est chargée dans sa propre mémoire de travail par le thread précédent (appelé thread A), une fois que le thread suivant (appelé thread B) a modifié boolean is=false, il se peut qu'il ne le soit pas. être écrit immédiatement dans la mémoire principale (mais dans cette question, il doit être écrit immédiatement dans la mémoire principale, car le thread se fermera après l'exécution de is=false). Même s'il est écrit immédiatement dans la mémoire principale, le thread A ne peut pas le faire. sera immédiatement chargé dans la mémoire de travail, afin que le programme continue. Ne se terminera-t-il pas ? C'est ce à quoi la plupart d'entre nous pensent, mais en fait, la JVM a été optimisée dans une large mesure pour le niveau matériel actuel, assurant essentiellement la synchronisation en temps opportun de la mémoire de travail et de la mémoire principale dans une large mesure, ce qui équivaut à l'utilisation de données volatiles. par défaut. Mais seulement au maximum ! Lorsque les ressources CPU sont toujours occupées, la synchronisation entre la mémoire de travail et la mémoire principale, c'est-à-dire la visibilité des variables, ne sera pas aussi ponctuelle ! La conclusion sera vérifiée plus tard.
Q2 : Pourquoi le programme se termine-t-il uniquement lorsqu'un bloc de code (1, 2, 3) dans le commentaire n'est pas commenté ?
A2 : Les codes avec les numéros de ligne 1 et 2 ont une caractéristique commune, c'est-à-dire qu'ils impliquent tous des verrous de synchronisation synchronisés. Ainsi, comme l'auteur de la question l'a deviné, la synchronisation forcera l'actualisation des valeurs des variables dans la mémoire principale. pile de threads ? , et la méthode sleep actualisera également les valeurs des variables dans la mémoire principale dans la pile de threads ? En fait, nous avons dit plus tôt que synchronisé assurera uniquement la visibilité des variables dans le bloc synchronisé, et la variable is n'est pas dans le bloc synchronisé, donc ce n'est évidemment pas dû à cela. Ensuite, nous ajoutons le code suivant après le code i++; :
pour(int k=0;k<100000;k++){
nouvel objet ();
}
Si vous l'exécutez à nouveau, le programme se terminera immédiatement ! Pourquoi? Dans A1 ci-dessus, nous avons dit que même avec l'optimisation JVM, lorsque le CPU est toujours occupé, la visibilité des données ne peut pas être bien garantie, tout comme le programme ci-dessus continue de boucler pour faire des opérations i++ et pourquoi le programme. arrêter après avoir ajouté le code ci-dessus ? Parce que pour un grand nombre de nouvelles opérations Object(), le CPU n'est plus la principale opération chronophage. Le vrai temps devrait résider dans l'allocation de mémoire (car la vitesse de traitement du CPU est évidemment plus rapide que la mémoire). , sinon il n'y aurait pas de registres CPU. ), donc une fois le CPU inactif, il suivra le benchmark d'optimisation JVM pour assurer la visibilité des données le plus rapidement possible, synchronisant ainsi la variable is de la mémoire principale vers la mémoire de travail, conduisant finalement à la fin du programme. C'est pourquoi la méthode sleep() n'implique pas d'opérations de synchronisation, mais le programme peut toujours être terminé, car la méthode sleep() libérera le CPU mais pas le verrou !
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Outils d'IA chauds

Undresser.AI Undress
Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover
Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool
Images de déshabillage gratuites

Clothoff.io
Dissolvant de vêtements AI

AI Hentai Generator
Générez AI Hentai gratuitement.

Article chaud

Outils chauds

Bloc-notes++7.3.1
Éditeur de code facile à utiliser et gratuit

SublimeText3 version chinoise
Version chinoise, très simple à utiliser

Envoyer Studio 13.0.1
Puissant environnement de développement intégré PHP

Dreamweaver CS6
Outils de développement Web visuel

SublimeText3 version Mac
Logiciel d'édition de code au niveau de Dieu (SublimeText3)

Guide du nombre parfait en Java. Nous discutons ici de la définition, comment vérifier le nombre parfait en Java ?, des exemples d'implémentation de code.

Guide du générateur de nombres aléatoires en Java. Nous discutons ici des fonctions en Java avec des exemples et de deux générateurs différents avec d'autres exemples.

Guide de Weka en Java. Nous discutons ici de l'introduction, de la façon d'utiliser Weka Java, du type de plate-forme et des avantages avec des exemples.

Guide du nombre de Smith en Java. Nous discutons ici de la définition, comment vérifier le numéro Smith en Java ? exemple avec implémentation de code.

Dans cet article, nous avons conservé les questions d'entretien Java Spring les plus posées avec leurs réponses détaillées. Pour que vous puissiez réussir l'interview.

Java 8 présente l'API Stream, fournissant un moyen puissant et expressif de traiter les collections de données. Cependant, une question courante lors de l'utilisation du flux est: comment se casser ou revenir d'une opération FOREAK? Les boucles traditionnelles permettent une interruption ou un retour précoce, mais la méthode Foreach de Stream ne prend pas directement en charge cette méthode. Cet article expliquera les raisons et explorera des méthodes alternatives pour la mise en œuvre de terminaison prématurée dans les systèmes de traitement de flux. Lire plus approfondie: Améliorations de l'API Java Stream Comprendre le flux Forach La méthode foreach est une opération terminale qui effectue une opération sur chaque élément du flux. Son intention de conception est

Guide de TimeStamp to Date en Java. Ici, nous discutons également de l'introduction et de la façon de convertir l'horodatage en date en Java avec des exemples.

Java est un langage de programmation populaire qui peut être appris aussi bien par les développeurs débutants que par les développeurs expérimentés. Ce didacticiel commence par les concepts de base et progresse vers des sujets avancés. Après avoir installé le kit de développement Java, vous pouvez vous entraîner à la programmation en créant un simple programme « Hello, World ! ». Une fois que vous avez compris le code, utilisez l'invite de commande pour compiler et exécuter le programme, et « Hello, World ! » s'affichera sur la console. L'apprentissage de Java commence votre parcours de programmation et, à mesure que votre maîtrise s'approfondit, vous pouvez créer des applications plus complexes.
