web-dev-qa-db-fra.com

Comment vérifier si un fichier est ouvert par un autre processus (Java/Linux)?

J'essaie de vérifier si un certain Java.io.File est ouvert par un programme externe. Sur Windows, j'utilise cette astuce simple:

try {
    FileOutputStream fos = new FileOutputStream(file);
    // -> file was closed
} catch(IOException e) {
    // -> file still open
}

Je sais que les systèmes Unix permettent d'ouvrir des fichiers dans plusieurs processus ... Existe-t-il une astuce similaire pour obtenir le même résultat sur des systèmes Unix?

Toute aide/piratage hautement apprécié :-)

26
salocinx

Voici un exemple d'utilisation de lsof pour les systèmes Unix:

public static boolean isFileClosed(File file) {
    try {
        Process plsof = new ProcessBuilder(new String[]{"lsof", "|", "grep", file.getAbsolutePath()}).start();
        BufferedReader reader = new BufferedReader(new InputStreamReader(plsof.getInputStream()));
        String line;
        while((line=reader.readLine())!=null) {
            if(line.contains(file.getAbsolutePath())) {                            
                reader.close();
                plsof.destroy();
                return false;
            }
        }
    } catch(Exception ex) {
        // TODO: handle exception ...
    }
    reader.close();
    plsof.destroy();
    return true;
}

J'espère que cela t'aides.

9

Vous pouvez exécuter à partir du programme Java l'utilitaire Unix lsof qui vous indique quel processus utilise un fichier, puis d'analyser sa sortie. Pour exécuter un programme à partir de code Java, utilisez, par exemple, les classes Runtime, Process, ProcessBuilder. Remarque: dans ce cas, votre programme Java ne sera pas portable, ce qui est contraire au concept de portabilité. Réfléchissez bien avant si vous en avez vraiment besoin :)

3
Alexey Berezkin

Celui-ci devrait également fonctionner pour les systèmes Windows. Mais attention, ne fonctionne pas pour Linux!

     private boolean isFileClosed(File file) {  
            boolean closed;
            Channel channel = null;
            try {
                channel = new RandomAccessFile(file, "rw").getChannel();
                closed = true;
            } catch(Exception ex) {
                closed = false;
            } finally {
                if(channel!=null) {
                    try {
                        channel.close();
                    } catch (IOException ex) {
                        // exception handling
                    }
                }
            }
            return closed;
    }
3
salocinx

Vous pouvez essayer ce code de type sémaphore pour un verrouillage de fichier avec @ZZ Coder

File file = new File(fileName);
FileChannel channel = new RandomAccessFile(file, "rw").getChannel();

FileLock lock = channel.lock();
try {
    lock = channel.tryLock();
    // Ok. You get the lock
} catch (OverlappingFileLockException e) {
    // File is open by someone else
  } finally {
   lock.release();
}
1
Aishwarya Shiva

Merci pour la suggestion originale. J'ai une petite mise à jour assez importante pour cette méthode:

FileOutputStream fos = null;
try {
    // Make sure that the output stream is in Append mode. Otherwise you will
    // truncate your file, which probably isn't what you want to do :-) 
    fos = new FileOutputStream(file, true);
    // -> file was closed
} catch(IOException e) {
    // -> file still open
} finally {
    if(fos != null) {
    try {
        fos.close();
    } catch (IOException e) {
        e.printStackTrace();
    }
}

À votre santé, Gumbatron

1
Gumbatron