20. Les flux 22. La sérialisation Imprimer Sommaire Consulter avec table des matières
Développons en Java v 2.00
Copyright (C) 1999-2014 Jean-Michel DOUDOUX.
21. NIO 2
chapitre 2 1
Niveau : niveau 3 Intermédiaire
L'API NIO 2 a été développée sous la JSR 203 et a été ajoutée au JDK dans la version 7 de Java SE. NIO 2 est une API plus moderne et plus complète pour l'accès au système de fichiers. Son but est en partie de remplacer la classe File de la très ancienne API IO.
NIO 2 propose d'étendre les fonctionnalités relatives aux entrées/sorties : l'utilisation du système de fichiers de manière facile et les lectures/écritures asynchrones.
L'API FileSystem simplifie grandement la manipulation de fichiers et répertoires d'un système de fichiers et ajoute des fonctionnalités attendues depuis longtemps. La nouvelle API de gestion et d'accès au système de fichiers est contenue dans le package java.nio.file et ses sous-packages.
Parmi les nouvelles fonctionnalités proposées par NIO2, on peut trouver :
le support des liens physiques et symboliques s'ils sont pris en charge par le système de fichiers
la gestion des attributs sur les fichiers des systèmes Dos et POSIX
Le support de notifications en cas de changement dans le contenu d'un répertoire (ajout, suppression, modification d'un fichier du répertoire) en utilisant l'API WatchService
le support du parcours d'un répertoire avec la possibilité de filtrer les fichiers obtenus
l'utilisation de channels asynchrones avec lesquels les opérations de lecture/écriture sont réalisées en utilisant un pool de threads
l'ajout de fonctionnalités de base comme la copie ou le déplacement de fichiers
l'utilisation de fabriques pour permettre à l'API d'être extensible : il est par exemple possible de créer sa propre implémentation d'un système de fichiers. Une implémentation permettant de gérer les fichiers zip est d'ailleurs fournie en standard.
NIO2 est probablement l'API de Java 7 qui sera la plus utilisée par les développeurs tant elle facilite la mise en oeuvre de fonctionnalités courantes d'entrées/sorties sur un système de fichiers.
Ce chapitre contient plusieurs sections :
Les entrées/sorties avec Java
Les principales classes et interfaces
L'interface Path
Glob
La classe Files
Le parcours du contenu de répertoires
L'utilisation de systèmes de gestion de fichiers
La lecture et l'écriture dans un fichier
Les liens et les liens symboliques
La gestion des attributs
La gestion des unités de stockages
Les notifications de changements dans un répertoire
La gestion des erreurs et la libération des ressources
L'interopérabilité avec le code existant
21.1. Les entrées/sorties avec Java
Depuis les débuts de Java, l'API java.io est essentiellement composée de classes et interfaces pour réaliser des opérations sur les flux d'octets ou de caractères. Seule la classe File permet des opérations sur les fichiers et les répertoires du système de fichiers.
L'utilisation du système de fichiers se fait donc en utilisant l'API java.io et notamment la classe java.io.File qui présente de nombreux inconvénients :
Plusieurs méthodes ne lèvent pas une exception en cas de problème mais renvoient un booléen. Ceci ne respecte pas ce qu'il est possible de faire pour gérer les erreurs avec Java, rend l'API inconsistante et ne permet pas de connaître l'origine du problème mais seulement de savoir que la fonctionnalité a échouée
La méthode rename() n'a pas le même comportement sur toutes les plate-formes
Il n'y a pas de réel support pour les liens symboliques (symbolic links)
Les métadonnées (attributs de permissions, propriétaire, sécurité, ...) sont peu ou mal supportées
Certaines fonctionnalités de base sont absentes de l'API comme la copie ou le déplacement d'un fichier
Certaines fonctionnalités sont peu performantes comme par exemple la méthode listFiles() avec un répertoire contenant de nombreux fichiers
...
La classe java.io.File existe depuis Java 1.0 mais elle a évoluée dans plusieurs versions de Java :
Version
Méthodes ajoutées
1.1
getCanonicalPath()
1.2
getParentFile(), getAbsoluteFile(),getCanonicalFile(), toURL(), isHidden(), createNewFile(), deleteOnExit(), listFiles(), setLastModified(), setReadOnly(), listRoots(), createTempFile(), compareTo()
1.4
création à partir d'une URI, toURI()
6
setWritable(), setReadable(), setExecutable(), canExecute(), getTotalSpace(), getFreeSpace(), getUsableSpace()
L'API NIO a été introduite dans Java 1.4 : elle propose entre autres l'utilisation de channels, buffers et charsets notamment pour permettre de réaliser des opérations de lectures/écritures non bloquantes (non blocking I/O).
L'API NIO 2 est une API plus moderne qui propose plusieurs caractéristiques :
la séparation des responsabilités : un chemin (Path) représente un élément du système de fichiers (FileSystem) stocké dans un système de stockage (FileStorage) et est manipulé en utilisant la classe Files
la gestion de toutes les erreurs se fait avec des exceptions
l'utilisation de fabriques permet de créer les différentes instances de l'API et de la rendre extensible
La classe java.io.File n'est pas deprecated mais à partir de Java 7, il est recommandé d'utiliser les classes et interfaces de l'API NIO 2 dans la mesure du possible : ceci doit être le cas dans les nouveaux développements d'autant que , pour faciliter l'intégration dans le code existant, il existe des fonctionnalités pour convertir un objet de type File en un objet de type Path et vice versa.
21.2. Les principales classes et interfaces
NIO 2 repose sur plusieurs classes et interfaces dont les principales sont :
Path : encapsule un chemin dans le système de fichiers
Files : contient des méthodes statiques pour manipuler les éléments du système de fichiers
FileSystemProvider : service provider qui interagit avec le système de fichiers sous-jacent
FileSystem : encapsule un système de fichiers
FileSystems : fabrique qui permet de créer une instance de FileSystem
Ces classes et interfaces sont regroupées dans le package java.nio.file et ses sous-packages :
java.nio.file
java.nio.file.attribute
L'interface Path décrit les fonctionnalités d'une classe qui encapsule un chemin sur le système de fichiers. Ce chemin est représenté sous la forme d'une séquence de noms qui compose la hiérarchie des répertoires du chemin. Cette séquence peut inclure le nom d'un fichier ou d'un répertoire comme dernier élément mais pas obligatoirement car un objet de type Path peut simplement encapsuler un sous-chemin.
Les méthodes de l'interface Path permettent uniquement de manipuler les éléments qui composent le chemin : elles n'ont aucune action sur le système de fichiers sous-jacent du chemin.
La classe FileStorage encapsule un système de stockage de fichiers. Elle permet d'obtenir des informations sur le système de stockage comme l'espace total ou l'espace libre. Une instance de type FileStorage est obtenue en invoquant la méthode Files.getFileStore() en lui passant en paramètre un objet de type Path encapsulant un élément du système de stockage.
La classe FileSystem est une fabrique pour créer des objets relatifs à un système de fichiers. La méthode getPath() permet d'obtenir une instance d'un chemin dans le système de fichiers. La méthode getFileStores() permet d'obtenir une collection de tous les systèmes de stockage utilisables.
La classe FileSystems permet de créer des objets de type FileSystem. La méthode statique getDefault() permet d'obtenir une instance du FileSystem par défaut. La classe FileSystems permet aussi de créer des instances personnalisées de classes de type FileSystem.
21.3. L'interface Path
La classe Path est une des interfaces principales de NIO 2 : elle encapsule une abstraction d'un chemin vers un élément du système de fichiers de manière dépendante du système d'exploitation sous-jacent.
Le chemin peut concerner plusieurs types d'éléments :
Un fichier
Un répertoire
Un lien symbolique : permet de faire référence à un fichier ou un autre répertoire
Un sous-chemin
Ce chemin peut être absolu (le chemin contient une racine) ou relatif (en combinaison avec le chemin courant pour obtenir le chemin absolu). La représentation d'un chemin dépend du système de fichiers sous-jacents : par exemple, tous les systèmes d'exploitation n'utilisent pas tous le même séparateur entre les éléments d'un chemin. Un objet de type Path encapsule le chemin d'un élément du système de fichiers composés d'un ensemble d'éléments organisés de façon hiérarchique grâce à un séparateur spécifique au système.
Une instance de type Path encapsule les informations sur le chemin permettant de localiser un fichier ou un répertoire dans un système de fichiers. Elle peut contenir la racine ou le nom du fichier mais aucun des deux n'est obligatoire : elle peut contenir un sous-chemin ou uniquement le nom du fichier.
Le chemin d'un fichier ou d'un répertoire encapsulé dans une instance de type Path n'a pas forcément d'existence physique sur le système de fichiers sous-jacents.
Les instances de type Path sont immuables et utilisables dans un contexte multithread.
Elle hérite de plusieurs interfaces : Comparable
La classe Path possède plusieurs méthodes qui peuvent être utilisées pour obtenir des informations sur le chemin, accéder aux éléments du chemin, convertir le chemin ou extraire des sous-chemins, ... Ces méthodes traitent le chemin lui-même mais sont sans action sur le système de fichiers sous-jacents. Aucune méthode ne concerne la gestion des extensions des fichiers.
21.3.1. L'obtention d'une instance de type Path
Il n'est pas possible de créer une instance de type Path sans utiliser une fabrique ou un helper qui invoque une fabrique.
Il existe plusieurs manières de créer un objet de type Path :
invoquer la méthode getPath() d'une instance de type FileSystem
invoquer la méthode Paths.get() qui invoque la méthode FileSystems.getDefault().getPath()
invoquer la méthode toPath() sur un objet de type java.io.File
La méthode getPath() de la classe FileSystem permet d'obtenir une instance de type Path.
Exemple ( code Java 7 ) :
1.
Path chemin = FileSystems.getDefault()
2.
.getPath("C:/Users/jm/AppData/Local/Temp/monfichier.txt");
La classe Paths est un helper qui permet de facilement créer des instances de type Path : c'est une fabrique qui propose deux surcharges de sa méthode get() qui attendent respectivement en paramètres un nombre variable d'objets de type String qui sont les éléments du chemin ou une URI.
Exemple ( code Java 7 ) :
1.
Path chemin1 = Paths.get("C:/Users/jm/AppData/Local/Temp/monfichier.txt");
2.
3.
Path chemin2 = Paths.get(URI.create("file:///C:/Users/jm/AppData/Local/Temp/monfichier.txt"));
4.
5.
Path chemin3 = Paths.get(System.getProperty("java.io.tmpdir"), "monfichier.txt");
Le chemin précisé peut utiliser le séparateur du système sous-jacent.
Exemple ( code Java 7 ) :
1.
Path chemin1 = Paths.get("C:\\Users\\jm\\AppData\\Local\\Temp\\monfichier.txt");
21.3.2. L'obtention d'éléments du chemin
Une instance de type Path stocke les éléments de la hiérarchie du chemin sous une forme séquentielle, l'élément le plus haut dans la hiérarchie (après la racine) ayant l'index 0 et l'élément le plus bas ayant l'index n-1, n étant le nombre d'éléments du chemin.
L'interface Path propose plusieurs méthodes pour retrouver un élément particulier ou un sous-chemin composé de plusieurs éléments en utilisant les index.
Méthode
Rôle
String getFileName()
Retourner le nom du dernier élément du chemin. Si le chemin concerne un fichier alors c'est le nom du fichier qui est retourné
Path getName(int index)
Retourner l'élément du chemin dont l'index est fourni en paramètre. Le premier élément possède l'index 0
int getNameCount()
Retourner le nombre d'éléments du chemin
Path getParent()
Retourner le chemin parent ou null s'il n'existe pas (dans ce cas, le chemin correspond à une racine)
Path getRoot()
Retourner la racine d'un chemin absolu (par exemple C:\ sous Dos ou / sous Unix) ou null pour un chemin relatif
String toString()
Retourner le chemin sous la forme d'une chaîne de caractères
Path subPath(int beginIndex, int endIndex)
Retourner un sous-chemin correspondant aux deux index fournis en paramètres
Exemple ( code Java 7 ) :
1.
Path path = Paths.get("C:/Users/jm/AppData/Local/Temp/monfichier.txt");
2.
System.out.println("toString() = " + path.toString());
3.
System.out.println("getFileName() = " + path.getFileName());
4.
System.out.println("getRoot() = " + path.getRoot());
5.
System.out.println("getName(0) = " + path.getName(0));
6.
System.out.println("getNameCount() = " + path.getNameCount());
7.
System.out.println("getParent() = " + path.getParent());
8.
System.out.println("subpath(0,3) = " + path.subpath(0,3));
Résultat :
1.
toString() = C:\Users\jm\AppData\Local\Temp\monfichier.txt
2.
getFileName() = monfichier.txt
3.
getRoot() = C:\
4.
getName(0) = Users
5.
getNameCount() = 6
6.
getParent() = C:\Users\jm\AppData\Local\Temp
7.
subpath(0,3) = Users\jm\AppData
Le chemin peut aussi être relatif.
Exemple ( code Java 7 ) :
1.
Path path = Paths.get("jm/AppData/Local/Temp/monfichier.txt");
2.
System.out.println("toString() = " + path.toString());
3.
System.out.println("getFileName() = " + path.getFileName());
4.
System.out.println("getRoot() = " + path.getRoot());
5.
System.out.println("getName(0) = " + path.getName(0));
6.
System.out.println("getNameCount() = " + path.getNameCount());
7.
System.out.println("getParent() = " + path.getParent());
8.
System.out.println("subpath(0,3) = " + path.subpath(0, 3));
Résultat :
1.
toString() = jm\AppData\Local\Temp\monfichier.txt
2.
getFileName() = monfichier.txt
3.
getRoot() = null
4.
getName(0) = jm
5.
getNameCount() = 5
6.
getParent() = jm\AppData\Local\Temp
7.
subpath(0,3) = jm\AppData\Local
Une instance de type Path implémente l'interface Iterator qui permet de réaliser une itération sur les éléments du chemin.
Exemple ( code Java 7 ) :
1.
Path path = Paths.get("C:/Users/jm/AppData/Local/Temp/monfichier.txt");
2.
for (Path name : path) {
3.
System.out.println(name);
4.
}
Résultat :
1.
Users
2.
jm
3.
AppData
4.
Local
5.
Temp
6.
monfichier.txt
21.3.3. La manipulation d'un chemin
L'interface Path propose plusieurs méthodes pour manipuler les chemins :
Méthode
Rôle
Path normalize()
Nettoyer le chemin en supprimant les éléments « . » et « .. » qu'il contient
Path relativize(Path other)
Retourner le chemin relatif à celui fourni en paramètres
Path resolve(Path)
Combiner deux chemins
La méthode normalize() permet d'expliciter un chemin en éliminant les éléments comme « . » et « .. »
Exemple ( code Java 7 ) :
1.
Path path = Paths.get("C:/Users/jm/AppData/Local/Temp/./monfichier.txt");
2.
3.
System.out.println("normalize() = " + path.normalize());
4.
path = Paths.get("C:/Users/admin/ ./../jm/AppData/Local/Temp/./monfichier.txt");
5.
6.
System.out.println("normalize() = " + path.normalize());
Résultat :
1.
normalize() = C:\Users\jm\AppData\Local\Temp\monfichier.txt
2.
normalize() = C:\Users\jm\AppData\Local\Temp\monfichier.txt
La méthode normalize() effectue une opération purement syntaxique : elle ne vérifie pas dans le système de fichiers le chemin qu'elle produit.
La méthode resolve() permet de combiner deux chemins. Elle attend en paramètre un chemin partiel qui ne doit pas commencer par un élément racine du système de fichiers. Si le chemin fourni en paramètre contient un élément racine, alors la méthode resolve() renvoie le chemin fourni en paramètre.
Exemple ( code Java 7 ) :
1.
Path path = Paths.get("C:/Users/jm/AppData/Local/");
2.
Path nouveauPath = path.resolve("Temp/monfichier.txt");
3.
System.out.println(nouveauPath);
4.
nouveauPath = path.resolve("C:/Temp");
5.
System.out.println(nouveauPath);
Résultat :
1.
C:\Users\jm\AppData\Local\Temp\monfichier.txt
2.
C:\Temp
La méthode Path.relativize() permet d'obtenir le chemin relatif à celui encapsulé dans l'instance de type Path. Ceci permet de définir le chemin entre deux chemins du système de fichiers : le chemin retourné est alors le chemin relatif entre le chemin d'origine et le chemin cible.
La méthode relativize() effectue l'inverse de la méthode resolve() : elle ajoute au besoin dans le chemin qu'elle renvoie des éléments ./ ou ../
Exemple ( code Java 7 ) :
1.
Path path1 = Paths.get("C:/Users/jm");
2.
Path path2 = Paths.get("C:/Users/test");
3.
Path path1VersPath2 = path1.relativize(path2);
4.
System.out.println(path1VersPath2);
5.
Path path2VersPath1 = path2.relativize(path1);
6.
System.out.println(path2VersPath1);
Résultat :
1.
..\test
2.
..\jm
Dans cet exemple, les deux chemins ont le même répertoire père : le résultat de l'invocation de la méthode relativize() renvoie simplement un chemin qui remonte au répertoire père et descend au répertoire cible.
Exemple ( code Java 7 ) :
1.
Path path1 = Paths.get("C:/");
2.
Path path2 = Paths.get("C:/Users/test");
3.
Path path1VersPath2 = path1.relativize(path2);
4.
System.out.println(path1VersPath2);
5.
Path path2VersPath1 = path2.relativize(path1);
6.
System.out.println(path2VersPath1);
Résultat :
1.
Users\test
2.
..\..
Une exception est levée si un chemin relatif et un chemin absolu sont utilisés lors de l'invocation de la méthode relativize().
Exemple ( code Java 7 ) :
1.
Path path1 = Paths.get("test");
2.
Path path2 = Paths.get("C:/Users/test");
3.
Path path1VersPath2 = path1.relativize(path2);
4.
System.out.println(path1VersPath2);
5.
Path path2VersPath1 = path2.relativize(path1);
6.
System.out.println(path2VersPath1);
Résultat :
1.
Exception in thread "main" java.lang.IllegalArgumentException: 'other' is different type
2.
of Path
3.
at sun.nio.fs.WindowsPath.relativize(Unknown Source)
4.
at sun.nio.fs.WindowsPath.relativize(Unknown Source)
5.
at com.jmdoudoux.test.nio2.TestNIO2.testRelativize3(TestNIO2.java:33)
6.
at com.jmdoudoux.test.nio2.TestNIO2.main(TestNIO2.java:9)
21.3.4. La comparaison de chemins
Une instance de type Path redéfinit la méthode equals() pour permettre de tester l'égalité de l'instance avec une autre instance.
L'interface Path hérite de l'interface Comparable, ce qui permet de trier des objets de type Path.
L'interface Path propose également des méthodes permettant de comparer le début ou la fin de deux chemins
Méthode
Rôle
int compareTo(Path other)
Comparer le chemin avec celui fourni en paramètre
boolean endsWith(Path other)
Comparer la fin du chemin avec celui fourni en paramètre
boolean endsWith(String other)
Comparer la fin du chemin avec celui fourni en paramètre
boolean startsWith(Path other)
Comparer le début du chemin avec celui fourni en paramètre
boolean startsWith(String other)
Comparer le début du chemin avec celui fourni en paramètre
Attention : une instance de type Path est dépendante du système de fichiers : il n'est donc pas possible de comparer deux instances de type Path associées à deux systèmes de fichiers différents.
L'interface Path propose les méthodes startsWith() et endsWith() qui permettent respectivement de tester si le chemin commence ou se termine par la chaîne de caractères fournie en paramètre.
Exemple ( code Java 7 ) :
01.
Path path1 = Paths.get("C:/Users/jm");
02.
Path path2 = Paths.get("C:/");
03.
04.
System.out.println(path1.startsWith("C:/"));
05.
System.out.println(path1.startsWith("C:/Users"));
06.
System.out.println(path1.startsWith(path2));
07.
System.out.println(path1.startsWith("C:"));
08.
System.out.println(path1.startsWith("Users"));
09.
System.out.println(path1.startsWith("/Users"));
Résultat :
1.
true
2.
true
3.
true
4.
false
5.
false
6.
false
21.3.5. La conversion d'un chemin
Les chemins encapsulés dans une instance de type Path ne sont pas toujours complets ou linéaires : par exemple un chemin relatif ne possède pas de racine ou un chemin peut contenir un lien symbolique qui fera dévier le cheminement lors de l'accès à la ressource encapsulée par le chemin.
L'interface Path propose donc plusieurs méthodes pour convertir un chemin.
Méthode
Rôle
Path toAbsolutePath()
Retourner le chemin absolu du chemin
Path toRealPath(LinkOption...)
Retourner le chemin physique du chemin notamment en résolvant les liens symboliques selon les options fournies. Peut lever une exception si le fichier n'existe pas ou s'il ne peut pas être accédé
URI toUri()
Retourner le chemin sous la forme d'une URI
La méthode Path.toAbsolutePath() permet d'obtenir le chemin absolu du chemin encapsulé dans l'instance de type Path.
La méthode toRealPath() renvoie un chemin dans lequel les liens symboliques du chemin fourni en paramètre ont été résolus par rapport au système de fichiers.
Exemple ( code Java 7 ) :
01.
path = Paths.get("C:/Users/jm/AppData/Local/Temp/monfichier.txt");
02.
System.out.println("toUri() = " + path.toUri());
03.
path = Paths.get("src/monfichier.txt");
04.
System.out.println("toAbsolutePath() = " + path.toAbsolutePath());
05.
try {
06.
System.out.println("toRealPath() = " + path.toRealPath(LinkOption.NOFOLLOW_LINKS));
07.
} catch (IOException ex) {
08.
ex.printStackTrace();
09.
}
Résultat :
1.
toUri() = file:///C:/Users/jm/AppData/Local/Temp/monfichier.txt
2.
toAbsolutePath() = C:\Users\jm\Documents\NetBeansProjects\JavaApplication1\src\monfichier.txt
3.
toRealPath() = C:\Users\jm\Documents\NetBeansProjects\JavaApplication1\src\monfichier.txt
21.4. Glob
Un glob est un pattern qui est appliqué sur des noms de fichiers ou de répertoires : c'est une version simplifiée des expressions régulières adaptée aux noms d'éléments d'un système de fichiers.
Plusieurs méthodes de la classe Files attendent un glob en paramètre.
L'interface PathMatcher définit une méthode pour des objets dont le but est de réaliser des comparaisons sur des chemins.
Méthode
Rôle
Boolean matches(Path path)
Renvoie une booléen qui précise si le chemin correspond au pattern
Pour obtenir une instance de type PathMatcher, il faut invoquer la méthode getPathMatcher() de la classe FileSystem qui attend en paramètre une chaîne de caractères précisant la syntaxe et le pattern.
Exemple ( code Java 7 ) :
1.
PathMatcher matcher = FileSystems.getDefault().getPathMatcher("glob:*.java");
2.
if (matcher.matches(path)) {
3.
System.out.println(path);
4.
}
La définition d'un glob utilise une syntaxe qui lui est propre :
Motif
Rôle
*
Aucun ou plusieurs caractères
**
Aucun ou plusieurs sous-répertoires
?
Un caractère quelconque
{}
Un ensemble de motifs
exemple : {htm, html}
[]
Un ensemble de caractères.
Exemple :
[A-Z] : toutes les lettres majuscules
[0-9] : tous les chiffres
[a-z,A-Z] : toutes les lettres indépendamment de la casse
Chaque élément de l'ensemble est séparé par un caractère virgule
Le caractère - permet de définir une plage de caractères
A l'intérieur des crochets, les caractères *, ? et / ne sont pas interprétés
\
Il permet d'échapper des caractères pour éviter qu'ils ne soient interprétés.
Il sert notamment à échapper le caractère \ lui-même
Les autres caractères
Ils se représentent eux-mêmes sans être interprétés
Exemples :
Glob
Explication
*.html
tous les fichiers ayant l'extension .html
???
trois caractères quelconques
*[0-9]*
tous les fichiers qui contiennent au moins un chiffre
*.{htm, html}
tous les fichiers dont l'extension est htm ou html
I*.java
tous les fichiers dont le nom commence par un i majuscule et possède une extension .java
Chaque implémentation de type FileSystem permet d'obtenir une instance de type PathMatcher en utilisant la méthode getPathMatcher() qui attend en paramètre un objet de type String contennant la syntaxe et le motif.
Exemple ( code Java 7 ) :
1.
String pattern = "glob:*.{text}";
2.
PathMatcher matcher = FileSystems.getDefault().getPathMatcher("glob:" + pattern);
Le paramètre contient la syntaxe du motif suivi du caractère deux-points et du motif qui sera utilisé pour vérifier la correspondance. Dans l'exemple ci-dessus, la syntaxe utilisée est de type glob.
La syntaxe glob est simple mais il est aussi possible d'utiliser une expression régulière en précisant la syntaxe regex.
Une implémentation peut proposer le support d'autres syntaxes. Il est aussi possible de définir sa propre implémentation de l'interface PathMatcher.
L'interface PathMatcher ne possède qu'une seule méthode nommée matches() qui attend en paramètre un objet de type Path et renvoie un booléen.
Exemple ( code Java 7 ) :
1.
PathMatcher matcher = FileSystems.getDefault().getPathMatcher("glob:*.{java,class}");
2.
Path filename = ...;
3.
if (matcher.matches(filename)) {
4.
System.out.println(filename);
5.
}
Il faut être vigilent lors de la définition du motif utilisé par le glob car le motif s'applique sur l'ensemble du chemin.
Exemple ( code Java 7 ) :
01.
public static void testGlob() throws IOException {
02.
final Path file1 = Paths.get("C:/java/test/test.java");
03.
final Path file2 = Paths.get("C:/java/test/test.txt");
04.
final Path file3 = file1.getFileName();
05.
06.
String pattern = "glob:**/*.{java,class}";
07.
System.out.println("Pattern " + pattern);
08.
09.
PathMatcher matcher = FileSystems.getDefault().getPathMatcher(pattern);
10.
System.out.println(file1 + " " + matcher.matches(file1));
11.
System.out.format("%-22s %b\n", file2, matcher.matches(file2));
12.
System.out.format("%-22s %b\n", file3, matcher.matches(file3));
13.
System.out.println("");
14.
15.
pattern = "glob:*.java";
16.
System.out.println("Pattern " + pattern);
17.
matcher = FileSystems.getDefault().getPathMatcher(pattern);
18.
System.out.println(file1 + " " + matcher.matches(file1));
19.
System.out.format("%-22s %b\n", file3, matcher.matches(file3));
20.
}
Résultat :
1.
Pattern glob:**/*.{java,class}
2.
C:\java\test\test.java true
3.
C:\java\test\test.txt false
4.
test.java false
5.
Pattern glob:*.java
6.
C:\java\test\test.java false
7.
test.java true
21.5. La classe Files
La classe Files est un helper qui contient une cinquantaine de méthodes statiques permettant de réaliser des opérations sur des fichiers ou des répertoires dont le chemin est encapsulé dans un objet de type Path.
La classe java.nio.file.files propose de nombreuses méthodes statiques permettant de réaliser des opérations de base sur les fichiers et les répertoires : création, ouverture, suppression, test d'existence, changement des permissions, ...
Ces méthodes concernent notamment :
La création d'éléments : createDirectory(), createFile(), createLink(), createSymbolicLink(), createTempFile(), createTempDirectory(), ...
La manipulation d'éléments : delete(), move(), copy(), ...
L'obtention du type d'un élément : isRegularFile(), isDirectory(), probeContentType(), ...
L'obtention de métadonnées et la gestion des permissions : getAttributes(), getPosixFilePermissions(), isReadable(), isWriteable(), size(), getFileAttributeView(), ...
NIO 2 propose une API qui facilite la manipulation des éléments du système de fichiers pour par exemple créer, supprimer, déplacer, renommer ou copier un fichier. La manipulation des fichiers et des répertoires est assurée par la classe java.nio.file.Files qui est une classe utilitaire composée uniquement d'une cinquantaine de méthodes statiques.
Les méthodes de la classe Files attendent généralement en paramètre au moins une instance de type Path. Certaines méthodes de la classe Files réalisent des opérations atomiques (celles-ci doivent être réalisées dans leur entièreté ou pas du tout) : ces opérations réussissent ou échouent.
21.5.1. Les vérifications sur un fichier ou un répertoire
La classe Files propose deux méthodes pour vérifier l'existence d'un élément dans le système de fichier :
Méthode
Rôle
boolean exists(Path)
vérifier l'existence sur le système de fichiers de l'élément dont le chemin est encapsulé dans le paramètre de type Path fourni
boolean notExists(Path)
vérifier que l'élément dont le chemin est encapsulé dans l'instance de type Path fournie en paramètre n'existe pas sur le système de fichiers
Lors d'un test d'existence d'une instance de type Path, le résultat peut avoir plusieurs valeurs :
L'existence de l'élément est vérifiée
L'inexistence de l'élément est vérifiée
La vérification n'a pas pu être réalisée car le statut de l'élément est inconnu : c'est par exemple le cas si l'élément n'est pas accessible
La vérification n'a pas pu être réalisée si les méthodes exists() et notExists() pour une même instance de type Path renvoient toutes les deux false.
Attention : !Files.exists(path) n'est donc pas équivalent à Files.notExists(path)
La classe Files propose plusieurs méthodes pour vérifier les droits d'accès ou le type d'un élément de type Path :
Méthode
Rôle
boolean isReadable(Path path)
Retourner true si le fichier peut être lu
boolean isWritable(Path path)
Retourner true si le fichier peut être modifié
boolean isHidden(Path path)
Retourner true si le fichier est caché
boolean isExecutable(Path path)
Retourner true si le fichier est exécutable
boolean isRegularFile(Path path)
Retourner true si l'objet encapsulé dans le Path est un fichier
boolean isDirectory(Path path)
Retourner true si l'objet encapsulé dans le Path est un répertoire
boolean isSymbolicLink(Path path)
Retourner true si l'objet encapsulé dans le Path est un lien symbolique
Exemple ( code Java 7 ) :
1.
public static void testAttributs() throws IOException {
2.
Path monFichier = Paths.get("C:/java/temp/monfichier.txt");
3.
boolean estLisible = Files.isRegularFile(monFichier) &
4.
Files.isReadable(monFichier);
5.
System.out.println(monFichier + " est lisible : "+estLisible);
6.
}
Résultat :
1.
C:\java\temp\monfichier.txt est lisible : true
La classe Files propose aussi plusieurs méthodes pour faire d'autres vérifications sur des éléments de type Path.
Méthode
Rôle
isSamePath(Path, Path)
Comparer les deux instances de Path pour déterminer si elles correspondent aux mêmes éléments dans le système de fichiers.
Ceci est pratique si l'un des deux Path est un lien symbolique.
Exemple ( code Java 7 ) :
01.
public static void sontIdentiques(String cheminCible, String cheminLien)
02.
throws IOException {
03.
Path lien = Paths.get(cheminLien);
04.
Path cible = Paths.get(cheminCible);
05.
if (Files.isSameFile(lien, cible)) {
06.
System.out.println("Fichiers identiques");
07.
} else {
08.
System.out.println("Fichiers différents");
09.
}
10.
}
21.5.2. La création d'un fichier ou d'un répertoire
L'API permet la création de fichiers, de répertoires permanents ou temporaires en utilisant plusieurs méthodes de la classe File :
Méthode
Rôle
Path createFile(Path path, FileAttribute... attrs)
Créer un fichier dont le chemin est encapsulé par l'instance de type Path fournie en paramètre
Path createDirectory(Path dir, FileAttribute... attrs)
Créer un répertoire dont le chemin est encapsulé par l'instance de type Path fournie en paramètre
Path createDirectories(Path dir, FileAttribute... attrs)
Créer dans le répertoire dont le chemin est fourni en paramètre un sous-répertoire avec les attributs fournis
Path createTempDirectory(Path dir, String prefix, FileAttribute... attrs)
Créer dans le répertoire dont le chemin est fourni en paramètre un sous-répertoire temporaire dont le nom utilisera le préfixe fourni
Path createTempDirectory(String prefix, FileAttribute... attrs)
Créer dans le répertoire temporaire par défaut du système, un sous-répertoire temporaire dont le nom utilisera la préfixe fourni
Path createTempFile(Path dir, String prefix, String suffix, FileAttribute... attrs)
Créer dans le répertoire dont le chemin est fourni en paramètre un fichier temporaire dont le nom utilisera le préfixe fourni
Path createTempFile(String prefix, String suffix, FileAttribute... attrs)
Créer dans le répertoire temporaire par défaut du système un fichier temporaire dont le nom utilisera le préfixe et le suffixe fournis
La méthode Files.createFile() permet de créer un fichier dont le chemin est encapsulé dans son paramètre de type Path.
La méthode createFile() attend en paramètres un objet de type Path et un varargs de type FileAttribute< ?> qui permet de préciser les attributs du fichier créé.
Exemple ( code Java 7 ) :
1.
Pathfichier=Paths.get("/home/jm/test.txt");
2.
Set
3.
FileAttribute
4.
Files.createFile(fichier,attr);
Si le chemin est uniquement fourni en paramètre de la méthode createFile(), le fichier est créé avec les attributs par défaut du système.
Exemple ( code Java 7 ) :
1.
Path monFichier = Paths.get("C:/temp/monfichier.txt");
2.
Path file = Files.createFile(monFichier);
Par défaut, une exception de type FileAlreadyExistsException est levée si le fichier à créer existe déjà.
La méthode createTempFile() permet de créer un fichier temporaire.
Elle possède deux surcharges :
Méthode
Rôle
createTempFile(Path dir, String prefix, String suffix, FileAttribute< ?>... attrs)
Créer un fichier temporaire dans le répertoire dont le chemin est fourni en paramètre
createTempFile(String prefix, String suffix, FileAttribute< ?>... attrs)
Créer un fichier temporaire dans le répertoire par défaut du système
Les deux surcharges attendent en paramètres un préfixe et un suffixe qui seront utilisés pour déterminer le nom du fichier et les attributs à utiliser lors de la création du fichier. Le préfixe et le suffixe peuvent être null : s'ils sont fournis, ils seront utilisés par l'implémentation de manière spécifique pour déterminer le nom du fichier. Le format du nom du fichier créé est dépendant de la plate-forme.
Exemple ( code Java 7 ) :
1.
public static void testCreateTempFile() throws IOException {
2.
Path tempFile = Files.createTempFile("monapp_", ".tmp");
3.
System.out.format("Fichier créé : %s%n", tempFile);
4.
}
Résultat :
1.
Fichier créé : C:\DOCUME~1\jm\LOCALS~1\Temp\monapp_242180026059597956.tmp
La méthode createDirectory() permet de créer un répertoire : elle attend en paramètre un objet de type Path qui encapsule le chemin ou le sous-chemin du répertoire et un varargs de type FileAttribute< ?> qui permet de préciser les attributs du nouveau répertoire.
Si aucun attribut n'est fourni en paramètre, alors le répertoire est créé avec les attributs par défaut du système.
Exemple ( code Java 7 ) :
1.
public static void testCreateDirectory() throws IOException {
2.
Path monRepertoire = Paths.get("C:/temp/mon_repertoire");
3.
Path file = Files.createDirectory(monRepertoire);
4.
}
Si le répertoire à créer existe déjà alors une exception de type FileAlreadyExistsException est levée.
La méthode createDirectory() ne permet que de créer un seul sous-répertoire : le chemin ou le sous-chemin fourni ne doit donc correspondre qu'à un nouveau sous-répertoire à créer dans un répertoire existant. Dans le cas contraire, une exception de type NoSuchFileException est levée.
Exemple ( code Java 7 ) :
1.
public static void testCreateDirectory() throws IOException {
2.
Path monRepertoire = Paths.get("C:/temp/niveau1/niveau2/mon_repertoire");
3.
Path file = Files.createDirectory(monRepertoire);
4.
}
Résultat :
01.
java.nio.file.NoSuchFileException:
02.
C:\temp\niveau1\niveau2\mon_repertoire
03.
at sun.nio.fs.WindowsException.translateToIOException(Unknown Source)
04.
at sun.nio.fs.WindowsException.rethrowAsIOException(Unknown Source)
05.
at sun.nio.fs.WindowsException.rethrowAsIOException(Unknown Source)
06.
at sun.nio.fs.WindowsFileSystemProvider.createDirectory(Unknown Source)
07.
at java.nio.file.Files.createDirectory(Unknown Source)
08.
at com.jmdoudoux.test.nio2.TestNIO2.testCreateDirectory(TestNIO2.java:199)
09.
at com.jmdoudoux.test.nio2.TestNIO2.main(TestNIO2.java:28)
Pour créer toute l'arborescence fournie dans le chemin, incluant la création d'un ou plusieurs sous-répertoires manquants dans l'arborescence, il faut utiliser la méthode createDirectories().
Exemple ( code Java 7 ) :
1.
public static void testCreateDirectories() throws IOException {
2.
Path monRepertoire = Paths.get("C:/temp/niveau1/niveau2/mon_repertoire");
3.
Path file = Files.createDirectories(monRepertoire);
4.
}
Pour créer un répertoire temporaire, il faut utiliser la méthode createTempDirectory() qui possède deux surcharges :
createTempDirectory(Path dir, String prefix, FileAttribute... attrs)
createTempDirectory(String prefix, FileAttribute... attrs)
La surcharge qui attend en paramètre un objet de type Path permet de préciser le sous-répertoire dans lequel le répertoire temporaire va être créé. La seconde surcharge crée le sous-répertoire temporaire dans le répertoire temporaire par défaut du système d'exploitation.
Le paramètres varargs de type FileAttributs permet de préciser les attributs qui seront associés au nouveau répertoire. Si aucun attribut n'est précisé alors ce sont les attributs par défaut du système qui seront utilisés.
Le paramètre prefix, qui peut être null, sera utilisé de manière dépendante de l'implémentation pour construire le nom du répertoire.
Exemple ( code Java 7 ) :
1.
public static void testCreateTempDirectory() throws IOException {
2.
Path repertoireTemp = Files.createTempDirectory(null);
3.
System.out.println(repertoireTemp);
4.
repertoireTemp = Files.createTempDirectory("monApp_");
5.
System.out.println(repertoireTemp);
6.
}
Résultat :
1.
C:\DOCUME~1\jm\LOCALS~1\Temp\2626334559178550265
2.
C:\DOCUME~1\jm\LOCALS~1\Temp\monApp_404075526480225045
21.5.3. La copie d'un fichier ou d'un répertoire
Ecrire sa propre méthode pour une fonctionnalité aussi basique que la copie d'un fichier ne présente pas beaucoup d'intérêt. Il est préférable d'utiliser une bibliothèque tierce comme Apache Commons IO ou Google Guava car cette fonctionnalité n'est pas proposée par l'API Java Core avant Java 7.
La classe Files propose plusieurs surcharges de la méthode copy() pour copier un fichier ou un répertoire.
Méthode
Rôle
Path copy(Path source, Path target, CopyOption... options)
Copier un élément avec les options précisées
long copy(InputStream in, Path target, CopyOption... options)
Copier tous les octets d'un flux de type InputStream vers un fichier
long copy(Path source, OutputStream out)
Copier tous les octets d'un fichier dans un flux de type OutputStream
La méthode Files.copy() permet de copier un fichier dont les chemins source et cible sont encapsulés dans ses deux paramètres de type Path.
Exemple ( code Java 7 ) :
1.
Path monFichier =
2.
Paths.get("C:\\temp\\monfichier.txt");
3.
Path monFichierCopie = Paths.get("C:\\temp\\monfichier - copie.txt");
4.
Path file = Files.copy(monFichier, monFichierCopie);
Une surcharge de la méthode copy() permet de préciser les options de copie du fichier en utilisant son troisième paramètre qui est un varargs de type CopyOption.
Plusieurs valeurs des énumérations StandardCopyOption et LinkOption qui implémentent l'interface CopyOption peuvent être utilisées avec la méthode copy() :
Valeur
Rôle
StandardCopyOption.COPY_ATTRIBUTES
La copie se fait en conservant les attributs du fichier : ceux-ci sont dépendants du système sous-jacent
StandardCopyOption.REPLACE_EXISTING
Remplacer le fichier cible s'il existe. Si le chemin cible est un répertoire non vide, une exception de type FileAlreadyExistsException est levée
LinkOption.NOFOLLOW_LINKS
Ne pas suivre les liens symboliques. Si le chemin à copier est un lien symbolique, c'est le lien lui-même qui est copié
Exemple ( code Java 7 ) :
1.
import static java.nio.file.StandardCopyOption.*;
2.
3.
// ...
4.
5.
Path monFichier = Paths.get("C:\\temp\\monfichier.txt");
6.
Path monFichierCopie = Paths.get("C:\\temp\\monfichier - copie.txt");
7.
Path file = Files.copy(monFichier, monFichierCopie, REPLACE_EXISTING);
Faute d'option indiquée, une exception est levée si le fichier cible existe déjà. La copie échoue si la destination existe sauf si l'option StandardCopyOption.REPLACE_EXISTING est utilisée.
La copie d'un lien symbolique duplique sa cible si l'option LinkOption.NOFOLLOW_LINKS est utilisée : dans ce cas, c'est le lien lui-même qui est copié.
Si l'option StandardCopyOption.ATOMIC_MOVE est utilisée avec la méthode copy(), alors une exception de type UnsupportedOperationException est levée.
Attention : il est possible d'utiliser la méthode copy() sur un répertoire cependant, le répertoire sera créé sans que les fichiers et.. ne le soient : quoi que contienne le répertoire, la méthode copy ne créé qu'un répertoire vide. Pour copier le contenu du répertoire, il faut parcourir son contenu et copier chacun des éléments un par un.
La méthode copy() possède deux surcharges qui permettent d'utiliser respectivement un objet de type InputStream comme source et un objet de type OutputStream comme cible.
Exemple ( code Java 7 ) :
1.
public static void copierFichier2() throws IOException {
2.
Path cible = Paths.get("c:/java/test/monfichier_copie.txt");
3.
URI uri = new File("c:/java/test/monfichier.txt").toURI();
4.
try (InputStream in = uri.toURL().openStream()) {
5.
Files.copy(in, cible);
6.
}
7.
}
21.5.4. Le déplacement d'un fichier ou d'un répertoire
Avant Java 7, la méthode rename() de la classe java.io.File ne fonctionnait pas sur tous les systèmes d'exploitation et généralement pas au travers du réseau. Bien que peu performante, la solution la plus sûre était de copier chaque octet du fichier source puis de supprimer ce fichier.
La méthode Files.move() permet de déplacer ou de renommer un fichier dont les chemins source et cible sont encapsulés dans ses deux paramètres de type Path.
Méthode
Rôle
move(Path source, Path target, CopyOption... options)
Déplacer ou renommer un élément avec les options précisées
Exemple ( code Java 7 ) :
1.
Path monFichier = Paths.get("C:\\temp\\monfichier.txt");
2.
Path monFichierCopie = Paths.get("C:\\temp\\monfichier.old");
3.
Path file = Files.move(monFichier, monFichierCopie);
Les options de déplacement du fichier peuvent être précisées en utilisant son troisième paramètre de type CopyOption.
Plusieurs valeurs de l'énumération StandardCopyOption qui implémente l'interface CopyOption peuvent être utilisées avec la méthode move() :
Valeur
Rôle
StandardCopyOption.REPLACE_EXISTING
Remplacement du fichier s'il existe
StandardCopyOption.ATOMIC_MOVE
Assure que le déplacement est réalisé sous la forme d'une opération atomique. Si l'atomicité de l'opération ne peut être garantie alors une exception de type AtomicMoveNotSupportedException est levée
Exemple ( code Java 7 ) :
1.
Path monFichier = Paths.get("C:\\temp\\monfichier.txt");
2.
Path monFichierCopie = Paths.get("C:\\temp\\monfichier.old");
3.
Path file = Files.move(monFichier, monFichierCopie, REPLACE_EXISTING, COPY_ATTRIBUTES);
Si la méthode move() est invoquée avec l'option StandardCopyOption.COPY_ATTRIBUTES alors une exception de type UnsupportedOperationException est levée.
Par défaut, l'invocation de la méthode move() dont le chemin cible existe déjà lève une exception de type FileAlreadyExistException. Pour écraser le fichier existant, il faut utiliser l'option StandardCopyOption.REPLACE_EXISTING.
Si le chemin source est un lien alors c'est le lien lui-même et non sa cible qui est déplacé.
Si les chemins cible et source fournis en paramètres de la méthode move() sont identiques alors l'invocation de la méthode n'a aucun effet.
Exemple ( code Java 7 ) :
1.
public static void testMove() throws IOException {
2.
Path source = Paths.get("C:/java/temp/monfichier.txt");
3.
Path cible = Paths.get("C:/java/temp/monfichier.txt");
4.
Files.move(source, cible);
5.
}
La méthode move() peut être utilisée sur un répertoire vide ou sur un répertoire non vide dont la cible est sur le même système de fichiers. Dans ce cas le répertoire est simplement renommé et il n'est pas nécessaire de déplacer récursivement le contenu du répertoire.
Exemple ( code Java 7 ) :
1.
public static void testMoveRepertoireVide() throws IOException {
2.
Path source = Paths.get("C:/java/temp/mon_repertoire");
3.
Path cible = Paths.get("C:/temp/mon_repertoire_copie");
4.
Files.move(source, cible);
5.
}
Exemple ( code Java 7 ) :
1.
public static void testRenommerRepertoire() throws IOException {
2.
Path source = Paths.get("C:/java/temp/mon_repertoire");
3.
Path cible = source.resolveSibling("mon_repertoire_copie");
4.
Files.move(source, cible);
5.
}
Si le répertoire cible existe déjà, même vide, alors une exception de type FileAlreadyExistsException est levée. Pour forcer le remplacement, il faut utiliser l'option REPLACE_EXISTING.
Exemple ( code Java 7 ) :
1.
public static void testMoveRepertoireVide() throws IOException {
2.
Path source = Paths.get("C:/java/temp/mon_repertoire");
3.
Path cible = Paths.get("C:/java/temp/mon_repertoire_copie");
4.
Files.move(source, cible, StandardCopyOption.REPLACE_EXISTING);
5.
}
Si le répertoire cible existe et n'est pas vide, alors une exception de type DirectoryNotEmptyException est levée.
Une exception de type AtomicNotSupportedException est levée si le déplacement du répertoire implique deux systèmes de fichiers différents entre la cible et la source et que l'option ATOMIC_MOVE est utilisée.
Exemple ( code Java 7 ) :
1.
// déplacer un fichier dans une autre unité de stockage
2.
source = Paths.get("c:/temp/cible.txt");
3.
cible = Paths.get("s:/cible.txt");
4.
try {
5.
Files.move(source, cible, ATOMIC_MOVE);
6.
} catch (final IOException ioe) {
7.
ioe.printStackTrace();
8.
}
Résultat :
1.
java.nio.file.AtomicMoveNotSupportedException:
2.
c:\temp\cible.txt -> s:\cible.txt: Impossible de déplacer le fichier vers un
3.
lecteur de disque différent.
4.
at sun.nio.fs.WindowsFileCopy.move(WindowsFileCopy.java:296)
5.
at sun.nio.fs.WindowsFileSystemProvider.move(WindowsFileSystemProvider.java:286)
6.
at java.nio.file.Files.move(Files.java:1339)
Les répertoires vides peuvent être déplacés. Si le répertoire n'est pas vide alors il est possible de le déplacer à condition que son contenu n'est pas besoin de l'être : ceci dépend du système d'exploitation sous-jacent qui peut simplement renommer le répertoire si celui-ci reste sur la même unité de stockage.
Sur la plupart des systèmes, le déplacement d'un répertoire vers une cible sur le même système de stockage se fait simplement en modifiant des entrées dans la table d'allocations des fichiers.
Par contre, le déplacement vers une autre unité de stockage implique forcément le déplacement du contenu du répertoire.
Pour tout autre problème lors de l'invocation de la méthode move(), comme pour toute opération d'entrée/sortie, une erreur peut survenir : dans ce cas, la méthode lève une exception de type IOException.
L'exécution de la méthode move() se fait de manière synchrone et bloquante.
Par défaut, lors de la copie ou du déplacement d'un fichier :
la copie échoue si le fichier cible existe déjà
les attributs du fichier peuvent être conservés entièrement, partiellement ou pas du tout
lors de la copie d'un lien symbolique, c'est la cible du lien qui est copiée et non le lien lui-même
lors du déplacement d'un lien symbolique, le lien lui-même est déplacé mais le fichier cible n'est pas déplacé
un répertoire est déplacé seulement s'il est vide ou si le déplacement consiste simplement à le renommer
21.5.5. La suppression d'un fichier ou d'un répertoire
L'API permet la suppression de fichiers, de répertoires ou de liens en utilisant l'une des deux méthodes de la classe Files :
Méthode
Rôle
void delete(Path path)
Supprimer un élément du système de fichiers
boolean deleteIfExist(Path path)
Supprimer un élément du système de fichiers s'il existe
La méthode Files.delete() permet de supprimer un fichier dont le chemin est encapsulé dans son paramètre de type Path. Elle lève une exception si la suppression échoue. Par exemple, une exception de type NoSuchFileException est levée si le fichier à supprimer n'existe pas dans le système de fichiers.
La suppression d'un lien symbolique supprime le lien mais ne supprime pas le fichier cible.
La suppression d'un répertoire échoue si le répertoire n'est pas vide.
Exemple ( code Java 7 ) :
01.
Path path = Paths.get("c:/java/test.txt");
02.
try {
03.
Files.delete(path);
04.
} catch (NoSuchFileException nsfe) {
05.
System.err.println("Fichier ou repertoire " + path + " n'existe pas");
06.
} catch (DirectoryNotEmptyException dnee) {
07.
System.err.println("Le repertoire " + path + " n'est pas vide");
08.
} catch (IOException ioe) {
09.
System.err.println("Impossible de supprimer " + path + " : " + ioe);
10.
}
La méthode deleteIfExist() permet de supprimer un élément du système de fichiers sans lever d'exception si celui-ci n'existe pas.
Exemple ( code Java 7 ) :
1.
Path path = Paths.get("c:/java/test.txt");
2.
try {
3.
Files.deleteIfExists(path);
4.
} catch (DirectoryNotEmptyException dnee) {
5.
System.err.println("Le repertoire " + path + " n'est pas vide");
6.
} catch (IOException ioe) {
7.
System.err.println("Impossible de supprimer " + path + " : " + ioe);
8.
}
21.5.6. L'obtention du type de fichier
NIO2 propose une fonctionnalité pour obtenir le type du contenu d'un fichier en utilisant la méthode probeContentType() de la classe Files
Méthode
Rôle
String probeContentType(Path path)
Retourner le type du contenu du fichier dont le chemin est passé en paramètre
Exemple ( code Java 7 ) :
01.
package com.jmdoudoux.test.nio2;
02.
03.
import java.io.IOException;
04.
import java.nio.file.Files;
05.
import java.nio.file.Path;
06.
import java.nio.file.Paths;
07.
08.
public class TestNIO2 {
09.
10.
public static void main(String[] args) {
11.
try {
12.
Path source = Paths.get("c:/java/temp/monfichier.txt");
13.
testProbeContent(source);
14.
source = Paths.get("c:/java/temp/monfichier.bin");
15.
testProbeContent(source);
16.
source = Paths.get("c:/java/temp/monfichier");
17.
testProbeContent(source);
18.
} catch (IOException e) {
19.
e.printStackTrace();
20.
}
21.
}
22.
23.
public static void testProbeContent(Path fichier) throws IOException {
24.
String type = Files.probeContentType(fichier);
25.
if (type == null) {
26.
System.out.println("Impossible de déteminer le type du fichier :"
27.
+ fichier);
28.
} else {
29.
System.out.println("le fichier " + fichier + " est du type : " + type);
30.
}
31.
}
32.
}
Résultat :
1.
le fichier c:\java\temp\monfichier.txt
2.
est du type : text/plain
3.
Impossible de déteminer le type du
4.
fichier : c:\java\temp\monfichier.bin
5.
Impossible de déteminer le type du
6.
fichier : c:\java\temp\monfichier
La méthode probeContentType() renvoie null si le type de contenu ne peut pas être déterminé.
Si le type a pu être déterminé, il est renvoyé sous la forme d'une chaîne de caractères dont le contenu respecte la norme MIME (Multipurpose Internet Mail Extensions) définit par la RFC 2045.
L'implémentation de cette méthode est dépendante de la plate-forme : sa fiabilité n'est donc pas garantie.
Il est possible de fournir une implémentation du type FileTypeDetector pour déterminer le type du contenu d'un fichier.
Si aucune implémentation de type FileTypeDetector ne peut déterminer le type, alors la méthode probeContentType() va demander au système de déterminer le type du contenu.
Pour définir sa propre implémentation, il faut créer une classe qui hérite de la classe abstraite FileTypeDetector et redéfinir sa méthode abstraite probeContentType() qui attend en paramètre un objet de type Path et renvoie une chaîne de caractères.
L'implémentation doit avoir un constructeur sans argument.
L'enregistrement de FileTypeDetector doit se faire un utilisant le service Provider de la JVM : le nom pleinement qualifié de la classe doit être dans un fichier java.nio.file.spi.FileTypeDetector contenu dans le sous-répertoire META-INF/services.
La détermination du type du contenu est généralement spécifique au système d'exploitation sous-jacent : utilisation de l'extension, de métadonnées dans un fichier associé ou lecture de tout ou partie du contenu du fichier.
21.6. Le parcours du contenu de répertoires
Les solutions proposées par NIO2 pour le parcours du contenu d'un répertoire remplacent avantageusement les méthodes list() et listfiles() de la classe java.io.File. Ces méthodes offrent de piètres performances notamment avec des répertoires contenant de nombreux fichiers et consomment beaucoup de ressources.
NIO2 propose plusieurs solutions pour parcourir le contenu d'un répertoire : elles sont plus complexes à mettre en oeuvre par rapport à la classe java.io.File mais sont aussi beaucoup plus performantes surtout avec des répertoires qui contiennent de nombreux fichiers.
21.6.1. Le parcours d'un répertoire
Il est possible d'utiliser une instance de l'interface java.nio.file.DirectoryStream qui permet de parcourir tous les éléments d'un répertoire en réalisant une itération sur les éléments qu'il contient.
La méthode newDirectoryStream() de la classe Files attend en paramètre un objet de type Path qui correspond au répertoire à parcourir et permet d'obtenir une instance de type DirectoryStream
La méthode iterator() retourne une instance d'un itérateur sur les éléments du répertoire : fichiers, liens, sous-répertoires, ...
L'itération sur les éléments permet de meilleures performances et une consommation réduite en ressources pour obtenir les mêmes résultats que l'invocation des méthodes list() et listFiles() de la classe java.io.File.
Attention : il est très important d'invoquer la méthode close() de l'instance de type DirectoryStream pour libérer les ressources utilisées.
Exemple ( code Java 7 ) :
01.
public static void testDirectoryStream() throws IOException {
02.
Path jdkPath = Paths.get("C:/Program Files/Java/jdk1.7.0_02");
03.
DirectoryStream
04.
try {
05.
Iterator
06.
while(iterator.hasNext()) {
07.
Path p = iterator.next();
08.
System.out.println(p);
09.
}
10.
} finally {
11.
stream.close();
12.
}
13.
}
Résultat :
01.
C:\Program Files\Java\jdk1.7.0\bin
02.
C:\Program Files\Java\jdk1.7.0\COPYRIGHT
03.
C:\Program Files\Java\jdk1.7.0\db
04.
C:\Program Files\Java\jdk1.7.0\demo
05.
C:\Program Files\Java\jdk1.7.0\include
06.
C:\Program Files\Java\jdk1.7.0\jre
07.
C:\Program Files\Java\jdk1.7.0\lib
08.
C:\Program Files\Java\jdk1.7.0\LICENSE
09.
C:\Program Files\Java\jdk1.7.0\README.html
10.
C:\Program Files\Java\jdk1.7.0\register.html
11.
C:\Program Files\Java\jdk1.7.0\register_ja.html
12.
C:\Program Files\Java\jdk1.7.0\register_zh_CN.html
13.
C:\Program Files\Java\jdk1.7.0\release
14.
C:\Program Files\Java\jdk1.7.0\sample
15.
C:\Program Files\Java\jdk1.7.0\src.zip
16.
C:\Program Files\Java\jdk1.7.0\THIRDPARTYLICENSEREADME.txt
L'ordre dans lequel les éléments sont fournis lors de l'itération n'est pas garanti. Des éléments spécifiques à certains systèmes ne sont pas retournés dans l'itération : c'est notamment le cas des éléments « . » (le répertoire courant) et « .. » (le répertoire parent) sur un système de type Unix.
Attention : l'implémentation de l'interface Iterable de l'instance de type DirectoryStream ne propose pas le support de la méthode remove() et son invocation lève une exception de type UnsupportedOperationException.
Exemple ( code Java 7 ) :
01.
public static void testDirectoryStream() throws IOException {
02.
Path jdkPath = Paths.get("C:/Program Files/Java/jdk1.7.0_02");
03.
DirectoryStream
04.
try {
05.
Iterator
06.
while(iterator.hasNext()) {
07.
Path p = iterator.next();
08.
System.out.println(p);
09.
Iterator.remove();
10.
}
11.
} finally {
12.
stream.close();
13.
}
14.
}
Résultat :
1.
C:\Program Files\Java\jdk1.7.0\bin
2.
Exception in thread "main"
3.
java.lang.UnsupportedOperationException
4.
at sun.nio.fs.WindowsDirectoryStream$WindowsDirectoryIterator.remove(Unknown
5.
Source)
6.
at com.jmdoudoux.test.nio2.TestNIO2.testDirectoryStream(TestNIO2.java:138)
7.
at com.jmdoudoux.test.nio2.TestNIO2.main(TestNIO2.java:25)
L'interface DirectoryStream hérite des interfaces Closeable et Iterable. Il est donc pratique de déclarer l'instance de type DirectoryStream
Exemple ( code Java 7 ) :
1.
public static void utilisationDirectoryStream() throws IOException {
2.
Path jdkPath = Paths.get("C:/Program Files/Java/jdk1.7.0");
3.
try (DirectoryStream
4.
for (Path entry : stream) {
5.
System.out.println(entry);
6.
}
7.
}
8.
}
Si une exception est levée durant l'itération, alors elle est encapsulée dans une exception unchecked de type DirectoryIteratorException.
Exemple ( code Java 7 ) :
01.
public static void testDirectoryStream3() {
02.
Path jdkPath = Paths.get("C:/Program Files/Java/jdk1.7.0_02");
03.
try (DirectoryStream
04.
for (Path entry : stream) {
05.
System.out.println(entry);
06.
}
07.
} catch (IOException | DirectoryIteratorException e) {
08.
e.printStackTrace();
09.
}
10.
}
Il est aussi possible de fournir un paramètre qui est une chaîne de caractères au format glob pour filtrer la liste des éléments retournés en fonction de leurs noms.
Exemple ( code Java 7 ) :
1.
public static void utilisationDirectoryStream() throws IOException {
2.
Path jdkPath = Paths.get("C:/Program Files/Java/jdk1.7.0");
3.
try (DirectoryStream
4.
for (Path entry : stream) {
5.
System.out.println(entry);
6.
}
7.
}
8.
}
Résultat :
1.
C:\Program Files\Java\jdk1.7.0\README.html
2.
C:\Program Files\Java\jdk1.7.0\register.html
3.
C:\Program Files\Java\jdk1.7.0\register_ja.html
4.
C:\Program Files\Java\jdk1.7.0\register_zh_CN.html
5.
C:\Program Files\Java\jdk1.7.0\src.zip
Attention : il n'est possible de n'obtenir qu'un seul itérateur d'une même instance de type DirectoryStream. Une seconde invocation de la méthode iterator() lève une exception de type IllegalStateException.
Exemple ( code Java 7 ) :
01.
public static void utilisationDirectoryStream() throws IOException {
02.
Path jdkPath = Paths.get("C:/Program Files/Java/jdk1.7.0_02");
03.
try (DirectoryStream
04.
for (Path entry : stream) {
05.
System.out.println(entry);
06.
Iterator
07.
}
08.
}
09.
}
Résultat :
1.
Exception in thread "main"
2.
java.lang.IllegalStateException: Iterator already obtained
3.
at sun.nio.fs.WindowsDirectoryStream.iterator(Unknown Source)
4.
at com.jmdoudoux.test.nio2.TestNIO2.utilisationDirectoryStream(TestNIO2.java:134)
5.
at com.jmdoudoux.test.nio2.TestNIO2.main(TestNIO2.java:24)
Il est possible de définir un filtre qui sera appliqué sur chacun des éléments du répertoire pour déterminer s'il doit être retourné ou non lors du parcours.
Pour cela, il faut créer une instance de type DirectoryStream.Filter
Exemple ( code Java 7 ) :
01.
public static void utilisationDirectoryStreamAvecFiltre() throws IOException {
02.
Path jdkPath = Paths.get("C:/Program Files/Java/jdk1.7.0_02");
03.
DirectoryStream.Filter
04.
public static final long HUIT_MEGABYTES = 8*1024*1024;
05.
06.
@Override
07.
public boolean accept(Path element) throws IOException {
08.
return Files.size(element) >= HUIT_MEGABYTES;
09.
}
10.
};
11.
12.
try (DirectoryStream
13.
for (Path entry : stream) {
14.
System.out.println(entry);
15.
}
16.
}
17.
}
Résultat :
1.
C:\Program Files\Java\jdk1.7.0_02\src.zip
21.6.2. Le parcours d'une hiérarchie de répertoires
La méthode Files.walkFileTree() permet de parcourir la hiérarchie d'un ensemble de répertoires en utilisant le motif de conception visiteur. Ce type de parcours peut être utilisé pour rechercher, copier, déplacer, supprimer, ... des éléments de la hiérarchie parcourue.
Il faut écrire une classe qui implémente l'interface java.nio.file.FileVisitor
Méthode
Rôle
FileVisitResult postVisitDirectory(T dir, IOException exc)
Le parcours sort d'un répertoire qui vient d'être parcouru ou une exception est survenue durant le parcours
FileVisitResult preVisitDirectory(T dir, BasicFileAttributes attrs)
Le parcours rencontre un répertoire, cette méthode est invoquée avant de parcourir son contenu
FileVisitResult visitFile(T file, BasicFileAttributes attrs)
Le parcours rencontre un fichier
FileVisitResult visitFileFailed(T file, IOException exc)
La visite d'un des fichiers durant le parcours n'est pas possible et une exception a été levée
Il est possible de contrôler les traitements du parcours en utilisant les objets de type FileVisitResult retournée par les méthodes de l'interface FileVisitor.
Les méthodes de l'interface FileVisitor renvoient toutes une valeur qui appartient à l'énumération FileVisitResult. Cette valeur permet de contrôler le processus de parcours de l'arborescence :
CONTINUE : poursuite du parcours
TERMINATE : arrêt immédiat du parcours
SKIP_SUBTREE : inhibe le parcours de la sous-arborescence. Si la méthode preVisitDirectory() renvoie cette valeur, le parcours du répertoire est ignoré
SKIP_SIBLING : inhibe le parcours des répertoires frères. Si la méthode preVisitDirectory() renvoie cette valeur alors le répertoire n'est pas parcouru et la méthode postVisitDirectory() n'est pas invoquée. Si la méthode postVisitDirectory() renvoie cette valeur, alors les autres répertoires frères qui n'ont pas encore été parcourus sont ignorés
Exemple ( code Java 7 ) :
1.
public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) {
2.
if (dir.getFileName().toString().equals("target")) {
3.
return SKIP_SUBTREE;
4.
}
5.
return CONTINUE;
6.
}
L'exemple ci-dessous parcourt l'arborescence et s'arrête dès que le fichier test.txt est trouvé.
Exemple ( code Java 7 ) :
1.
public FileVisitResult visitFile(Path file, BasicFileAttributes attr) {
2.
if (file.getFileName().equals("test.txt")) {
3.
System.out.println("Fichier trouve");
4.
return TERMINATE;
5.
}
6.
return CONTINUE;
7.
}
L'API propose la classe java.nio.file.SimpleFileVisitor qui est une implémentation de l'interface FileVisitor. Le plus simple est donc de créer une classe fille qui hérite de la classe SimpleFileVisitor et de redéfinir les méthodes utiles selon les besoins.
L'exemple ci-dessous affiche tous les fichiers .java en ignorant les répertoires target.
Exemple ( code Java 7 ) :
01.
public static void testWalkFileTree() throws IOException {
02.
final Path repertoire = Paths.get("C:/java/projets");
03.
Files.walkFileTree(repertoire, new SimpleFileVisitor
04.
05.
@Override
06.
public FileVisitResult visitFile(final Path file,
07.
final BasicFileAttributes attrs) throws IOException {
08.
final String nom = file.getFileName().toString();
09.
System.out.println("Fichier : " + nom);
10.
return FileVisitResult.CONTINUE;
11.
}
12.
13.
@Override
14.
public FileVisitResult preVisitDirectory(final Path dir,
15.
final BasicFileAttributes attrs) throws IOException {
16.
FileVisitResult result = FileVisitResult.CONTINUE;
17.
System.out.println("Répertoire : " + dir);
18.
return result;
19.
}
20.
});
21.
}
Pour lancer le parcours de la hiérarchie d'un répertoire, il faut utiliser la méthode walkFileTree() de la classe Files qui propose deux surcharges :
Path walkFileTree(Path start, FileVisitor visitor)
Path walkFileTree(Path start, Set
La première surcharge attend en paramètre le chemin du répertoire qui doit être parcouru et une instance de type FileVisitor qui va encapsuler les traitements du parcours.
La seconde surcharge attend deux paramètres supplémentaires qui permettent de préciser des options sous la forme d'un ensemble de type FileVisitOption et un entier qui permet de limiter le niveau de profondeur du parcours dans la hiérarchie.
L'énumération FileVisitOption ne contient que la valeur FOLLOW_LINKS qui permet de demander de suivre les liens rencontrés lors du parcours. Par défaut, les liens symboliques ne sont pas suivis par le WalkFileTree. Pour suivre les liens symboliques, il faut préciser l'utilisation de l'option FOLLOW_LINKS.
Exemple ( code Java 7 ) :
1.
final Path repertoire = Paths.get("C:/java/projets");
2.
3.
EnumSet
4.
5.
Files.walkFileTree(repertoire, options, Integer.MAX_VALUE, new
6.
SimpleFileVisitor
7.
// ...
8.
});
Si l'option FOLLOW_LINK est utilisée, le walkFileTree est capable de détecter les références circulaires lors du parcours. Dans ce cas, la méthode visitFileFailed() sera invoquée et elle aura une exception de type FileSystemLoopException en paramètre.
Exemple ( code Java 7 ) :
01.
@Override
02.
public FileVisitResult visitFileFailed(Path file, IOException ioe) {
03.
if (ioe instanceof FileSystemLoopException) {
04.
System.err.println("Reference circulaire detectee : " + file);
05.
} else {
06.
ioe.printStackTrace();
07.
}
08.
return FileVisitResult.CONTINUE;
09.
}
Important : il n'est pas possible de présumer de l'ordre de parcours des répertoires.
Si les traitements modifient le système de fichiers, il est important de faire particulièrement attention dans l'implémentation du FileVisitor. Par exemple :
Si le parcours est utilisé pour supprimer une sous-arborescence, il est nécessaire de supprimer les fichiers contenus par un répertoire avant de supprimer le répertoire lui-même.
Si le parcours est utilisé pour copier une sous-arborescence, il faut créer le sous-répertoire avant de copier les fichiers qu'il doit contenir
21.6.3. Les opérations récursives
Les fonctionnalités offertes par la classe Files ne s'appliquent pas de manière récursive : il est nécessaire de parcourir l'arborescence en utilisant une des deux techniques ci-dessus pour réaliser des opérations sur un répertoire.
Par exemple, la méthode size() de la classe Files ne s'applique que sur un fichier. Pour déterminer la taille d'un répertoire (en fait la taille des fichiers qu'il contient), il faut écrire du code qui va parcourir son contenu et cumuler les tailles des fichiers qu'il contient.
Exemple ( code Java 7 ) :
01.
public static long getDirectorySize(final Path repertoire) throws IOException {
02.
final AtomicLong size = new AtomicLong();
03.
if (!Files.isDirectory(repertoire)) {
04.
throw new IllegalArgumentException(
05.
"Le chemin n'est pas celui d'un répertoire");
06.
}
07.
Files.walkFileTree(repertoire, new SimpleFileVisitor
08.
09.
@Override
10.
public FileVisitResult visitFile(Path file, BasicFileAttributes attrs)
11.
throws IOException {
12.
if (Files.isRegularFile(file)) {
13.
size.addAndGet(attrs.size());
14.
}
15.
return FileVisitResult.CONTINUE;
16.
}
17.
18.
@Override
19.
public FileVisitResult preVisitDirectory(Path dir,
20.
BasicFileAttributes attrs) throws IOException {
21.
FileVisitResult resultat = FileVisitResult.CONTINUE;
22.
if (!dir.equals(repertoire)) {
23.
size.addAndGet(getDirectorySize(dir));
24.
resultat = FileVisitResult.SKIP_SUBTREE;
25.
}
26.
return resultat;
27.
}
28.
});
29.
return size.get();
30.
}
Il est possible de télécharger séparément les exemples du JDK : plusieurs de ces exemples situés dans le sous-répertoire sample/nio/file concernent des fonctionnalités utilisant des opérations récursives avec l'API NIO2.
21.7. L'utilisation de systèmes de gestion de fichiers
Un système de gestion de fichiers est encapsulé par un objet de type FileSystem qui permet de créer des objets qui pourront interagir avec lui.
Il faut utiliser la fabrique FileSystems pour obtenir une instance de type FileSystem.
21.7.1. La classe FileSystems
La classe FileSystems est une fabrique pour obtenir des instances de type FileSystem.
La méthode getDefault() renvoie une instance de type FileSystem qui encapsule le système de fichiers de la JVM.
La méthode getFileSystem() renvoie une instance de type FileSystem qui encapsule le système de fichiers dont l'URI est fourni en paramètre.
Plusieurs surcharges de la méthode newFileSystem() permettent de créer une instance spécifique de type FileSystem.
21.7.2. La classe FileSystem
La classe FileSystem encapsule un système de fichiers. C'est essentiellement une fabrique d'instances d'objets dépendants du système encapsulé notamment : Path, PathMatcher, FileStores, WatchService, ...
Pour obtenir une instance de la classe FileSystem qui encapsule le système de fichiers par défaut, il faut utiliser la méthode getDefault() de la classe FileSystems.
Les systèmes de fichiers n'utilisent pas tous le même séparateur dans les chemins de leurs éléments : par exemple, Windows utilise le caractère antislash, les systèmes de type Unix utilisent le caractère slash, ...
Pour connaître le séparateur utilisé par le système, il est possible d'invoquer la méthode getSeparator() de la classe FileSystem.
Exemple ( code Java 7 ) :
1.
private static void testGetSeparator() {
2.
String separator = FileSystems.getDefault().getSeparator();
3.
System.out.println(separator);
4.
}
Résultat :
1.
\
La méthode getRootDirectories() permet d'obtenir un objet de type Iterable
Exemple ( code Java 7 ) :
1.
public static void getRootDirectories() throws IOException {
2.
Iterable
3.
for (Path name: dirs) {
4.
System.err.println(name);
5.
}
6.
}
Résultat :
1.
C:\
21.7.3. La création d'une implémentation de FileSystem
La classe FileSystem est extensible.
Il est par exemple possible de développer ses propres implémentations permettant d'offrir différentes vues d'un système de fichiers (cacher des fichiers sensibles, accès en lecture seule à tous les éléments du système, ...).
Il faut créer une classe qui hérite de la classe FileSystemProvider et une classe qui hérite de la classe FileSystem.
Exemple ( code Java 7 ) :
1.
public class MonFileSystem extends FileSystem {
2.
// ...
3.
}
La prise en compte du FileSystem se fait en utilisant le service Provider de la JVM. Il faut donc packager le Filesystem dans une archive de type jar contenant un sous-répertoire META-INF/services avec un fichier java.nio.file.spi.FileSystemProvider listant les noms pleinement qualifiés des sous-classes de type FileSystemProvider.
L'implémentation d'un FileSystem n'a pas besoin d'être liée à un «vrai» système de fichiers.
21.7.4. Une implémentation de FileSystem pour les fichiers Zip
L'implémentation du JDK propose en standard une implémentation spéciale de la classe FileSystem pour faciliter la manipulation de fichiers compressés au format ZIP. Son utilisation rend la manipulation d'archives de type zip beaucoup plus aisée que l'utilisation des classes du package java.util.zip.
Il faut utiliser la fabrique FileSystems pour créer une instance de type FileSystem en invoquant la méthode newFileSystem() et en lui passant un paramètre une instance de type Path qui encapsule le chemin de l'archive à manipuler.
Il est alors possible d'utiliser cette instance de FileSystem pour obtenir des chemins contenus dans l'archive puisque l'archive est vue elle-même comme un système de fichiers particulier. L'utilisation de ces chemins se fait de la même manière que pour les chemins obtenus d'une instance de type FileSystem encapsulant un système de fichiers du système d'exploitation.
L'exemple ci-dessous affiche le contenu d'un fichier contenu dans une archive de type jar.
Exemple ( code Java 7 ) :
01.
public static void testZip() throws IOException {
02.
// Path de l'archive
03.
final Path jarfile = Paths.get("c:/java/test/archive.jar");
04.
05.
// création d'une instance de FileSystem pour gérer les zip
06.
final FileSystem fs = FileSystems.newFileSystem(jarfile, null);
07.
// Path du fichier à accéder dans l'archive
08.
final Path mf = fs.getPath("META-INF", "MANIFEST.MF");
09.
10.
// lecture et affichage du fichier contenu dans l'archive
11.
try (BufferedReader readBuffer = Files.newBufferedReader(mf,
12.
Charset.defaultCharset())) {
13.
String ligne = "";
14.
while ((ligne = readBuffer.readLine()) != null) {
15.
System.out.println(ligne);
16.
}
17.
}
18.
}
L'extraction d'un fichier d'une archive de type zip se fait simplement en invoquant la méthode copy() de la classe Files en lui passant en paramètres une instance de type Path du chemin dans l'archive et une instance de type Path du chemin cible.
L'exemple ci-dessous extrait un fichier contenu dans une archive de type jar.
Exemple ( code Java 7 ) :
01.
public static void testZip() throws IOException {
02.
// Path de l'archive
03.
final Path jarfile = Paths.get("c:/java/test/archive.jar");
04.
05.
// creation d'une instance de FileSystem pour gérer les zip
06.
final FileSystem fs = FileSystems.newFileSystem(jarfile, null);
07.
08.
// Path du fichier cible
09.
final Path cible = Paths.get("c:/java/test/MANIFEST.MF");
10.
Files.deleteIfExists(cible);
11.
12.
// extraire l'élément de l'archive
13.
Files.copy(fs.getPath("/META-INF/MANIFEST.MF"), cible);
14.
if (Files.exists(cible)) {
15.
System.out.println("fichier " + cible.getFileName() +
16.
" extrait de l'archive " + jarfile);
17.
}
18.
}
Pour créer une archive de type zip vide, il faut créer une instance de type FileSystem en utilisant la méthode newFileSystem() et en lui passant en paramètre :
une URI du chemin de l'archive dont le protocole est jar:file:
une collection de type Map qui contienne une occurrence ayant pour clé "create" et pour valeur "true"
L'exemple ci-dessous crée une archive de type zip vide.
Exemple ( code Java 7 ) :
1.
private static FileSystem creerZipFileSystem(Path zipFile) throws IOException {
2.
final URI uri = URI.create("jar:file:" + zipFile.toUri().getPath());
3.
4.
final Map
5.
env.put("create", "true");
6.
return FileSystems.newFileSystem(uri, env);
7.
}
L'ajout d'un fichier dans une archive se fait en utilisant la méthode copy() de la classe Files avec comme paramètres le chemin de la source et le chemin dans l'archive.
L'exemple ci-dessous ajoute un nouveau fichier dans une nouvelle archive de type zip.
Exemple ( code Java 7 ) :
01.
public static void testAjouterZip() throws IOException {
02.
final Path pathZip = Paths.get("c:/java/test/monarchive.zip");
03.
04.
Files.deleteIfExists(pathZip);
05.
06.
// important : invoquer la méthode close() du FS
07.
try (FileSystem fs = creerZipFileSystem(pathZip)) {
08.
Path source = Paths.get("c:/java/test/monfichier.txt");
09.
Path dest = fs.getPath("/", "monfichier.txt");
10.
Files.copy(source, dest, StandardCopyOption.COPY_ATTRIBUTES);
11.
}
12.
}
Pour que le fichier soit correctement ajouté, il est important d'invoquer la méthode close() sur l'instance de type FileSystem qui encapsule l'archive. Dans l'exemple ci-dessus, cette invocation est assurée par l'utilisation d'un try-with-resource.
21.8. La lecture et l'écriture dans un fichier
La lecture et l'écriture dans un fichier se font toujours de la même façon avec NIO2 mais l'API propose des méthodes utilitaires pour faciliter le travail.
La gestion des opérations de types entrées/sorties a évolué au fur et à mesure des versions de Java.
IO
NIO
NIO2
Java 1.0 et 1.1
Java 1.4 (JSR 151)
Java 7 (JSR 203)
Synchrone bloquant
Synchrone non bloquant
Asynchrone non bloquant
File
InputStream
OutputStream
Reader (Java 1.1)
Writer (Java 1.1)
Socket
RandomAccessFile
FileChannel
SocketChannel
ServerSocketChannel
(Charset, Selector,
ByteBuffer)
Path
AsynchronousFileChannel
AsynchronousByteChannel
AsynchronousSocketChannel
AsynchronousServerSocketChannel
SeekableByteChannel
La classe Files propose plusieurs méthodes pour faciliter la lecture ou l'écriture de fichiers et de flux selon les besoins allant des plus simples aux plus complexes.
Les méthodes readAllBytes() et readAllLines() permettent de lire l'intégralité du contenu d'un fichier respectivement d'octets et texte. Deux surcharges de la méthode write() permettent d'écrire l'intégralité d'un fichier. Ces méthodes sont à réserver pour de petits fichiers.
Les méthodes newBufferedReader() et newBufferedWriter() sont des helpers pour faciliter la création d'objets de types BufferedReader et BufferedWriter permettant la lecture et l'écriture de fichiers de type texte en utilisant un tampon.
Les méthodes newInputStream() et newOutputStream() sont des helpers pour faciliter la création d'objets permettant la lecture et l'écriture de fichiers d'octets.
Ces quatre méthodes sont des helpers pour créer des objets du package java.io.
La méthode newByteChannel() est un helper pour créer un objet de type SeekableByteChannel.
La classe FileChannel propose des fonctionnalités avancées sur l'utilisation d'un fichier (verrous, mapping direct à une zone de la mémoire, ...) : cette classe a été enrichie pour fonctionner avec NIO2.
21.8.1. Les options d'ouverture d'un fichier
L'énumération StandardOpenOption implémente l'interface OpenOption et définit les options d'ouverture standard d'un fichier :
Valeur
Rôle
APPEND
Si le fichier est ouvert en écriture alors les données sont ajoutées au fichier. Cette option doit être utilisée avec les options CREATE ou WRITE
CREATE
Créer un nouveau fichier s'il n'existe pas sinon le fichier est ouvert
CREATE_NEW
Créer un nouveau fichier : si le fichier existe déjà alors une exception est levée
DELETE_ON_CLOSE
Supprimer le fichier lorsque son flux associé est fermé : cette option est utile pour des fichiers temporaires
DSYNC
Demander l'écriture synchronisée des données dans le système de stockage sous-jacent (pas d'utilisation des tampons du système)
READ
Ouvrir le fichier en lecture
SPARSE
Indiquer au système que le fichier est clairsemé ce qui peut lui permettre de réaliser certaines optimisations si l'option est supportée par le système de fichiers (c'est notamment le cas avec NTFS)
SYNC
Demander l'écriture synchronisée des données et des métadonnées dans le système de stockage sous-jacent
TRUNCATE_EXISTING
Si le fichier existe et qu'il est ouvert en écriture alors il est vidé. Cette option doit être utilisée avec l'option WRITE
WRITE
Ouvrir le fichier en écriture
Ces options sont utilisables avec toutes les méthodes qui ouvrent des fichiers. Elles ne sont pas toutes mutuellement exclusives.
21.8.2. La lecture et l'écriture de l'intégralité d'un fichier
La classe Files propose les méthodes readAllLines() et readAllBytes() qui renvoient respectivement une collection de type List
La méthode readAllLines() de la classe Files permet de lire l'intégralité d'un fichier et de renvoyer son contenu sous la forme d'une collection de chaînes de caractères.
Exemple ( code Java 7 ) :
1.
List
2.
FileSystems.getDefault().getPath("monfichier.txt"), StandardCharsets.UTF_8);
3.
for (String ligne : lignes)
4.
System.out.println(ligne);
La méthode readAllLines() attend en paramètre un objet de type Path qui encapsule le chemin du fichier à lire et un objet de type Charset qui précise le jeu d'encodage de caractères du fichier. Elle s'occupe d'ouvrir le fichier, lire le contenu et fermer le flux.
La méthode readAllBytes() de la classe Files permet de lire l'intégralité d'un fichier et renvoyer son contenu sous la forme d'un tableau d'octets.
Exemple ( code Java 7 ) :
1.
Path file = FileSystems.getDefault().getPath("monfichier.bin");
2.
byte[] contenu = Files.readAllBytes(file);
La méthode write() permet d'écrire le contenu d'un fichier. Elle possède deux surcharges :
Path write(Path path, byte[] bytes, OpenOption... options)
Path write(Path path, Iterable lines, Charset cs, OpenOption... options)
Exemple ( code Java 7 ) :
1.
final Path pathSource = Paths.get("c:/java/source.txt");
2.
final Path pathCible = Paths.get("c:/java/cible.txt");
3.
final List
4.
Files.write(pathCible, lignes, Charset.defaultCharset());
Exemple ( code Java 7 ) :
1.
final Path pathSource = Paths.get("c:/java/source.bin");
2.
final Path pathCible = Paths.get("c:/java/cible.bin");
3.
// lire et écrire tout le fichier
4.
final byte[] bytes = Files.readAllBytes(pathSource);
5.
Files.write(pathCible, bytes);
21.8.3. La lecture et l'écriture bufférisée d'un fichier
Avant Java 7, pour lire un fichier avec un tampon, il fallait invoquer le constructeur de la classe BufferedReader en lui passant en paramètre un objet de type Reader.
Exemple :
1.
BufferedReader in = new BufferedReader(new FileReader("monfichier.txt"));
A partir de Java 7, il est possible d'utiliser la méthode newBufferedReader() de la classe Files.
Exemple ( code Java 7 ) :
1.
BufferedReader in = Files.newBufferedReader(Paths.get("monfichier.txt"),
2.
Charset.forName("UTF-8"));
Le résultat est quasiment le même mais il est nécessaire de préciser le jeu d'encodage des caractères. La classe FileReader utilise toujours le jeu par défaut du système. Même si ce n'est pas une bonne pratique, il est possible d'obtenir ce jeu d'encodage de caractères en invoquant la méthode java.nio.charset.Charset.defaultCharset().
La méthode newBufferedReader() de la classe Files renvoie un objet de type BufferedReader qui permet de lire le fichier dont le chemin et le jeu de caractères d'encodage sont fournis en paramètres.
Exemple ( code Java 7 ) :
01.
public static void testNewBufferedReader() throws IOException {
02.
Path sourcePath = Paths.get("C:/java/temp/monfichier.txt");
03.
try (BufferedReader reader = Files.newBufferedReader(sourcePath,
04.
StandardCharsets.UTF_8)) {
05.
String line = null;
06.
while ((line = reader.readLine()) != null) {
07.
System.out.println(line);
08.
}
09.
}
10.
}
La méthode newBufferedReader() ouvre un fichier de type texte pour des lectures avec un tampon. Elle retourne un objet de type BufferedReader.
Exemple ( code Java 7 ) :
01.
Path fichier = Paths.get("monfichier.txt");
02.
Charset charset = Charset.forName("US-ASCII");
03.
try (BufferedReader reader = Files.newBufferedReader(fichier, charset)) {
04.
String line = null;
05.
while ((line = reader.readLine()) != null) {
06.
System.out.println(line);
07.
}
08.
} catch (IOException ioe) {
09.
ioe.printStacktrace();
10.
}
La méthode newBufferedWriter() ouvre un fichier de type texte pour des écritures avec un tampon. Elle retourne un objet de type BufferedWriter.
Exemple ( code Java 7 ) :
1.
Path fichier = Paths.get("monfichier.txt");
2.
Charset charset = Charset.forName("US-ASCII");
3.
String contenu = "Contenu du fichier";
4.
try (BufferedWriter writer = Files.newBufferedWriter(fichier, charset)) {
5.
writer.write(contenu, 0, contenu.length());
6.
} catch (IOException ioe) {
7.
ioe.printStacktrace();
8.
}
21.8.4. La lecture et l'écriture d'un flux d'octets
Les méthodes newInputStream() et newOutputStream() permettent d'obtenir une instance de type InputStream et une instance de type OutputStream sur le fichier dont le chemin est fourni en paramètre :
Méthode
Rôle
InputStream newInputStream(Path path, OpenOption... options)
Créer un objet de type InputStream
OutputStream newOutputStream(Path path, OpenOption... options)
Créer un objet de type OutputStream
Les méthodes newInputStream() et newOutputStream() attendent en paramètres un objet de type Path et un varargs de type OpenOption.
La méthode newInputStream() ouvre un fichier pour des lectures sans tampon. Elle retourne un objet de type InputStream.
Exemple ( code Java 7 ) :
01.
public static void testNewInputStream() throws IOException {
02.
Path path = Paths.get("c:/java/test/monfichier.txt");
03.
try (InputStream in = Files.newInputStream(path);
04.
BufferedReader reader = new BufferedReader(new InputStreamReader(in))) {
05.
String line = null;
06.
while ((line = reader.readLine()) != null) {
07.
System.out.println(line);
08.
}
09.
} catch (IOException x) {
10.
System.err.println(x);
11.
}
12.
}
La méthode newOutputStream() ouvre un fichier pour des écritures sans tampon. Elle retourne un objet de type OutputStream. Si aucun paramètre de type OpenOption n'est précisé, la méthode va utiliser les paramètres CREATE et TRUNCATE_EXISTING par défaut (créer le fichier s'il n'existe pas et le vider s'il existe).
Exemple ( code Java 7 ) :
1.
public static void testNewOutputStream() throws IOException {
2.
Path path = Paths.get("c:/java/test/monfichier.txt");
3.
try (OutputStream out = Files.newOutputStream(path,
4.
StandardOpenOption.TRUNCATE_EXISTING, StandardOpenOption.WRITE)) {
5.
out.write('X');
6.
}
7.
}
21.8.5. La lecture et l'écriture d'un fichier avec un channel
L'API Java NIO propose de réaliser des opérations d'entrées/sorties utilisant des channels et des tampons (ByteBuffer) ce qui améliore les performances par rapport à l'API Java IO.
Par défaut les flux de java.io lisent ou écrivent uniquement un octet ou un caractère à la fois.
Les opérations de lectures/écritures de java.nio utilisent un tampon (ByteBuffer). L'interface ByteChannel propose des fonctionnalités de base pour de telles lectures ou écritures.
La méthode newByteChannel() de la classe Files renvoie une instance d'un channel NIO de type SeekableByteChannel. Elle possède deux surcharges :
SeekableByteChannel newByteChannel(Path path, OpenOption... options)
SeekableByteChannel newByteChannel(Path path, Set options, FileAttribute... attrs)
Ces deux surcharges permettent d'ouvrir ou de créer un fichier et de lui associer un channel en fonction des paramètres d'ouverture de type OpenOption fournis. Par défaut le channel est ouvert en lecture (option READ).
Exemple ( code Java 7 ) :
01.
final Path path = Paths.get("C:/java/test/fichier.bin");
02.
03.
Files.deleteIfExists(path);
04.
05.
try (SeekableByteChannel sbc = Files.newByteChannel(path,
06.
StandardOpenOption.WRITE, StandardOpenOption.SYNC)) {
07.
08.
// ...
09.
}
L'interface java.nio.channels.SeekableByteChannel ajoute à l'interface ByteChannel la possibilité de gérer une position dans le channel, de vider un channel et d'obtenir la taille du fichier associé au channel. Cela permet de se déplacer dans le channel pour réaliser une opération de lecture ou d'écriture sans avoir à parcourir les données jusqu'à la position désirée. Un SeekableByteChannel est donc un channel qui possède des fonctionnalités similaires à celles proposées par la classe java.io.RandomAccessFile.
L'interface SeekableByteChannel hérite des interfaces : AutoCloseable, ByteChannel, Channel, Closeable, ReadableByteChannel et WritableByteChannel.
Elle propose plusieurs méthodes pour permettre de se déplacer dans le fichier avant de réaliser une opération de lecture ou d'écriture.
Méthode
Rôle
long position()
Retourner la position courante dans le channel
SeekableByteChannel position(long newPosition)
Changer la position dans le channel
int read(ByteBuffer dst)
Lire un ensemble d'octets du channel dans le tampon fourni en paramètre. Retourne le nombre d'octets lus ou -1 si la fin du channel est atteinte
long size()
Retourner la taille en octets du flux auquel le channel est connecté
SeekableByteChannel truncate(long size)
Tronquer le contenu de l'élément sur lequel le channel est connecté à la taille fournie en paramètre. Cela permet de redimensionner la taille du flux associé au channel avec la valeur fournie en paramètre
int write(ByteBuffer src)
Ecrire les octets fournis en paramètre à la position courante dans le channel
La méthode read() tente une lecture pour remplir le nombre d'octets du tampon passé en paramètre. Elle renvoie -1 si la fin du flux est atteinte. La position courante dans le channel est augmentée de la taille des données lues.
La méthode write() écrit les octets du tampon passé en paramètre à partir de la position courante dans le channel. Si le fichier est ouvert avec l'option APPEND, alors la position courante est située à la fin du fichier. Elle renvoie le nombre d'octets écrits. La position courante dans le channel est augmentée de la taille des données écrites.
La surcharge de la méthode position() qui attend un paramètre de type long permet de déplacer la position courante dans le channel. Elle renvoie le channel lui-même pour permettre un chaînage des appels de cette méthode. La taille du flux connecté au channel n'est pas modifiée si la valeur fournie en paramètre est supérieure à sa taille totale. L'utilisation de cette méthode n'est pas recommandée avec un channel ouvert avec l'option APPEND.
La méthode truncate() permet de réduire la taille totale du flux connecté au channel. Si la taille fournie en paramètre est inférieure à la taille totale courante, alors les octets entre la taille fournie et la taille totale sont perdus. Si la taille fournie est supérieure ou égale à la taille du flux connecté au channel alors l'invocation de la méthode n'a aucun effet. Une implémentation de cette interface peut interdire l'utilisation de cette méthode si le channel est ouvert avec l'option APPEND.
Exemple ( code Java 7 ) :
01.
final ByteBuffer donneesBonjour = ByteBuffer.wrap("Bonjour".getBytes());
02.
final ByteBuffer donneesBonsoir = ByteBuffer.wrap("Bonsoir".getBytes());
03.
04.
final Path path = Paths.get("C:/java/test/fichier.bin");
05.
06.
Files.deleteIfExists(path);
07.
try (FileChannel fileChannel = FileChannel.open(path,
08.
StandardOpenOption.CREATE, StandardOpenOption.WRITE,
09.
StandardOpenOption.SYNC)) {
10.
fileChannel.position(100);
11.
fileChannel.write(donneesBonjour);
12.
}
13.
14.
try (SeekableByteChannel sbc = Files.newByteChannel(path,
15.
StandardOpenOption.WRITE, StandardOpenOption.SYNC)) {
16.
sbc.position(200);
17.
sbc.write(donneesBonsoir);
18.
}
La méthode Files.newByteChannel() permet de créer une instance de type SeekableByteChannel. Si le fichier connecté au channel est sur le système de fichiers par défaut, il est possible de caster l'objet retourné en un objet de type FileChannel.
La classe abstraite FileChannel propose des fonctionnalités avancées à utiliser sur un channel connecté à un fichier :
des octets peuvent être lus ou écrits sans modifier la position courante dans le channel
une région du fichier peut être mappée directement en mémoire (cette fonctionnalité est intéressante pour manipuler de gros fichiers)
l'écriture de données peut être forcée pour être faite directement sur le système de stockage afin d'éviter une perte de données en cas de crash du système
une région du fichier peut être verrouillée pour empêcher l'accès par d'autres applications
21.9. Les liens et les liens symboliques
Il existe deux types de liens :
liens physiques (hard links) : ils permettent de faire référence à un élément physique du système de fichiers qui doit exister. Si le fichier cible est modifié alors le lien est aussi modifié.
liens symboliques (symbolic links) : ils permettent de faire référence à un autre élément du système de fichiers. Si l'élément cible est supprimé alors le lien existe toujours mais il est invalide.
La classe Files propose deux méthodes pour créer des liens physiques et des liens symboliques.
Méthode
Rôle
Path createSymbolicLink(Path link, Path target, FileAttribute... attrs)
Créer un lien symbolique vers un élément
Path createLink(Path link, Path existing)
Créer un lien physique
21.9.1. La création d'un lien physique
Les liens physiques (hard links) possèdent quelques restrictions :
le fichier cible doit exister
le fichier cible doit être sur la même partition
il possède les mêmes attributs que le fichier cible
Pour créer un lien, il faut invoquer la méthode createLink() de la classe Files qui attend en paramètre deux objets de type Path : le premier est le chemin du lien, le second est le chemin du fichier cible qui s'il n'existe pas lèvera une exception de type NoSuchFileException.
Exemple ( code Java 7 ) :
1.
public static void testCreateLink() throws IOException {
2.
Path lien = Paths.get("c:/java/test/monlien.lnk");
3.
Path cible = Paths.get("c:/java/test/monfichier.txt");
4.
Files.createLink(lien, cible);
5.
}
21.9.2. La création d'un lien symbolique
La méthode createSymbolicLink() de la classe Files permet de créer un lien symbolique. Le premier paramètre de type Path est le chemin du lien symbolique. Le second paramètre de type Path est le chemin vers le fichier ou le répertoire cible. Le paramètre de type varargs FileAttributes permet de préciser les options du lien qui seront utilisées lors de sa création.
Exemple ( code Java 7 ) :
1.
Path lien = Paths.get("/home/jm/monlien");
2.
Path cible = Paths.get("/home/jm/monfichier.txt");
3.
Files.createSymbolicLink(lien, cible);
4.
if (Files.isSameFile(lien, cible)) {
5.
System.out.println("Identique");
6.
} else {
7.
System.out.println("Non identique");
8.
}
L'utilisation des liens symboliques est conditionnée par le fait que le système d'exploitation sous-jacent propose un support de ces liens. Si le système sous-jacent ne supporte pas les liens symboliques, une exception de type UnsupportedOperationException est levée lors de l'invocation de la méthode createSymbolicLink().
Exemple sous Windows XP
Exemple ( code Java 7 ) :
01.
public static void testSymbolicLink() {
02.
Path newLink = Paths.get("C:/test_link");
03.
Path target = Paths.get("C:/Users/test");
04.
try {
05.
Files.createSymbolicLink(newLink, target);
06.
} catch (IOException ioe) {
07.
ioe.printStackTrace();
08.
} catch (UnsupportedOperationException uoe) {
09.
// Le systeme de fichiers ne supporte pas les liens symboliques.
10.
uoe.printStackTrace();
11.
}
12.
}
Le support des liens symboliques est aussi contrôlé par un SecurityManager en utilisant l'option LinkPermission("symbolic") : leur support est désactivé par défaut. Une exception de type SecurityException peut donc être levée si un SecurityManager est utilisé et que les droits adéquats ne sont pas activés.
21.9.3. L'utilisation des liens et des liens symboliques
La méthode toRealPath() de l'interface Path permet de retourner un chemin dont les liens symboliques contenus dans le chemin sont résolus.
La méthode isSymbolicLink() de l'interface Path permet de déterminer si l'élément précisé par le chemin est un lien symbolique ou non.
La méthode readSymbolicLink() de la classe Files renvoie le chemin de la cible du lien symbolique ou lève une exception de type NotLinkException si l'élément dont le chemin fourni en paramètre n'est pas un lien symbolique.
La suppression d'un lien se fait en utilisant la méthode delete() de la classe Files : dans ce cas, c'est le lien qui est supprimé et non le fichier cible.
Certaines méthodes de la classe Files attendent en paramètre un varargs de type LinkOption. L'option LinkOption.NOFOLLOW_OPTIONS permet de demander de ne pas suivre les liens pour réaliser l'action demandée.
21.10. La gestion des attributs
Les éléments d'un système de fichiers possèdent des métadonnées généralement nommés attributs : le type d'éléments (fichier, répertoire, lien), la taille, la date de création et de modification, les permissions d'utilisation, ... Le nombre de ces métadonnées et la façon dont elles sont gérées sont dépendantes du système d'exploitation.
NIO 2 permet de gérer les permissions sur les fichiers. Malheureusement, ces permissions sont dépendantes du système de fichiers sous-jacent. NIO 2 propose des classes dédiées pour chaque système de fichiers supporté qui sont regroupées dans le package java.nio.file.attribute.
L'accès aux métadonnées a été enrichi avec NIO 2 : certains attributs de base sont accessibles par la classe Files d'autres sont accessibles au travers de vues.
L'implémentation par défaut propose plusieurs vues pour les principaux types de système d'exploitation :
Basic : cette vue est commune à tous les systèmes d'exploitation
Dos : cette vue est dédiée aux systèmes d'exploitation Windows
Posix : cette vue est dédiée aux systèmes d'exploitation de type Unix like avec notamment une gestion sur des permissions adaptées à ce type de système
Il est aussi possible qu'une implémentation spécifique soit fournie par un tiers ou encore, de développer sa propre implémentation.
21.10.1. La gestion individuelle des attributs
La classe Files propose plusieurs méthodes pour obtenir individuellement certains de ces attributs pour un élément dont le chemin est fourni en paramètre.
Méthode
Rôle
boolean isDirectory(Path, LinkOption)
Renvoyer un booléen qui précise si l'élément est un répertoire
boolean isRegularFile(Path, LinkOption...)
Renvoyer un booléen qui précise si l'élément est un fichier
boolean isSymbolicLink(Path)
Renvoyer un booléen qui précise si l'élément est un lien symbolique
boolean isHidden(Path)
Renvoyer un booléen qui précise si l'élément est caché
FileTime getLastModifiedTime(Path, LinkOption...)
Renvoyer la date/heure de dernière modification de l'élément
Path setLastModifiedTime(Path, FileTime)
Modifier la date de dernière modification de l'élément
UserPrincipal getOwner(Path, LinkOption...)
Renvoyer le propriétaire du fichier
Path setOwner(Path, UserPrincipal)
Modifier le propriétaire du fichier
Set
Renvoyer les droits d'un élément d'un système de type Unix
Path setPosixFilePermissions(Path, Set
Modifier les droits d'un élément d'un système de type Unix
Object getAttribute(Path, String, LinkOption...)
Obtenir la valeur d'un attribut de l'élément
Path setAttribute(Path, String, Object, LinkOption...)
Modifier la valeur d'un attribut de l'élément
boolean isExecutable()
Renvoyer un booléen qui précise si l'élément peut être exécuté
boolean isReadable()
Renvoyer un booléen qui précise si l'élément peut être lu
boolean isWritable()
Renvoyer un booléen qui précise si l'élément peut être modifié
long size(Path)
Renvoyer la taille en octets d'un fichier
Il est possible d'utiliser la méthode getOwner(Path) de la classe Files pour obtenir un objet de type UserPrincipal qui encapsule le propriétaire du fichier.
Exemple ( code Java 7 ) :
1.
public static void testGetOwner() throws IOException {
2.
Path fichier = Paths.get("C:/java/temp/monfichier.txt");
3.
UserPrincipal owner = Files.getOwner(fichier);
4.
System.out.println(owner);
5.
}
Résultat :
1.
THINKPAD_X60S\jm (User)
21.10.2. La gestion de plusieurs attributs
Si l'application a besoin de plusieurs attributs d'un même élément, il est plus efficace d'utiliser une des surcharges de la méthode readAttributes() qui renvoie un objet encapsulant des attributs d'une même famille. Les performances peuvent être dégradées si le système de fichiers est consulté plusieurs fois pour obtenir des attributs.
Méthode
Rôle
Map
Renvoyer une collection d'attributs lus en une seule opération
A readAttributes(Path, Class, LinkOption...)
Renvoyer un objet qui encapsule les attributs lus en une seule opération. Le type de cet objet est précisé en paramètre
Exemple ( code Java 7 ) :
01.
public static void lectureBasicAttributs() {
02.
Path monFichier = Paths.get("C:/Users/jm/AppData/Local/Temp/monfichier.txt");
03.
BasicFileAttributes basicAttrs;
04.
try {
05.
basicAttrs = Files.readAttributes(monFichier, BasicFileAttributes.class);
06.
07.
System.out.println("creationTime = " + basicAttrs.creationTime());
08.
System.out.println("lastAccessTime = " + basicAttrs.lastAccessTime());
09.
System.out.println("lastModifiedTime = " + basicAttrs.lastModifiedTime());
10.
System.out.println("isDirectory = " + basicAttrs.isDirectory());
11.
System.out.println("isOther = " + basicAttrs.isOther());
12.
System.out.println("isRegularFile = " + basicAttrs.isRegularFile());
13.
System.out.println("isSymbolicLink = " + basicAttrs.isSymbolicLink());
14.
System.out.println("size = " + basicAttrs.size());
15.
System.out.println("fileKey = " + basicAttrs.fileKey());
16.
} catch (IOException ex) {
17.
ex.printStackTrace();
18.
}
19.
}
Résultat :
01.
creationTime = 2011-07-19T14:12:07.916077Z
02.
lastAccessTime = 2011-07-19T14:12:07.916077Z
03.
lastModifiedTime = 2011-07-23T16:39:05.957393Z
04.
isDirectory = false
05.
isOther = false
06.
isRegularFile = true
07.
isSymbolicLink = false
08.
size = 16
09.
fileKey = null
Pour obtenir une instance de type BasicFileAttributes, il faut invoquer la méthode readAttributes() de la classe Files en lui passant en paramètre le chemin du fichier et une instance de type Class pour la classe BasicFileAttributes. Il est aussi possible de préciser des options sous la forme d'un varargs de l'énumération de type LinkOption.
La valeur LinkOption.NOFOLLOW_LINKS indique de ne pas suivre les liens symboliques.
La méthode readAttributes() permet de lire en une seule opération plusieurs attributs encapsulés dans l'objet retourné lors de son invocation, ce qui est plus efficace que de lire ces attributs un par un.
Les attributs creationTime, lastModifiedTime et lastAccessTime encapsulés dans la classe BasicFilesAttributes sont de type java.nio.file.attribute.FileTime qui encapsule un horodatage.
Il est possible de créer une instance de la classe FileTime en utilisant les méthodes :
from(long, TimeUnit) : créer une instance à partir de la valeur et de l'unité fournies en paramètre
fromMillis(long) : créer une instance à partir du nombre de millisecondes fourni en paramètre
Exemple ( code Java 7 ) :
1.
public static void testSetLastModifiedTime() throws IOException {
2.
Path fichier = Paths.get("c:/java/test/monfichier.txt");
3.
long currentTime = System.currentTimeMillis();
4.
FileTime ft = FileTime.fromMillis(currentTime);
5.
Files.setLastModifiedTime(fichier, ft);
6.
}
La méthode fileKey() renvoie un objet qui encapsule une clé unique du fichier dans le système de fichiers si celui-ci supporte cette fonctionnalité sinon elle renvoie null.
21.10.3. L'utilisation des vues
Les différents types de systèmes de fichiers possèdent des attributs communs mais possèdent aussi des attributs spécifiques. La notion de vue regroupe plusieurs attributs ce qui permet d'obtenir ces attributs en une fois. L'API propose en standard plusieurs vues qui sont spécialisées :
BasicFileAttributeView : propose une vue qui contient des attributs communs à tous les systèmes de fichiers
DosFileAttributeView : propose une vue qui permet un support des quatre attributs spécifiques à un système de fichiers de type DOS (readonly, hidden, system et archive)
PosixFileAttributeView : propose une vue qui permet un support des attributs spécifiques à un système de fichiers de type Posix notamment la gestion des droits pour le propriétaire, le groupe et les autres utilisateurs.
FileOwnerAttributeView : propose une vue qui permet une gestion du propriétaire de l'élément qui correspond par défaut à celui qui a créé l'élément
AclFileAtributeView : propose une vue qui permet le support de la gestion des droits de type ACL
UserDefinedFileAttributeView : propose une vue qui permet le support de métadonnées spécifiques à un système de fichiers
Une vue peut permettre un accès en lecture seule aux données ou permettre leur mise à jour.
Un système de fichier ne peut être supporté que par la BasicFileAttributeView ou être supporté par plusieurs vues. Un système de fichiers peut même proposer une ou plusieurs vues spécifiques qui ne sont pas fournies en standard par l'API.
Pour obtenir une vue spécifique, il faut utiliser la méthode getFileAttributeView() de la classe Files en précisant le type de la vue souhaitée.
Exemple ( code Java 7 ) :
01.
public static void testBasicFileAttributeView() throws IOException {
02.
Path path = Paths.get("c:/java/test/monfichier.txt");
03.
BasicFileAttributeView basicView = Files.getFileAttributeView(path,
04.
BasicFileAttributeView.class);
05.
if (basicView != null) {
06.
BasicFileAttributes basic = basicView.readAttributes();
07.
08.
System.out.println("isRegularfile " + basic.isRegularFile());
09.
System.out.println("isDirectory " + basic.isDirectory());
10.
System.out.println("isSymbolicLink " + basic.isSymbolicLink());
11.
System.out.println("isOther " + basic.isOther());
12.
System.out.println("size " + basic.size());
13.
System.out.println("creationTime " + basic.creationTime());
14.
System.out.println("lastAccesstime " + basic.lastAccessTime());
15.
System.out.println("lastModifiedTime " + basic.lastModifiedTime());
16.
}
17.
}
Les informations de la vue basic peuvent aussi être obtenues en utilisant la classe Files : cependant l'utilisation de la vue permet d'obtenir toutes les informations avec un seul accès à l'élément du système d'exploitation.
21.10.4. La gestion des permissions DOS
La classe DosFileAttributes encapsule les attributs d'un élément d'un système de fichiers de type DOS : read only, hidden, archive et system.
Exemple ( code Java 7 ) :
01.
public static void testDosFileAttributes() throws IOException {
02.
Path fichier = Paths.get("C:/java/temp/monfichier.txt");
03.
try {
04.
DosFileAttributes attr = Files.readAttributes(fichier,
05.
DosFileAttributes.class);
06.
System.out.println("isReadOnly = " + attr.isReadOnly());
07.
System.out.println("isHidden = " + attr.isHidden());
08.
System.out.println("isArchive = " + attr.isArchive());
09.
System.out.println("isSystem = " + attr.isSystem());
10.
} catch (UnsupportedOperationException ueo) {
11.
ueo.printStackTrace();
12.
}
13.
}
Résultat :
1.
isReadOnly = false
2.
isHidden = false
3.
isArchive = true
4.
isSystem = false
Il est aussi possible d'utiliser les méthodes getAttribute() et setAttribute() de la classe Files. L'inconvénient de ces méthodes est que l'attribut concerné est fourni sous la forme d'une chaîne de caractères. Celle-ci doit être composée du nom de la vue suivi du caractère deux points suivi du nom de l'attribut.
Exemple ( code Java 7 ) :
01.
public static void testGetFileAttribute() throws IOException {
02.
Path fichier = Paths.get("C:/java/temp/monfichier.txt");
03.
try {
04.
System.out.println("isReadOnly = " +
05.
Files.getAttribute(fichier,"dos:readonly", LinkOption.NOFOLLOW_LINKS));
06.
System.out.println("isHidden = " +
07.
Files.getAttribute(fichier,"dos:hidden", LinkOption.NOFOLLOW_LINKS));
08.
System.out.println("isArchive = " +
09.
Files.getAttribute(fichier,"dos:archive",LinkOption.NOFOLLOW_LINKS));
10.
System.out.println("isSystem = " +
11.
Files.getAttribute(fichier,"dos:system", LinkOption.NOFOLLOW_LINKS));
12.
} catch (UnsupportedOperationException ueo) {
13.
ueo.printStackTrace();
14.
}
15.
}
Si le nom de l'attribut fourni en paramètre n'est pas supporté alors une exception de type IllegalArgumentException est levée.
Exemple ( code Java 7 ) :
01.
public static void testGetFileAttribute() throws IOException {
02.
Path fichier = Paths.get("C:/java/temp/monfichier.txt");
03.
try {
04.
System.out.pr intln("isReadOnly = " +
05.
Files.getAttribute(fichier,"dos:readolny", LinkOption.NOFOLLOW_LINKS));
06.
} catch (UnsupportedOperationException ueo) {
07.
ueo.printStackTrace();
08.
}
09.
}
Résultat :
01.
Exception
02.
in thread "main" java.lang.IllegalArgumentException: 'readolny' not
03.
recognized
04.
at sun.nio.fs.AbstractBasicFileAttributeView$AttributesBuilder.
05.
at sun.nio.fs.AbstractBasicFileAttributeView$AttributesBuilder.create(Unknown Source)
06.
at sun.nio.fs.WindowsFileAttributeViews$Dos.readAttributes(Unknown Source)
07.
at sun.nio.fs.AbstractFileSystemProvider.readAttributes(Unknown Source)
08.
at java.nio.file.Files.readAttributes(Unknown Source)
09.
at java.nio.file.Files.getAttribute(Unknown Source)
10.
at com.jmdoudoux.test.nio2.TestNIO2.testGetFileAttribute(TestNIO2.java:385)
11.
at com.jmdoudoux.test.nio2.TestNIO2.main(TestNIO2.java:57)
La méthode setAttribute() de la classe Files permet de modifier un attribut d'un élément du système de fichiers.
Exemple ( code Java 7 ) :
1.
public static void testSetFileAttribute() throws IOException {
2.
Path fichier = Paths.get("C:/java/temp/monfichier.txt");
3.
try {
4.
Files.setAttribute(fichier,"dos:hidden", false);
5.
} catch (UnsupportedOperationException ueo) {
6.
ueo.printStackTrace();
7.
}
8.
}
21.10.5. La gestion des permissions Posix
La gestion des permissions de type Posix se fait sur trois niveaux : propriétaire, groupe et autres utilisateurs.
Avant Java 7, la modification des attributs d'un fichier sur système POSIX devait se faire en utilisant la méthode System.exec() ou en invoquant une méthode native.
Avec NIO 2, il faut utiliser les classes PosixFilePermission et PosixFilePermissions pour gérer les permissions des systèmes de fichiers respectant la norme POSIX.
Exemple ( code Java 7 ) :
1.
Path monFichier = Paths.get("/tmp/monfichier.txt");
2.
Set
3.
PosixFilePermissions.fromString("rw-rw-r--");
4.
FileAttribute
5.
PosixFilePermissions.asFileAttribute(filePermissions);
6.
Files.createFile(monFichier, fileAttribute);
Attention : les attributs réellement positionnés sur le fichier peuvent être différents en fonction de règles définies sur le système de fichiers comme par exemple l'utilisation d'un umask sous un système de type Unix.
L'interface PosixFileAttributes qui hérite de l'interface BasicFileAttributes propose des méthodes pour obtenir le propriétaire, le groupe de l'élément du système de fichiers et les permissions.
Méthode
Rôle
UserPrincipal owner()
Renvoyer le propriétaire
GroupPrincipal()
Renvoyer le groupe
Set
Renvoyer les permissions de lecture/écriture/exécution du propriétaire, du groupe et des autres
Exemple ( code Java 7 ) :
1.
Path fichier = Paths.get("/home/jm/test.txt");
2.
PosixFileAttributes attrs = Files.readAttributes(fichier, PosixFileAttributes.class);
3.
UserPrincipal owner = attrs.owner();
4.
GroupPrincipal group = attrs.group();
5.
System.out.println("Le fichier appartient à " + owner + ":" + group);
L'énumération PosixFilePermission contient des valeurs pour gérer les droits de lecture, écriture et exécution pour le propriétaire, le groupe et les autres : OWNER_READ, OWNER_WRITE, OWNER_EXECUTE, GROUP_READ, GROUP_WRITE, GROUP_EXECUTE, OTHERS_READ, OTHERS_WRITE, OTHERS_EXECUTE.
Les permissions sont encapsulées dans une collection de type Set d'éléments de type PosixFilePermission.
Exemple ( code Java 7 ) :
1.
PosixFilePermission[] permissionsArray = {
2.
PosixFilePermission.OWNER_READ, PosixFilePermission.OWNER_WRITE,
3.
PosixFilePermission.GROUP_READ, PosixFilePermission.GROUP_WRITE };
4.
Set
5.
Arrays.asList(permissionsArray));
Les gestions des permissions peut se faire en manipulant directement la collection.
Exemple ( code Java 7 ) :
1.
Set
2.
permissions.add(PosixFilePermission.OTHERS_READ);
3.
permissions.remove(PosixFilePermission.GROUP_WRITE);
4.
Files.setPosixFilePermissions(path, permissions);
La classe Files propose la méthode getPosixFilePermissions(Path, LinkOption ...) qui renvoie une collection de type Set
La classe PosixFilePermissions propose des méthodes pour faciliter la manipulation d'un ensemble de permissions.
Méthode
Rôle
static FileAttribute
Créer une instance de type FileAttribute qui encapsule l'ensemble des permissions fournies en paramètre
static Set
Renvoyer un ensemble de permissions à partir d'une chaîne de caractères au format rwxrwxrwx
static String toString(Set
Renvoyer une représentation de l'ensemble des permissions sous la forme d'une chaîne de caractères au format rwxrwxrwx
La méthode toString() de la classe PosixFilePermissions renvoie une chaîne de caractères qui représente les permissions.
Exemple ( code Java 7 ) :
1.
Path fichier = Paths.get("/home/jm/test.txt");
2.
PosixFileAttributes attrs = Files.readAttributes(fichier, PosixFileAttributes.class);
3.
Set
4.
System.out.println(PosixFilePermissions.toString(permissions));
Inversement, la méthode fromString() permet de renvoyer une collection de permissions à partir de leur représentation sous la forme d'une chaîne de caractères.
Exemple ( code Java 7 ) :
1.
Path fichier = Paths.get("/home/jm/test.txt");
2.
Set
3.
FileAttribute
4.
PosixFilePermissions.asFileAttribute(perms);
5.
Files.createFile(fichier, attr);
La méthode setPosixFilePermission(Path, Set
Exemple ( code Java 7 ) :
1.
Set
2.
Files.setPosixFilePermissions(fichier, permissions);
21.11. La gestion des unités de stockages
Les fichiers et les répertoires contenus dans un système de fichiers sont stockés dans un périphérique de stockage. Ces systèmes de stockage peuvent être des unités physiques sous la forme de disques (disque dur, SSD, ...) ou des unités logiques (partitions sur un disque, ...).
La classe java.nio.file.FileStore encapsule un système de stockage.
Le point d'entrée d'un système de stockage est dépendant du système d'exploitation :
Sous Windows : c'est un volume désigné par une lettre suivi du caractère « : », les lettres A et B sont réservés aux lecteurs de disquettes, la lettre C est la partition de boot, les autres lettres sont attribuées aux autres partitions, disques ou systèmes de stockage externes
Sous Unix : c'est un point de montage qui correspond à un répertoire dans le système de fichiers
Pour obtenir une instance de la classe FileStore qui encapsule le système de stockage, il faut utiliser la méthode getFileStore() de la classe Files en lui passant en paramètres une instance de type Path qui encapsule un élément du système de fichiers correspondant au système de stockage.
La méthode getFileStores() de la classe FileSystem permet d'obtenir une instance de type Iterable
Exemple ( code Java 7 ) :
1.
Iterable
2.
for (FileStore fileStore : fileStores) {
3.
System.out.println(fileStore);
4.
System.out.println("name : "+ fileStore.name() + ", type : "
5.
+ fileStore.type());
6.
}
La méthode supportsFileAttributView() permet de vérifier si une vue relative aux méta-données est supportée ou non par le FileStore.
Exemple ( code Java 7 ) :
01.
for (FileStore store : FileSystems.getDefault().getFileStores()) {
02.
System.out.println(store);
03.
System.out.println("Support BasicFileAttribute : "
04.
+ store.supportsFileAttributeView(BasicFileAttributeView.class));
05.
System.out.println("Support DosFileAttribute : "
06.
+ store.supportsFileAttributeView(DosFileAttributeView.class));
07.
System.out.println("Support PosixFileAttribute : "
08.
+ store.supportsFileAttributeView(PosixFileAttributeView.class));
09.
}
La classe FileStore possède aussi plusieurs méthodes pour obtenir des informations concernant la taille du système de stockage :
sur l'espace totale avec la méthode getTotalSpace()
sur l'espace disponible avec la méthode getUsableSpace()
sur l'espace non alloué avec la méthode getUnallocatedSpace().
Exemple ( code Java 7 ) :
01.
final int UN_GIGA = 1024 * 1024 * 1024;
02.
for (FileStore store : FileSystems.getDefault().getFileStores()) {
03.
try {
04.
long total = store.getTotalSpace() / UN_GIGA;
05.
long used = (store.getTotalSpace() - store.getUnallocatedSpace()) / UN_GIGA;
06.
long avail = store.getUsableSpace() / UN_GIGA;
07.
System.out.format("%-20s total=%5dGo used=%5dGo avail=%5dGo%n", store,
08.
total, used, avail);
09.
} catch (IOException e) {
10.
e.printStackTrace();
11.
}
12.
}
21.12. Les notifications de changements dans un répertoire
Avant Java 7, pour obtenir des notifications lorsque les éléments d'un répertoire sont modifiés, il était nécessaire de développer son propre mécanisme de polling ou d'utiliser une bibliothèque comme JPathWatch ou JNotify.
Un polling sur le contenu du répertoire permet de savoir si une modification est intervenue dans les fichiers d'un répertoire : ceci consiste à rechercher des modifications de façon périodique en vérifiant le statut de tous les fichiers du répertoire par rapport à leur précédent état.
Java 7 propose l'API WatchService qui offre cette fonctionnalité en standard : NIO2 propose la classe WatchService qui permet d'obtenir des événements sur des actions réalisées sur un répertoire surveillé du système de fichiers. L'API WatchService est performante mais elle n'est pas récursive.
L'utilisation de l'API WatchService pour obtenir des notifications requiert la mise en oeuvre de plusieurs étapes :
créer une instance de type WatchService
enregistrer cette instance auprès du répertoire concerné en précisant le type de notifications auquel on souhaite s'abonner (création, modification, suppression). Un objet de type WatchKey est obtenu suite à cet enregistrement
utiliser une boucle pour obtenir les événements encapsulés dans un objet de type WatchKey
utiliser l'objet de type WatchKey : il faut parcourir et traiter les événements qu'il contient
chaque objet de type WatchKey doit être réinitialisé
une fois que l'objet WatchService n'est plus utile, il est préférable d'invoquer sa méthode close() pour libérer les ressources natives utilisées
21.12.1. La surveillance d'un répertoire
L'implémentation de la classe WatchService s'appuie généralement sur le mécanisme d'événements sous-jacent du système d'exploitation (ChangeNotification sous Windows, inotify sous Linux, FSEvents sous Mac OS X). Si un tel mécanisme n'existe pas alors l'implémentation va utiliser un mécanisme de polling. Dans tous les cas, cette implémentation est spécifique à chaque JVM et système d'exploitation.
Pour obtenir une instance de type WatchService, il faut invoquer la méthode newWatchService() de la classe FileSystem.
Exemple ( code Java 7 ) :
1.
WatchService watchService = FileSystems.getDefault().newWatchService();
Un objet de type WatchService peut s'utiliser sur un objet qui implémente l'interface Watchable. L'interface Path hérite de l'interface Watchable. L'interface Watchable définit deux surcharges de la méthode register() qui attendent en paramètre une instance de type WatchService et les types d'événements qui doivent être capturés.
Il faut donc créer une instance de type Path qui encapsule le chemin du répertoire que l'on souhaite surveiller. La surveillance d'un répertoire se fait en enregistrant l'objet de type WatchService auprès de l'objet de type Path qui encapsule le chemin du répertoire.
Exemple ( code Java 7 ) :
1.
final Path dir = Paths.get("c:/java/test");
2.
3.
WatchKey key = dir.register(watcher,
4.
StandardWatchEventKinds.ENTRY_CREATE,
5.
StandardWatchEventKinds.ENTRY_DELETE,
6.
StandardWatchEventKinds.ENTRY_MODIFY);
La méthode register() attend en paramètre un objet de type WatchService et un ensemble de varargs de type WatchEvent.Kind qui permet de préciser les types d'événements à revecoir. La méthode register() attend donc en paramètre l'instance de type WatchService et accepte plusieurs types événements définis dans la classe java.nio.file.StandardWatchEventKinds.
Les types d'événements concernant les modifications dans un répertoire sont définis dans la classe StandardWatchEventKinds sous la forme de champs statiques de type WatchEvent.Kind
WatchEvent.Kind
un nouvel élément est créé ou renommé dans le répertoire
WatchEvent.Kind
un élément du répertoire est modifié
WatchEvent.Kind
un élément du répertoire est supprimé ou renommé. Les modifications/suppressions du répertoire lui-même ne sont pas concernées
WatchEvent.Kind
Aucun commentaire:
Enregistrer un commentaire