Ein weiterer Eckfall, in dem dies passieren könnte: Wenn Sie eine JAR-Datei über eine URL
und später lesen / schreiben, versuchen Sie, dieselbe Datei innerhalb derselben JVM-Sitzung zu löschen.
File f = new File("/tmp/foo.jar");
URL j = f.toURI().toURL();
URL u = new URL("jar:" + j + "!/META-INF/MANIFEST.MF");
URLConnection c = u.openConnection();
// open a Jar entry in auto-closing manner
try (InputStream i = c.getInputStream()) {
// just read some stuff; for demonstration purposes only
byte[] first16 = new byte[16];
i.read(first16);
System.out.println(new String(first16));
}
// ...
// i is now closed, so we should be good to delete the jar; but...
System.out.println(f.delete()); // says false!
Grund dafür ist, dass die interne JAR-Dateiverarbeitungslogik von Java dazu neigt, JarFile
Einträge zwischenzuspeichern:
// inner class of `JarURLConnection` that wraps the actual stream returned by `getInputStream()`
class JarURLInputStream extends FilterInputStream {
JarURLInputStream(InputStream var2) {
super(var2);
}
public void close() throws IOException {
try {
super.close();
} finally {
// if `getUseCaches()` is set, `jarFile` won't get closed!
if (!JarURLConnection.this.getUseCaches()) {
JarURLConnection.this.jarFile.close();
}
}
}
}
Und jede JarFile
(vielmehr die zugrunde liegende ZipFile
Struktur) würde vom Zeitpunkt der Erstellung bis zum close()
Aufruf ein Handle für die Datei enthalten :
public ZipFile(File file, int mode, Charset charset) throws IOException {
// ...
jzfile = open(name, mode, file.lastModified(), usemmap);
// ...
}
// ...
private static native long open(String name, int mode, long lastModified,
boolean usemmap) throws IOException;
Zu diesem NetBeans-Problem gibt es eine gute Erklärung .
Anscheinend gibt es zwei Möglichkeiten, dies zu "beheben":
Sie können das Zwischenspeichern von JAR-Dateien deaktivieren - für die aktuelle URLConnection
oder für alle zukünftigen URLConnection
s (global) in der aktuellen JVM-Sitzung:
URL u = new URL("jar:" + j + "!/META-INF/MANIFEST.MF");
URLConnection c = u.openConnection();
// for only c
c.setUseCaches(false);
// globally; for some reason this method is not static,
// so we still need to access it through a URLConnection instance :(
c.setDefaultUseCaches(false);
[HACK WARNING!] Sie können das manuell JarFile
aus dem Cache löschen, wenn Sie damit fertig sind. Der Cache-Manager sun.net.www.protocol.jar.JarFileFactory
ist paketprivat, aber etwas Reflexionszauber kann die Arbeit für Sie erledigen:
class JarBridge {
static void closeJar(URL url) throws Exception {
// JarFileFactory jarFactory = JarFileFactory.getInstance();
Class<?> jarFactoryClazz = Class.forName("sun.net.www.protocol.jar.JarFileFactory");
Method getInstance = jarFactoryClazz.getMethod("getInstance");
getInstance.setAccessible(true);
Object jarFactory = getInstance.invoke(jarFactoryClazz);
// JarFile jarFile = jarFactory.get(url);
Method get = jarFactoryClazz.getMethod("get", URL.class);
get.setAccessible(true);
Object jarFile = get.invoke(jarFactory, url);
// jarFactory.close(jarFile);
Method close = jarFactoryClazz.getMethod("close", JarFile.class);
close.setAccessible(true);
//noinspection JavaReflectionInvocation
close.invoke(jarFactory, jarFile);
// jarFile.close();
((JarFile) jarFile).close();
}
}
// and in your code:
// i is now closed, so we should be good to delete the jar
JarBridge.closeJar(j);
System.out.println(f.delete()); // says true, phew.
Bitte beachten Sie: All dies basiert auf Java 8 codebase ( 1.8.0_144
); Sie funktionieren möglicherweise nicht mit anderen / späteren Versionen.