需求:数据量太大,第一个用户查数据库后写入一个文件内,后面的用户去读文件。
问题:本地测试完全没问题,发布生产写入环节报java.nio.channels.AsynchronousException错误,导致后面用户读文件数据为空。
代码写文件使用的是异步文件通道:AsynchronousFileChannel
Java NIO中的FileChannel是一个连接到文件的通道。可以通过文件通道读写文件。FileChannel无法设置为非阻塞模式,他总是运行在阻塞模式下。在Java 7中,AsynchronousFileChannel被添加到Java NIO。AsynchronousFileChannel使读取数据,并异步地将数据写入文件成为可能。
所以AsynchronousFileChannel是异步非阻塞的!
最开始是怀疑生产环境Linux上创建的文件权限不够导致写入失败,后来发现不是这个问题。
仔细看代码后发现,由于是异步的:当管道建立后,没来得及写入文件,管道就被关闭了。
解决方法:
1.Future f = channel.write(buffer,0)后使用f.get()将方法阻塞住,保证运行完写文件的代码再把通道关闭,或者直接让Thread.sleep(xx);
2.异步方法有个回调函数,重写回调函数的complete方法,完成方法里加channel.close(),保证方法写入结束后再关闭通道
至于本地没报错,而生产环境报错,可能原因由于网络延迟较小、资源较为充足,可能会导致异步操作的执行速度较快,从而在正常情况下能够成功完成操作,不会触发错误或异常。
但是在生产环境中,由于网络延迟、负载或资源限制等因素,异步操作的执行速度可能会变慢,甚至可能超过了一定的时间限制。在这种情况下,如果代码没有适当地处理超时或错误情况,就可能导致错误的发生。