在项目实际应用中,由于下载文件内容都比较大,如果同时有很多用户同时在下载,JVM的内存就会升的很高,甚至崩溃。为了避免很多用户同时下载,特引入Semaphore控制一次最多有配置个线程能进入实际下载的代码,即而控制JVM内存不会升的很高而导致崩溃。
import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; import java.io.UnsupportedEncodingException; import java.net.URLEncoder; import java.util.List; import java.util.Map; import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Future; import java.util.concurrent.Semaphore; import javax.servlet.ServletOutputStream; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.apache.commons.io.IOUtils; import org.springframework.beans.factory.InitializingBean; import org.springframework.stereotype.Component; @Component("DownloadView.csv") public class DownloadView extends AbstractCsvView implements InitializingBean { //允许的最大线程数 private String threadNum=PropertyUtil.getProperty("threadNum"); // 线程池 private ExecutorService exec = null; // 只能threadNum个线程同时访问 private Semaphore semp = new Semaphore(Integer.parseInt(threadNum), true); @Override protected void buildExcelDocument(Map<String, Object> model, List<String> csvList, HttpServletRequest request, HttpServletResponse response) throws Exception { String fileName =""; String url = request.getParameter("outputInfo"); if(StringUtil.isNotEmpty(url)){ fileName= url.substring(url.lastIndexOf("/") + 1, url.length()); } super.setUrl(url); try { response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(fileName, response.getCharacterEncoding())); } catch (UnsupportedEncodingException e) { throw new Exception("不支持此编码格式"); } } @SuppressWarnings("unchecked") @Override protected void renderMergedOutputModel(final Map<String, Object> model, HttpServletRequest request, HttpServletResponse response) throws Exception, IOException, InterruptedException, ExecutionException { exec = Executors.newCachedThreadPool(); response.setContentType(getContentType()); final String url = request.getParameter("outputInfo"); final ServletOutputStream out=response.getOutputStream(); final HttpServletRequest _request = request; final HttpServletResponse _response = response; InputStream in=null; try{ in= new FileInputStream(url); }catch(Exception e){ throw new Exception("找不到对应的文件:"+url); } final InputStream fis=in; final String encode=super.getEncoding(); Callable<Boolean> call = new Callable<Boolean>() { @Override public Boolean call() { try { // 获取许可 semp.acquire(); List<String> csvList = null; //IOUtils.readLines()是一次性读取整个文件 //readline() 和 .readlines()之间的差异是后者一次读取整个文件,像read()一样。 //readlines()自动将文件内容分析成一个行的列表, //readline()每次只读取一行,通常比 readlines()慢得多。 //仅当没有足够内存可以一次读取整个文件时,才应该使用readline(). csvList = IOUtils.readLines(fis); buildExcelDocument(model, csvList, _request, _response); if (encode == null) { IOUtils.writeLines(csvList,encode, out); } else { IOUtils.writeLines(csvList, encode, out, encode); } //Thread.sleep((long) (2000)); return true; } catch (Exception e) { System.out.println(e); return false; } finally { semp.release(); } } }; Future<Boolean> future=null; if(!exec.isShutdown()){ future= exec.submit(call); } exec.shutdown(); if((Boolean) future.get()){ System.out.println("success"); }else{ System.out.print("fail"); } } @Override public void afterPropertiesSet() throws Exception { } }
AbstractCsvView.java
import java.io.IOException; import java.io.InputStream; import java.util.ArrayList; import java.util.List; import java.util.Locale; import java.util.Map; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.apache.commons.io.IOUtils; import org.springframework.core.io.Resource; import org.springframework.core.io.support.LocalizedResourceHelper; import org.springframework.web.servlet.support.RequestContextUtils; import org.springframework.web.servlet.view.AbstractView; public abstract class AbstractCsvView extends AbstractView{ /** The content type for an csv response */ private static final String CONTENT_TYPE = "text/csv"; /** The extension to look for existing templates */ private static final String EXTENSION = ".csv"; private String lineEnding; private String encoding; /** The url at which the template to use is located */ private String url; /** * Default Constructor. * Sets the content type of the view to "text/csv". */ public AbstractCsvView() { setContentType(CONTENT_TYPE); } /** * Set the URL of the Excel workbook source, without localization part nor extension. */ public void setUrl(String url) { this.url = url; } public void setLineEnding(String lineEnding) { this.lineEnding = lineEnding; } public void setEncoding(String encoding) { this.encoding = encoding; } public String getLineEnding() { return lineEnding; } public String getEncoding() { return encoding; } public String getUrl() { return url; } @Override protected boolean generatesDownloadContent() { return true; } @Override protected void renderMergedOutputModel(Map<String, Object> model, HttpServletRequest request, HttpServletResponse response) throws Exception { // Set the content type and get the output stream. response.setContentType(getContentType()); List<String> csvList = null; if (this.url != null) { InputStream in = getTemplateSource(this.url, request); csvList = IOUtils.readLines(in); }else{ csvList = new ArrayList<String>(); } buildExcelDocument(model, csvList, request, response); if(this.encoding == null){ IOUtils.writeLines(csvList, this.lineEnding, response.getOutputStream()); }else{ IOUtils.writeLines(csvList, this.lineEnding, response.getOutputStream(), this.encoding); } } protected InputStream getTemplateSource(String url, HttpServletRequest request) throws IOException { LocalizedResourceHelper helper = new LocalizedResourceHelper(getApplicationContext()); Locale userLocale = RequestContextUtils.getLocale(request); Resource inputFile = helper.findLocalizedResource(url, EXTENSION, userLocale); // Create the Excel document from the source. if (logger.isDebugEnabled()) { logger.debug("Loading Excel workbook from " + inputFile); } return inputFile.getInputStream(); } /** * Subclasses must implement this method to create an csv List * document, given the model. * @param model the model Map * @param csvList * @param request in case we need locale etc. Shouldn't look at attributes. * @param response in case we need to set cookies. Shouldn't write to it. * @throws Exception in case of failure */ protected abstract void buildExcelDocument(Map<String, Object> model, List<String> csvList, HttpServletRequest request, HttpServletResponse response) throws Exception; }
相关推荐
Matlab领域上传的视频均有对应的完整代码,皆可运行,亲测可用,适合小白; 1、代码压缩包内容 主函数:main.m; 调用函数:其他m文件;无需运行 运行结果效果图; 2、代码运行版本 Matlab 2019b;若运行有误,根据提示修改;若不会,私信博主; 3、运行操作步骤 步骤一:将所有文件放到Matlab的当前文件夹中; 步骤二:双击打开main.m文件; 步骤三:点击运行,等程序运行完得到结果; 4、仿真咨询 如需其他服务,可私信博主; 4.1 博客或资源的完整代码提供 4.2 期刊或参考文献复现 4.3 Matlab程序定制 4.4 科研合作
【使用教程】 一、环境配置 1、建议下载anaconda和pycharm 在anaconda中配置好环境,然后直接导入到pycharm中,在pycharm中运行项目 anaconda和pycharm安装及环境配置参考网上博客,有很多博主介绍 2、在anacodna中安装requirements.txt中的软件包 命令为:pip install -r requirements.txt 或者改成清华源后再执行以上命令,这样安装要快一些 软件包都安装成功后才算成功 3、安装好软件包后,把anaconda中对应的python导入到pycharm中即可(不难,参考网上博客) 二、环境配置好后,开始训练(也可以训练自己数据集) 1、数据集准备 需要准备yolo格式的目标检测数据集,如果不清楚yolo数据集格式,或者有其他数据训练需求,请看博主yolo格式各种数据集集合链接:https://blog.csdn.net/DeepLearning_/article/details/127276492 更多详情介绍,见资源内的项目说明
# 基于Spring Boot和Vue的学生作业互评系统 ## 项目简介 本项目是一个基于Spring Boot和Vue框架开发的学生作业互评系统。系统主要功能包括学生作业的提交、教师作业的布置、作业的批改与评分、以及学生之间的作业互评。通过该系统,教师可以方便地管理课程和作业,学生可以在线提交作业并参与互评,从而提高作业质量和学习效果。 ## 项目的主要特性和功能 1. 用户管理 支持学生、教师和管理员三种角色的用户管理。 提供用户注册、登录、密码修改等功能。 2. 课程管理 教师可以创建和管理课程,学生可以选课。 支持课程信息的查看和编辑。 3. 作业管理 教师可以布置作业,设置作业的截止日期和评分标准。 学生可以在线提交作业,查看作业提交状态。 4. 作业批改与评分 教师可以对学生提交的作业进行批改和评分。 学生可以查看自己的作业评分和教师的评语。
PHP学生成绩查询(源代码+论文)
c语言学生信息系统
Android的多媒体框架OpenCore介绍
AutocompleteTest
2023-04-06-项目笔记-第三百一十八阶段-课前小分享_小分享1.坚持提交gitee 小分享2.作业中提交代码 小分享3.写代码注意代码风格 4.3.1变量的使用 4.4变量的作用域与生命周期 4.4.1局部变量的作用域 4.4.2全局变量的作用域 4.4.2.1全局变量的作用域_1 4.4.2.316局变量的作用域_316- 2024-11-15
可以识别视频语音自动生成字幕SRT文件的开源 Windows-GUI 软件工具
c语言情人节的红玫瑰
c语言24点游戏源码
c语言实现的汉诺塔演示程序
android_jni操作指南
网上绝无仅有的Log分析教程及例子
Jumpserver 是一款由python编写开源的跳板机(堡垒机)系统,实现了跳板机应有的功能。基于ssh协议来管理,客户端无需安装agent。 Jumpserver 3.0 架构上和 2.0 变化较大,建议全新安装一套环境来体验。如需升级,请务必升级前进行备份。 支持常见系统: 1、redhat centos 2、debian 3、suse ubuntu 4、freebsd 5、其他ssh协议硬件设备 特点: 完全开源,GPL授权 Python编写,容易再次开发 实现了跳板机基本功能,认证、授权、审计 集成了Ansible,批量命令等 支持WebTerminal Bootstrap编写,界面美观 自动收集硬件信息 录像回放 命令搜索 实时监控 批量上传下载
# 基于Arduino框架的自动称重系统 ## 项目简介 本项目是一个基于Arduino框架的自动称重系统。它利用Arduino硬件和Adafruit的ADS1115 ADC(模数转换器)库,实现了从负载单元读取重量数据并通过串行通信将数据传输到PC或其他设备的功能。项目还包含了LCD屏幕显示和LED指示灯的控制,以及对数据库的操作和Web交互的支持。 ## 项目的主要特性和功能 1. 硬件连接与通信: 项目使用了Arduino和ADS1115 ADC之间的串行通信,实现了从负载单元读取重量数据的功能。 2. 数据处理: 通过ADC读取的重量数据被处理并转换为可读的数值,然后通过串行端口发送到PC或其他设备。 3. 用户界面: 包含了LCD屏幕显示和LED指示灯的控制,用于实时显示重量数据或指示重量状态。 4. 数据库操作: 项目支持通过串行通信与数据库交互,实现数据的存储和查询。
西安旅游管理系统 SSM毕业设计 附带论文 启动教程:https://www.bilibili.com/video/BV1GK1iYyE2B
python 显示二维码如下方法 import argparse import qrcode as qrcode import matplotlib.pyplot as plt filename = '二维码.png' # 获取二维码 def make_qr_code(): # 接收cmd命令里面的参数 parse = argparse.ArgumentParser() parse.add_argument("--t", type=str, default = None) args = parse.parse_args() content = args.t print("generate content "+content) q = qrcode.QRCode() q.add_data(content) q.make() img = q.make_image() img.save(filename) plt.title = '登录二维码' plt.imshow(img)
浮动搜索框(SearchManager)