在 Symfony 中动态生成文件的最佳实践

2020-05-30  本文已影响0人  forks1990

所谓最佳,当然是我自己的理解,算不得数的,也想不出更确切的定语。如果改为“一种实践”不是我想要的,这不是一种选项,只用能用上,我是要复用这套实践,对个人来说没有更好的定语了。既然与我是最佳,也定会有人认同,或者阅读全文后认同,或者在若干年后认同。所以也算得上最佳吧,只是在最佳上再加上个定语更确切些:我和所有认同他是最佳实践的人们的最佳实践。循环定义了,但人对事物的理解何尝不是在循环中逐渐明晰,循环就循环了吧。

定义

动态生产文件是指即时生成pdf,excel,word,png,jpg等等,内容是实时性的,下次下载时需要重新生成。

最佳实践

  1. 动态生成的结果保存在临时文件中
  2. 使用BinaryFileResponse,将临时文件返回(下载)给用户
  3. 设置deleteFileAfterSend(),用户下载后临时文件被自动删除
class ReportController extends AbstractController
{
    public function __invoke(ReportEngine $reportEngine): Response
    {
        $pdfFile = $reportEngine->generate();
        return $this
            ->file($pdfFile, $reportEngine->buildFilename(), ResponseHeaderBag::DISPOSITION_INLINE)
            ->deleteFileAfterSend();
    }
}

说明:

  1. 临时文件和用户下载的文件名无关,临时文件是随机和服务器端唯一的
  2. deleteFileAfterSend()不是100%可靠,额外的机制比如 cron 定时任务还是有必要
# 每日凌晨2点删除一天前生成的 /tmp/export-report-* 文件
0 2 * * * find /tmp/ -mtime 1 -name export-report-* -delete

分析

最佳实践使用临时文件做中转,没有使用显而易见的完全使用内存的方法。

优缺点

优点:

  1. 适应性广,几乎100%的文件生成库都会支持保存文件
  2. 文件生成库使用 Stream 等方式将内存使用降到最低
  3. BinaryFileRespose 使用 stream_copy_to_stream() 将文件流直接下载给客户端。
  4. 生成结果不占用内存,php5.3 以后支持内存回收,需要时可以及时释放,不需要等待下载完成
  5. 简单,顺着这条路现有的东西可直接整合

缺点:

效率问题,使用文件缓存通常会被认为效率底下。如果在现实中用能接受的效率换到上述的优点,岂不是最佳实践。真实的效率和假想的效率是两回事。

上一篇 下一篇

猜你喜欢

热点阅读