Bunun olabileceği başka bir köşe durumu: bir JAR dosyasını bir JAR dosyasını okur / yazarsanız URL
ve daha sonra aynı dosyayı aynı JVM oturumu içinde silmeye çalışırsanız.
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();
try (InputStream i = c.getInputStream()) {
byte[] first16 = new byte[16];
i.read(first16);
System.out.println(new String(first16));
}
System.out.println(f.delete());
Bunun nedeni, Java'nın dahili JAR dosyası işleme mantığının JarFile
girişleri önbelleğe alma eğiliminde olmasıdır :
class JarURLInputStream extends FilterInputStream {
JarURLInputStream(InputStream var2) {
super(var2);
}
public void close() throws IOException {
try {
super.close();
} finally {
if (!JarURLConnection.this.getUseCaches()) {
JarURLConnection.this.jarFile.close();
}
}
}
}
Ve her biri JarFile
(daha ziyade, temel ZipFile
yapı), yapım close()
anından çağrılana kadar dosyanın bir tanıtıcısını tutacaktır :
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;
Bu NetBeans sorunu hakkında iyi bir açıklama var .
Görünüşe göre bunu "düzeltmenin" iki yolu var:
JAR dosyası önbelleğe almayı devre dışı bırakabilirsiniz - mevcut JVM oturumundaki mevcut URLConnection
veya gelecekteki tüm e-postalar için URLConnection
(global olarak):
URL u = new URL("jar:" + j + "!/META-INF/MANIFEST.MF");
URLConnection c = u.openConnection();
c.setUseCaches(false);
c.setDefaultUseCaches(false);
[HACK WARNING!] İşinizJarFile
bittiğinde önbellekten manuel olarak temizleyebilirsiniz . Önbellek yöneticisi sun.net.www.protocol.jar.JarFileFactory
pakete özeldir, ancak biraz yansıma sihri işi sizin için halledebilir:
class JarBridge {
static void closeJar(URL url) throws Exception {
Class<?> jarFactoryClazz = Class.forName("sun.net.www.protocol.jar.JarFileFactory");
Method getInstance = jarFactoryClazz.getMethod("getInstance");
getInstance.setAccessible(true);
Object jarFactory = getInstance.invoke(jarFactoryClazz);
Method get = jarFactoryClazz.getMethod("get", URL.class);
get.setAccessible(true);
Object jarFile = get.invoke(jarFactory, url);
Method close = jarFactoryClazz.getMethod("close", JarFile.class);
close.setAccessible(true);
close.invoke(jarFactory, jarFile);
((JarFile) jarFile).close();
}
}
JarBridge.closeJar(j);
System.out.println(f.delete());
Lütfen dikkat: Tüm bunlar Java 8 kod tabanına ( 1.8.0_144
) dayanmaktadır ; diğer / sonraki sürümlerle çalışmayabilirler.