Files.createDirectories () создает исключение FileAlreadyExistsException, но не каталог - PullRequest
0 голосов
/ 05 ноября 2018

Этот вопрос задает аналогичный вопрос. Однако в моем случае каталог не существует ни до, ни после вызова Files.createDirectories(). Это происходит в Oracle JDK 10.0.2.

Вот мой код ...

Set<PosixFilePermission> perms;
FileAttribute<?> attr;
Path path;
File directory;

directory = new File("/test/http/localhost_4452/UCF2b/Live");
path      = directory.toPath();
perms     = EnumSet.noneOf(PosixFilePermission.class);

perms.add(PosixFilePermission.OWNER_READ);
perms.add(PosixFilePermission.OWNER_WRITE);
perms.add(PosixFilePermission.OWNER_EXECUTE);

attr = PosixFilePermissions.asFileAttribute(perms);

try
{
   if (!directory.exists())
      Files.createDirectories(path, attr);
}
catch (IOException e)
{
   if (!directory.exists())
   {
      ... collect more information about the state of the directory and its parent path ...
      ... add this information as a suppressed exception ...
      throw e;
   }
   // else do nothing and assume another thread created the directory
}

Вот исключение ...

java.nio.file.FileAlreadyExistsException: /test/http/localhost_4452/UCF2b/Live
at java.base/sun.nio.fs.UnixException.translateToIOException(UnixException.java:94)
at java.base/sun.nio.fs.UnixException.rethrowAsIOException(UnixException.java:111)
at java.base/sun.nio.fs.UnixException.rethrowAsIOException(UnixException.java:116)
at java.base/sun.nio.fs.UnixFileSystemProvider.createDirectory(UnixFileSystemProvider.java:385)
at java.base/java.nio.file.Files.createDirectory(Files.java:682)
at java.base/java.nio.file.Files.createAndCheckIsDirectory(Files.java:789)
at java.base/java.nio.file.Files.createDirectories(Files.java:735)
at ...my code...

Текущий пользователь - root. Здесь собрана диагностическая информация о каталоге и его родительских каталогах. Эта информация собирается в блоке catch, если каталог не существует.

+--------------------------------------+--------+----------+--------------------------+---------+-----------+-------+--------+---------+-------+-------+
|            Path                      | Exists |  Length  |        Modified          |  Owner  | Directory | File  | Hidden | Execute | Read  | Write |
+--------------------------------------+--------+----------+--------------------------+---------+-----------+-------+--------+---------+-------+-------+
| /test/http/localhost_4452/UCF2b/Live | false  |   0.00 B | 1970-01-01T00:00:00Z     | (null)  | false     | false | false  | false   | false | false |
| /test/http/localhost_4452/UCF2b      | true   |  4.00 kB | 2018-11-04T20:32:09.769Z | root    | true      | false | false  | true    | true  | true  |
| /test/http/localhost_4452            | true   |  4.00 kB | 2018-11-04T20:18:26.849Z | root    | true      | false | false  | true    | true  | true  |
| /test/http                           | true   |  4.00 kB | 2018-11-04T20:11:42.605Z | root    | true      | false | false  | true    | true  | true  |
| /test/                               | true   | 20.00 kB | 2018-11-04T20:32:09.768Z | root    | true      | false | false  | true    | true  | true  |
| /                                    | true   |  4.00 kB | 2018-11-04T20:09:22.061Z | root    | true      | false | false  | true    | true  | true  |
+--------------------------------------+--------+----------+--------------------------+---------+-----------+-------+--------+---------+-------+-------+

Как видите, код проверяет, существует ли каталог перед вызовом Files.createDirectories(), и проверяет, что каталог не существует после вызова. Исключение случается редко. Что я не понимаю? Как мне создать каталог? Если я просто повторю вызов на Files.createDirectories(), он продолжит сбой.

Редактировать: этот код вызывается несколькими потоками. Это означает, что несколько потоков могут вызывать Files.createDirectories(), но код не перебрасывает исключение, если каталог оказывается существующим. Другими словами, другой поток (ы) должен будет создать и удалить каталог в нужный момент, поскольку directory.exists() равно false до и после Files.createDirectories(). Кроме того, это идеальное время должно сохраняться, поскольку, как только программа решает эту проблему, она продолжает происходить для определенного каталога.

Ответы [ 2 ]

0 голосов
/ 06 ноября 2018

Я не могу воспроизвести проблему очень часто. Я решил заменить Files.createDirectories() на Files.createDirectory() (т.е. создать каждый каталог в пути по мере необходимости). Может быть, это будет работать. Возможно, это поможет пролить свет на основную проблему.

Редактировать: Прошло 1 неделя с момента попытки выше и около 1000 выполнений модульных тестов. Проблема не возникла снова. Я собираюсь предположить, что приведенный выше ответ работает.

0 голосов
/ 05 ноября 2018

Скорее всего, этот код используется несколькими потоками, потому что только это может объяснить, почему код входит в блок if, если условие false, а также что это происходит редко. Поток проверяет, что каталог не существует, и прежде чем он сможет его создать, другой поток создает его. Вы должны использовать синхронизацию, если это так.

synchronized (this) {
    if (!directory.exists())
        Files.createDirectories(path, attr);
}
...