อีกกรณีหนึ่งที่อาจเกิดขึ้นได้: ถ้าคุณอ่าน / เขียนไฟล์ JAR ผ่านURL
และในภายหลังให้ลองลบไฟล์เดียวกันภายในเซสชัน JVM เดียวกัน
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());
เหตุผลคือตรรกะการจัดการไฟล์ JAR ภายในของ Java มีแนวโน้มที่จะแคชJarFile
รายการ:
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();
}
}
}
}
และแต่ละอันJarFile
(แต่เป็นZipFile
โครงสร้างพื้นฐาน) จะถือแฮนเดิลของไฟล์ตั้งแต่เวลาสร้างจนถึงclose()
เรียกใช้:
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;
มีคำอธิบายที่ดีในการเป็นปัญหา NetBeans นี้
เห็นได้ชัดว่ามีสองวิธีในการ "แก้ไข" สิ่งนี้:
คุณสามารถปิดใช้งานการแคชไฟล์ JAR - สำหรับปัจจุบันURLConnection
หรือสำหรับอนาคตทั้งหมดURLConnection
(ทั่วโลก) ในเซสชัน JVM ปัจจุบัน:
URL u = new URL("jar:" + j + "!/META-INF/MANIFEST.MF");
URLConnection c = u.openConnection();
c.setUseCaches(false);
c.setDefaultUseCaches(false);
[คำเตือนแฮ็ก!]คุณสามารถล้างJarFile
แคชออกจากแคชด้วยตนเองเมื่อดำเนินการเสร็จสิ้น ตัวจัดการแคชsun.net.www.protocol.jar.JarFileFactory
เป็นแพ็กเกจส่วนตัว แต่เวทมนตร์สะท้อนบางอย่างสามารถทำให้งานสำเร็จสำหรับคุณ:
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());
โปรดทราบ:ทั้งหมดนี้ใช้ Java 8 codebase ( 1.8.0_144
); อาจใช้ไม่ได้กับเวอร์ชันอื่น / ที่ใหม่กว่า