IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Java Discussion :

[Multi-Threads] Erreur java.util.ConcurrentModificationException lors de parcours d'une liste


Sujet :

Java

  1. #1
    Membre régulier
    Profil pro
    Inscrit en
    Décembre 2006
    Messages
    279
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Décembre 2006
    Messages : 279
    Points : 102
    Points
    102
    Par défaut [Multi-Threads] Erreur java.util.ConcurrentModificationException lors de parcours d'une liste
    Bonjour,

    J'ai une liste initial contenant un ensemble de nom de fichiers (supposant 20).

    J'ai un ensemble de threads (supposant 3) qui traiteront les elements de liste de cette maniere :

    - Chaque thread a une liste, elle contient un ensemble de la liste initial (Thread 1 contient 7, Thread 2 contient 7 et Thread 3 contient 6)
    - Le traitement se fait sur chaque element de la liste du thread

    En ce moment j'ai pu repartir la liste initial sur l'ensemble des listes

    Quand j'esseye de parcourir la liste du thread dans le thread (methode run), j'ai cette exception :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    Exception in thread "Thread-2" java.util.ConcurrentModificationException
    	at java.util.concurrent.CopyOnWriteArrayList$COWSubList.checkForComodification(CopyOnWriteArrayList.java:1127)
    	at java.util.concurrent.CopyOnWriteArrayList$COWSubList.size(CopyOnWriteArrayList.java:1167)
    	at fr.gouv.finances.tools.sftppoller.poller.ThreadReader.run(ThreadReader.java:21)
    La premiere partie de la methode run est :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    Iterator<String> it = getOwnFiles().iterator();
    		while (it.hasNext()) {
    			if (!Poller.toStop) {
    				logger.debug("Taille de la OwnFiles : " + getOwnFiles().size());
    				String fileName = it.next();
                            ...
                            }
                     ...
                    }
    La liste du thread OwnFiles est initialisé de cette maniere :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    private List<String> ownFiles = new CopyOnWriteArrayList<String>();
    Comment résoudre ce probleme ?

    Merci

  2. #2
    Membre chevronné
    Avatar de eulbobo
    Homme Profil pro
    Développeur Java
    Inscrit en
    Novembre 2003
    Messages
    786
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 45
    Localisation : France, Rhône (Rhône Alpes)

    Informations professionnelles :
    Activité : Développeur Java

    Informations forums :
    Inscription : Novembre 2003
    Messages : 786
    Points : 1 993
    Points
    1 993
    Par défaut
    Quand tu travailles avec un CopyOnWriteArray, il faut te dire que quasiment toutes les opérations sur la liste te renvoie une NOUVELLE image de la liste.

    En gros, quand tu demandes un Iterator sur ton array, tu obtiens un iterateur qui pointe sur une copie locale de la liste au moment où tu as demandé l'itérateur. Du coup, ton code n'a pas de raison de planter là

    Par contre, en faisant un .size(), tu faut que tu aies conscience que cet élément fait référence à l'état général de ton objet list (qui peut donc varier)


    Maintenant, l'erreur que tu as n'est pas cohérente avec ton code : ce genre d'erreur ne peut à priori arriver que si tu utilises un subList à un moment ou à un autre (qui te renvoie une liste qui N'EST PAS une CopyOnWriteArray)
    Je ne suis pas mort, j'ai du travail !

  3. #3
    Membre régulier
    Profil pro
    Inscrit en
    Décembre 2006
    Messages
    279
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Décembre 2006
    Messages : 279
    Points : 102
    Points
    102
    Par défaut la methode sublist
    Oui,j'utilise bien une sublist pour remplir les differentes ownListdes threads de cette maniere :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    files.subList(j, nbFilesInThread + j)
    sachant que la liste files est bien du type

  4. #4
    Expert éminent sénior
    Avatar de adiGuba
    Homme Profil pro
    Développeur Java/Web
    Inscrit en
    Avril 2002
    Messages
    13 938
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Développeur Java/Web
    Secteur : Transports

    Informations forums :
    Inscription : Avril 2002
    Messages : 13 938
    Points : 23 190
    Points
    23 190
    Billets dans le blog
    1
    Par défaut
    Salut,


    Les List retournée par subList() correspondent à des vues sur la liste principale.
    Dans le cas d'une CopyOnWriteArrayList, si la liste originale est modifiée la vue est invalide : https://docs.oracle.com/javase/8/doc...bList-int-int-


    Tu veux faire quoi exactement avec tout çà ? Ca me semble un peu lourd d'utiliser une CopyOnWriteArrayList !


    a++

  5. #5
    Membre régulier
    Profil pro
    Inscrit en
    Décembre 2006
    Messages
    279
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Décembre 2006
    Messages : 279
    Points : 102
    Points
    102
    Par défaut
    J'ai redéfini la méthode subList et j'ai plus cette erreur.

    je vous remercie pour votre aide

  6. #6
    Expert éminent sénior
    Avatar de adiGuba
    Homme Profil pro
    Développeur Java/Web
    Inscrit en
    Avril 2002
    Messages
    13 938
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Développeur Java/Web
    Secteur : Transports

    Informations forums :
    Inscription : Avril 2002
    Messages : 13 938
    Points : 23 190
    Points
    23 190
    Billets dans le blog
    1
    Par défaut
    Mouais... mais c'est quoi le but du traitement ? Il doit sûrement y avoir plus simple...


    a++

  7. #7
    Membre régulier
    Profil pro
    Inscrit en
    Décembre 2006
    Messages
    279
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Décembre 2006
    Messages : 279
    Points : 102
    Points
    102
    Par défaut
    Le but du traitement est de traiter une liste contenant un ensemble fichiers avec plusieurs threads.

    Pour cela, la solution qu'on a choisie est de crée une liste pour chaque thread, ces listes sont alimenter en fonction du nombre de la liste principale

  8. #8
    Expert éminent sénior
    Avatar de adiGuba
    Homme Profil pro
    Développeur Java/Web
    Inscrit en
    Avril 2002
    Messages
    13 938
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Développeur Java/Web
    Secteur : Transports

    Informations forums :
    Inscription : Avril 2002
    Messages : 13 938
    Points : 23 190
    Points
    23 190
    Billets dans le blog
    1
    Par défaut
    Mais pourquoi passer par une CopyOnWriteArrayList ? Un liste par thread serait amplement suffisant...

    Sans même parler de l'API de Stream de Java 8.


    a++

  9. #9
    Membre régulier
    Profil pro
    Inscrit en
    Décembre 2006
    Messages
    279
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Décembre 2006
    Messages : 279
    Points : 102
    Points
    102
    Par défaut
    Initialement ce n'etait pas une CopyOnWriteArrayList , c'etait ArrayList

    On chercheons une solution sur le net, j'ai trouvé que c'est recommandé de changé ArrayList par CopyOnWriteArrayList

    Dans ce lien : mettre la CopyOnWriteArrayList

  10. #10
    Expert éminent sénior
    Avatar de adiGuba
    Homme Profil pro
    Développeur Java/Web
    Inscrit en
    Avril 2002
    Messages
    13 938
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Développeur Java/Web
    Secteur : Transports

    Informations forums :
    Inscription : Avril 2002
    Messages : 13 938
    Points : 23 190
    Points
    23 190
    Billets dans le blog
    1
    Par défaut
    Citation Envoyé par facilus68 Voir le message
    Dans ce lien : mettre la CopyOnWriteArrayList
    Je trouve cette solution carrément affreuse !!! C'est sale.


    J'espère que ton jeu de données restera toujours très "petit" car le contraire risquerait de poser problème...


    a++

+ Répondre à la discussion
Cette discussion est résolue.

Discussions similaires

  1. Erreur "java.util.ConcurrentModificationException"
    Par Krisiun dans le forum Langage
    Réponses: 8
    Dernier message: 02/11/2013, 22h53
  2. Message d'erreur : java.util.ConcurrentModificationException
    Par leara500 dans le forum Débuter avec Java
    Réponses: 2
    Dernier message: 02/02/2013, 15h47
  3. Erreur java util.ConcurrentModificationException
    Par dsryam dans le forum Général Java
    Réponses: 9
    Dernier message: 30/01/2011, 15h08
  4. Réponses: 10
    Dernier message: 17/06/2009, 17h05
  5. [JDOM] Exception:Exception in thread "main" java.util.ConcurrentModificationException
    Par solawe dans le forum Format d'échange (XML, JSON...)
    Réponses: 3
    Dernier message: 10/06/2009, 18h33

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo