Gestion des fichiers et entrée/sortie
Le langage de programmation Java prend en charge les entrées/sorties depuis la toute première version. Cependant, en raison du fort désir de Java pour l'indépendance de la plate-forme, les versions antérieures de la fonctionnalité d'entrée/sortie mettaient l'accent sur la portabilité plutôt que sur la fonctionnalité. En conséquence, ils n'étaient pas toujours faciles à travailler.
Les entrées/sorties Java classiques
La classe File est la pierre angulaire de la manière originale de Java d'effectuer les entrées/sorties de fichier. Cette abstraction peut représenter à la fois des fichiers et des répertoires, mais ce faisant, elle est parfois un peu difficile à gérer et conduit à un code comme celui-ci :
- import java.io.File;
-
- class Main {
- public static void main (String[] args) {
- /* Demande un objet fichier pour représenter le répertoire personnel de l'utilisateur */
- File homedir=new File(System.getProperty("user.home"));
-
- /* Crée un objet pour représenter un fichier de configuration (devrait déjà être présent dans le répertoire personnel) */
- File f=new File(homedir,"app.conf");
-
- /* Vérifiez que le fichier existe, qu'il est vraiment un fichier et qu'il est lisible */
- if(f.exists() && f.isFile() && f.canRead()) {
- /* Créer un objet fichier pour un nouveau répertoire de configuration */
- File configdir = new File(f,".configdir");
- /* Et le crée */
- configdir.mkdir();
- /* Enfin, déplacez le fichier de configuration vers son nouvel emplacement */
- f.renameTo(new File(configdir,".config"));
- }
- }
- }
Cette exemple montre une partie de la flexibilité possible avec la classe File, mais démontre également certains des problèmes avec l'abstraction. Il est très général et nécessite donc de nombreuses méthodes pour interroger un objet File afin de déterminer ce qu'il représente réellement et ses capacités.
Fichiers
La classe File contient un très grand nombre de méthodes, mais certaines fonctionnalités de base (notamment un moyen de lire le contenu réel d'un fichier) ne sont pas et n'ont jamais été fournies directement. Voici un bref résumé des méthodes de la classe File :
- import java.io.File;
- import java.net.URI;
-
- class Main {
- public static void main (String[] args) {
- File homedir=new File(System.getProperty("user.home"));
- File f=new File(homedir,"app.conf");
- File rep=new File(f,".configdir");
- boolean resultat;
-
- /* Gestion des permissions : */
- boolean peuExecuter = f.canExecute();
- boolean peuLire = f.canRead();
- boolean peuEcrire = f.canWrite();
-
- resultat = f.setReadOnly();
- resultat = f.setExecutable(true);
- resultat = f.setReadable(true);
- resultat = f.setWritable(true);
-
- /* Différentes vues d'un nom de fichier : */
- File fichierAbsolue = f.getAbsoluteFile();
- File fichierCanonique = f.getCanonicalFile();
- String nomAbsolue = f.getAbsolutePath();
- String nomCanonique = f.getCanonicalPath();
- String nom = f.getName();
- String nomParent = f.getParent();
- URI fichierURI = f.toURI();
-
- /* Fichier de méta-données : */
- boolean existes = f.exists();
- boolean estceAbsolue = f.isAbsolute();
- boolean estceRepertoire = f.isDirectory();
- boolean estceFichier = f.isFile();
- boolean estceCache = f.isHidden();
- long heureModification = f.lastModified();
- boolean resultatOk = f.setLastModified(heureModification);
- long longueurFichier = f.length();
-
- /* Opération de gestion de fichier : */
- File fichierDest = new File(f,"nouveaunom.txt");
- boolean renomme = f.renameTo(fichierDest);
- boolean supprime = f.delete();
-
- /* Créer n'écrasera pas le fichier existant : */
- boolean creationOk = f.createNewFile();
-
- /* Gestion des fichier temporaire : */
- File temporaire = File.createTempFile("mon-temp",".tmp");
- temporaire.deleteOnExit();
-
- /* Gestion de répertoire */
- boolean creeRepertoire = rep.mkdir();
- String[] nomsFichier = rep.list();
- File[] fichiers = rep.listFiles();
- }
- }
La classe File contient également quelques méthodes ne convenant pas parfaitement à l'abstraction. Ils impliquent en grande partie d'interroger le système de fichiers (par exemple, demander l'espace libre disponible) sur lequel le fichier réside :
- import java.io.File;
-
- class Main {
- public static void main (String[] args) {
- File homedir=new File(System.getProperty("user.home"));
- File f=new File(homedir,"app.conf");
-
- long libre = f.getFreeSpace();
- long total = f.getTotalSpace();
- long utilise = f.getUsableSpace();
-
- File[] racines = File.listRoots(); /* Toutes les racines de système de fichiers disponibles */
- }
- }
Les flux de données (Stream)
L'abstraction de flux de données d'entrée/sorties (à ne pas confondre avec les flux utilisés lors du traitement des API de Java 8 Collection) était présente dans Java 1.0, comme moyen de traiter le flux de données séquentiel d'octets à partir de disques ou d'autres sources. Le coeur de cette API est une paire de classes abstraites, InputStream et OutputStream. Ceux-ci sont très largement utilisés, et en fait les flux de données d'entrée et de sortie standard, appelés System.in et System.out, sont des flux de données de ce type. Ce sont des champs publiques et statiques de la classe System, et sont souvent utilisés même dans les programmes les plus simples :
System.out.println("Bonjour Gladir.com!");
Des sous-classes spécifiques de flux de données, y compris FileInputStream et FileOutputStream, peuvent être utilisées pour opérer sur des octets individuels dans un fichier - par exemple, en comptant toutes les fois où le code ASCII 103 (minuscule g) apparaît dans un fichier :
- import java.io.FileInputStream;
- import java.io.IOException;
-
- class Main {
- public static void main (String[] args) {
- try(FileInputStream entree = new FileInputStream("/Users/gladir/texte.txt")) {
- byte[] tampon=new byte[4096];
- int longueur,compteur=0;
- while((longueur = entree.read(tampon)) > 0) {
- for(int i=0;i < longueur; i++) if(tampon[i] == 103) compteur++;
- }
- System.out.println("'g' est vue: "+compteur);
- } catch(IOException e) {
- e.printStackTrace();
- }
- }
- }
Cette approche du traitement des données sur disque manque de flexibilité - la plupart des développeurs pensent en termes de caractères et non d'octets. Pour permettre cette situation, les flux de données sont généralement combinés avec les classes Reader et Writer de haut niveau, fournissant un niveau d'interaction de flux de caractères, plutôt que le flux de données d'octets de bas niveau fourni par InputStream et OutputStream et leurs sous-classes.
Lectures et écritures
En passant à une abstraction traitant des caractères plutôt que des octets, les développeurs se voient présenter une API beaucoup plus familière et cachant de nombreux problèmes liés au codage de caractères, à Unicode,... Les classes Reader et Writer sont destinées à superposer les classes de flux de données d'octets et à supprimer le besoin de gestion de bas niveau des flux de données d'entrée/sortie. Ils ont plusieurs sous-classes étant souvent utilisées pour se superposer, telles que :