异步方法中的Spring Boot while循环暂停应用程序

时间:2018-08-20 13:32:38

标签: java spring-boot

我正在尝试创建一个监视程序,以查找对特定文件夹的更改。我已经创建了一个监视程序,并将其放在Async方法中,但是当我从服务调用它时,由于监视程序方法中的while循环,应用程序暂停了。就像该方法不在新线程中执行一样。

这是包含我要执行的方法的类;

    @Service
public class FileWatcher {

    @Async
    public Future<Object> watch(Path path, DataParser parser) throws IOException, InterruptedException {
        WatchService watchService = FileSystems.getDefault().newWatchService();

        path.register(
                watchService,
                StandardWatchEventKinds.ENTRY_CREATE,
                StandardWatchEventKinds.ENTRY_DELETE,
                StandardWatchEventKinds.ENTRY_MODIFY);

        WatchKey key;
        while ((key = watchService.take()) != null) {
            for (WatchEvent<?> event : key.pollEvents()) {
                File file = new File(path.toString() + "/" + event.context());

                if (event.kind() == StandardWatchEventKinds.ENTRY_MODIFY) {
                    parser.fileChanged(file);
                }

                if (event.kind() == StandardWatchEventKinds.ENTRY_CREATE) {
                    parser.fileCreated(file);
                }

                if (event.kind() == StandardWatchEventKinds.ENTRY_DELETE) {
                    parser.fileRemoved(file);
                }
            }
            key.reset();
        }

        return null;
    }

}

然后,我在服务的构造函数中调用它。

@Service
public class SPIService {

    private final String DATAFOLDER = "/spi";

    private Path dataPath;

    public SPIService(@Value("${com.zf.trw.visualisation.data.path}") String dataPath) {
        this.dataPath = Paths.get(dataPath + DATAFOLDER);

        FileWatcher fileWatcher = new FileWatcher();
        try {
            fileWatcher.watch(this.dataPath, new SPIParser());
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

为什么这不起作用?是因为我从服务的构造函数中调用方法吗?

1 个答案:

答案 0 :(得分:3)

您正在使用new FileWatcher(),这意味着该实例不是托管bean。这也意味着@Async被忽略(这说明您的应用程序停止了)。您需要改为@Autowire FileWatcher

还请注意,您的解决方案对我来说似乎非常可疑,不仅是因为存在无限循环,而且是因为@Async方法中存在一个无限循环(这有一些重要的后果)。我至少会使用一个单线程线程池。