本人在使用jfinal作一个项目的时候,须要下载一些临时文件,就是有程序产生,而且被用户下载后就失效的小文件,在使用jfinal的renderFile(file);
后,调用 file.delete();发现,文件没法下载,因而找问题java
查看jfinal controller的renderFile源码后发现,调用render系列方法后,并无立刻执行render操做,代码以下:app
/** * Render with file */ public void renderFile(String fileName) { render = renderFactory.getFileRender(fileName); } /** * Render with file */ public void renderFile(File file) { render = renderFactory.getFileRender(file); }
因而可知,renderFile只不过是产生了一个render对象而已,这样就能解释了,由于调用renderFile后咱们把文件删除了,因而就没法下载了,因而继续找,而后找到调用这个方法的地方,发现是在一个叫作ActionHandler
的了类里面,找到这个类的handler方法,关键部位代码以下:ide
Controller controller = action.getControllerClass().newInstance(); controller.init(request, response, urlPara[0]); if (devMode) { boolean isMultipartRequest = ActionReporter.reportCommonRequest(controller, action); new ActionInvocation(action, controller).invoke(); if (isMultipartRequest) ActionReporter.reportMultipartRequest(controller, action); } else { new ActionInvocation(action, controller).invoke(); } Render render = controller.getRender(); if (render instanceof ActionRender) { String actionUrl = ((ActionRender)render).getActionUrl(); if (target.equals(actionUrl)) throw new RuntimeException("The forward action url is the same as before."); else handle(actionUrl, request, response, isHandled); return ; } if (render == null) render = renderFactory.getDefaultRender(action.getViewPath() + action.getMethodName()); render.setContext(request, response, action.getViewPath()).render();
咱们发如今 new ActionInvocation(action, controller).invoke();
后才去执行的render方法,render.setContext(request, response, action.getViewPath()).render();
this
问题已经找到了,因而咱们须要解决,刚开始问群里的时候,你们都是建议使用定时删除,队列的方式来执行,我我的不喜欢这种绕圈子的作法,因而有了方案1url
本身写代码去renderFile,这样就能够避免这个问题了,代码大体以下(基本上是从FileRender
类中拷贝而来):设计
String contentType = getRequest().getServletContext().getMimeType(file.getName()); if (contentType == null) { contentType = "application/octet-stream"; // "application/octet-stream"; } System.out.println(contentType); getResponse().setContentType(contentType); getResponse().setContentLength((int) file.length()); OutputStream out = getResponse().getOutputStream(); FileInputStream in = new FileInputStream(file); try { byte[] buf= new byte[1024]; for(int n;(n = in.read(buf)) != -1;) { out.write(buf,0,n); } } finally { try { in.close(); out.close(); } finally { file.delete(); file.deleteOnExit(); } }
这种代码解决了问题,可是仍是不够好,我相信大部分人也不喜欢这种方式,虽然能够封装一个方法去作这个事情.后来用了一种更好的方式来解决这个问题,并且比较符合jfinal的设计哲学code
自定义render对象
因为jfinal已经有了FileRender,咱们能够直接继承它,咱们重写 render方法就行了,这种方法作到了代码重用,我的很喜欢,代码以下:继承
public class TempFileRender extends FileRender { private String fileName; private File file; public TempFileRender(String fileName) { super(fileName); this.fileName = fileName; } public TempFileRender(File file) { super(file); this.file = file; } @Override public void render() { try { super.render(); } finally { if(null != fileName) { file = new File(fileName); } if(null != file) { file.delete(); file.deleteOnExit(); } } } }
这样在controller中使用就只须要 render(new TempFileRender(file|fileName))
了队列
打完收工