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

Collection et Stream Java Discussion :

Stream Java 8 : Comment enregistrer dans un fichier proprement ?


Sujet :

Collection et Stream Java

  1. #1
    Modérateur
    Avatar de kolodz
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Avril 2008
    Messages
    2 211
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 36
    Localisation : France, Rhône (Rhône Alpes)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : High Tech - Produits et services télécom et Internet

    Informations forums :
    Inscription : Avril 2008
    Messages : 2 211
    Points : 8 316
    Points
    8 316
    Billets dans le blog
    52
    Par défaut Stream Java 8 : Comment enregistrer dans un fichier proprement ?
    Bonjour,

    En utilisant la "nouvelle" API Stream de Java 8, je suis tombé sur un cas d'utilisation où je n'ai pas trouvé de documentation, mais qu'il me semble important de traiter correctement. A savoir la sauvegarde d'un Stream<String> dans un fichier.

    J'ai bien une version fonctionnelle, mais celle-ci me semble être purement bancale dans la gestion des erreurs.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    package org.k.developpez.forum;
     
    import java.io.IOException;
    import java.nio.charset.Charset;
    import java.nio.file.Files;
    import java.nio.file.Paths;
     
    public class Test {
     
    	public static void main(String[] args) throws IOException {
    		LineConsumer lineConsumer =  new LineConsumer("OutputFile.txt");
    		// On filtre les lignes contenant le text qu'on veux depuis le fichier de base
    		Files.lines(Paths.get("InputFile.txt"), Charset.defaultCharset())
    				.filter(line -> line.contains("Enregistrement technique"))
    				.forEach(lineConsumer);
    		lineConsumer.close();
    	}
    }
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    package org.k.developpez.forum;
     
    import java.io.FileWriter;
    import java.io.IOException;
    import java.util.function.Consumer;
     
    public class LineConsumer implements Consumer<String> {
     
    	private FileWriter writer;
     
    	public LineConsumer(String file) throws IOException {
    		writer = new FileWriter(file, true);			
     
    	}
     
    	@Override
    	public void accept(String line) {
    		//TODO Ici avoir un try catch me pose problème. Cependant, je ne sais pas comment gérer correctement celle-ci au niveau de la classe test.
    		try {
    			writer.write(line + "\r\n");
    		} catch (IOException e) {
    			e.printStackTrace();
    		}
     
    	}
     
    	public void close(){
    		try {
    			writer.flush();
    			writer.close();
    		} catch (IOException e) {
    			e.printStackTrace();
    		}
     
    	}
     
    }
    Citation Envoyé par Exemple de fichier InputFile.txt
    Enregistrement technique Blabla
    Enregistrement fonctionnelle dudule
    Enregistrement technique Titi
    une ligne bizarre
    Si vous avez une idée sur la façon de procéder qui serai potentiellement plus propre, n'hésitez pas !

    Cordialement,
    Patrick Kolodziejczyk.
    Si une réponse vous a été utile pensez à
    Si vous avez eu la réponse à votre question, marquez votre discussion
    Pensez aux FAQs et aux tutoriels et cours.

  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
    Déclare ta propre interface fonctionnelle qui peut renvoyer une exception et le tour est joué !

    Ou une qui wrap le tout dans une RunTimeException, mais tu peux avoir envie d'avoir des exception checked
    Je ne suis pas mort, j'ai du travail !

  3. #3
    Modérateur
    Avatar de joel.drigo
    Homme Profil pro
    Ingénieur R&D - Développeur Java
    Inscrit en
    Septembre 2009
    Messages
    12 430
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 54
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Ingénieur R&D - Développeur Java
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Septembre 2009
    Messages : 12 430
    Points : 29 131
    Points
    29 131
    Billets dans le blog
    2
    Par défaut
    Les streams orientent à faire du unchecked : mais on peut utiliser Files.write(), en collectant auparavant. Je pense que c'est mieux d'ouvrir l'IO stream après avoir fini le traitement du stream de toute manière.
    L'expression "ça marche pas" ne veut rien dire. Indiquez l'erreur, et/ou les comportements attendus et obtenus, et donnez un Exemple Complet Minimal qui permet de reproduire le problème.
    La plupart des réponses à vos questions sont déjà dans les FAQs ou les Tutoriels, ou peut-être dans une autre discussion : utilisez la recherche interne.
    Des questions sur Java : consultez le Forum Java. Des questions sur l'EDI Eclipse ou la plateforme Eclipse RCP : consultez le Forum Eclipse.
    Une question correctement posée et rédigée et vous aurez plus de chances de réponses adaptées et rapides.
    N'oubliez pas de mettre vos extraits de code entre balises CODE (Voir Mode d'emploi de l'éditeur de messages).
    Nouveau sur le forum ? Consultez Les Règles du Club.

  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,


    Dans ce cas précis il serait plus simple d'utiliser directement un PrintStream :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    		try (PrintStream out = new PrintStream("OutputFile.txt")) {
    			Files.lines(Paths.get("InputFile.txt"), Charset.defaultCharset())
    					.filter(line -> line.contains("Enregistrement technique"))
    					.forEach(out::println);
    		}

    Mais ton exemple montre l'un des gros problèmes que les checked-exceptions imposent aux lambdas.
    En effet vu que le code des lambdas n'est pas exécuté au "même moment", il est impossible d'avoir un gestion "checked" des exceptions.

    Du coup le code suivant ne fonctionne pas bien qu'équivalent :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    		try (FileWriter out = new FileWriter("OutputFile.txt", true)) {
    			Files.lines(Paths.get("InputFile.txt"), Charset.defaultCharset())
    					.filter(line -> line.contains("Enregistrement technique"))
    					.map(s -> s + "\r\n")
    					.forEach(out::write);
    		}
    En effet la méthode write() de FileWriter déclare une IOException, ce qui est incompatible avec la déclaration du Consumer<T> attendu par forEach()...



    Les checked-exceptions s'adapte mal aux expressions lambdas...
    Enfin pas seulement le problème était déjà là avant avec les classes anonymes, mais c'était moins visible.


    Bref il serait temps de remettre en cause ce principe...


    a++

  5. #5
    Modérateur
    Avatar de kolodz
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Avril 2008
    Messages
    2 211
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 36
    Localisation : France, Rhône (Rhône Alpes)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : High Tech - Produits et services télécom et Internet

    Informations forums :
    Inscription : Avril 2008
    Messages : 2 211
    Points : 8 316
    Points
    8 316
    Billets dans le blog
    52
    Par défaut
    Merci pour la réponse !
    Et désolé pour le temps de réaction. (Vacances, reprise tout ça....)

    C'est effectivement beaucoup plus propre d'utiliser un [c]PrintStream[\c] qui est beaucoup plus dans la logique des Stream
    Ce qui fait 4 lignes sans besoin de définir un [c]Consumer[\c] maison. C'est à ce niveau que cela me semblait bizarre qu'il y ai pas quelque chose qui règle déjà cette problématique simple. Mais, c'est juste moi qui n'ai pas penser au [c]PrintStream[\c]!

    Cordialement,
    Patrick Kolodziejczyk.
    Si une réponse vous a été utile pensez à
    Si vous avez eu la réponse à votre question, marquez votre discussion
    Pensez aux FAQs et aux tutoriels et cours.

  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
    Citation Envoyé par kolodz Voir le message
    C'est à ce niveau que cela me semblait bizarre qu'il y ai pas quelque chose qui règle déjà cette problématique simple.
    Le PrintStream n'est qu'une solution spécifique.
    Les checked-exceptions sont toujours problématique dans de nombreux cas...

    Il y a eu des discussions sur ce problème, et plusieurs solutions ont été envisagés.
    Mais à chaque fois cela aboutissait à de grosse complication, que ce soit dans l'implémentation des lambdas ou des APIs
    Du coup ils en sont resté à conseiller l'utilisation d'unchecked-exception.


    Le fait est que le problème ne vient pas des lambdas, mais du principe de checked-exception.
    La meilleure solution serait de supprimer cette notion, qui pose plus de problème qu'autre chose...


    a++

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

Discussions similaires

  1. Réponses: 0
    Dernier message: 21/05/2008, 18h39
  2. comment enregistrer dans fichier
    Par aymen14 dans le forum Langage
    Réponses: 2
    Dernier message: 10/05/2008, 22h35
  3. Réponses: 6
    Dernier message: 28/02/2007, 16h18
  4. Réponses: 1
    Dernier message: 28/01/2006, 12h36
  5. comment ne pas enregistrer dans le fichier log?
    Par trotters213 dans le forum MS SQL Server
    Réponses: 14
    Dernier message: 21/03/2005, 14h56

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