我已经建立一个简单的测试,创建和删除文件(名称不变)在一个无限循环。 该测试并运行了几秒钟(有时超过77000次迭代!),然后失败,出现此异常:
Exception in thread "main" java.io.IOException: Access is denied
at java.io.WinNTFileSystem.createFileExclusively(Native Method)
at java.io.File.createNewFile(Unknown Source)
at DeleteTest.main(DeleteTest.java:11)
下面是测试逻辑:
final File f = new File(pathname);
while (true) {
final boolean create = f.createNewFile();
if (!create) {
System.out.println("crate failed");
} else {
final boolean delete = f.delete();
if (!delete) {
System.out.println("delete failed");
}
}
}
这怎么可能? 删除通话不会失败。 它会告诉。 所以删除总是成功,但createNewFile
失败。 这是MSDN说,关于Win32 API函数DeleteFile
:
该功能的DeleteFile标志着在近删除的文件。 因此,直到最后句柄文件被关闭,不会发生文件删除。 后续调用CreateFile打开文件失败并ERROR_ACCESS_DENIED。
所以createNewFile
不关闭文件? OpenJDK的来源告诉我们,文件被关闭:
JNIEXPORT jboolean JNICALL
Java_java_io_Win32FileSystem_createFileExclusively(JNIEnv *env, jclass cls,
jstring pathname)
{
jboolean rv = JNI_FALSE;
DWORD a;
WITH_PLATFORM_STRING(env, pathname, path) {
int orv;
int error;
JVM_NativePath((char *)path);
orv = JVM_Open(path, JVM_O_RDWR | JVM_O_CREAT | JVM_O_EXCL, 0666);
if (orv < 0) {
if (orv != JVM_EEXIST) {
error = GetLastError();
// If a directory by the named path already exists,
// return false (behavior of solaris and linux) instead of
// throwing an exception
a = GetFileAttributes(path);
if ((a == INVALID_FILE_ATTRIBUTES) ||
!(a & FILE_ATTRIBUTE_DIRECTORY)) {
SetLastError(error);
JNU_ThrowIOExceptionWithLastError(env, path);
}
}
} else {
JVM_Close(orv);
rv = JNI_TRUE;
}
} END_PLATFORM_STRING(env, path);
return rv;
}
任何人都可以解释这种现象?