JAVA导出文件

public class DownFileUtils {
    /**
     * 下载项目目录下的模板文件
     * @param httpRequest  请求信息
     * @param httpResponse 响应信息
     * @param pathFileName 带路径的文件名字
     */
    public static void downloadLocalTemplate(HttpServletRequest httpRequest, HttpServletResponse httpResponse, String pathFileName) throws Exception {
        // 通过路径获取项目中的文件
        byte[] bytes = ResourceUtil.readBytes(pathFileName);
        InputStream fis = new ByteArrayInputStream(bytes);

        // 分隔路径
        String[] split = pathFileName.split("/");
        // 获取模板名称
        String rawFileName = split[split.length - 1];
        // 名称
        String fileName = FileNameUtil.getPrefix(rawFileName);
        // 类型
        String suffix = FileNameUtil.getSuffix(rawFileName);

        // 构建响应信息
        httpResponse.setContentType("application/application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=utf-8");
        httpResponse.setHeader("Content-Disposition", "attachment;filename=" + getFileName(fileName, httpRequest) + "." + suffix);
        ServletOutputStream outputStream = httpResponse.getOutputStream();

        if (fis.available() != 0) {
            // 设置响应码
            httpResponse.setStatus(200);
        }

        // 导出文件流
        IOUtils.copy(fis, outputStream);
        fis.close();
        outputStream.flush();
        outputStream.close();
    }

    /**
     * 处理文件名字编码
     * @param fileName 文件名字
     * @param request  请求信息
     * @return 编码后文件名
     * @throws Exception 异常信息
     */
    private static String getFileName(String fileName, HttpServletRequest request) throws Exception {
        String ua = request.getHeader("User-Agent").toLowerCase();
        if (ua.contains("macintosh") && !ua.contains("chrome")) {
            //浏览器: safari
            return new String(fileName.getBytes(), "ISO8859-1");
        }
        //默认utf-8
        return URLEncoder.encode(fileName, StandardCharsets.UTF_8.name());
    }
}

Java中,实现文件导出功能的限流通常是为了防止因大量并发请求导致的系统资源耗尽或响应速度变慢。你可以通过以下几种策略来实现: 1. **线程池限流**:使用`java.util.concurrent.Executors`创建固定大小的线程池,当有请求到来时,先检查线程池是否还有空闲线程,如果有则执行,如果没有,则加入队列等待。这样可以限制同时处理导出任务的数量。 ```java ExecutorService executor = Executors.newFixedThreadPool(10); Future<String> result = executor.submit(() -> { // 文件导出操作 }); ``` 2. **令牌桶算法**:模拟现实中水龙头的流量控制机制,每次请求获取一个令牌(通常是单位时间内的导出次数),如果没有足够的令牌则拒绝请求,直到令牌刷新。 3. **RateLimiter**:Java提供了一个`java.util.concurrent.RateLimiter`类,用于基于令牌的速率限制。这允许你在一段时间内分配一定数量的“令牌”给请求,并在令牌耗尽时阻塞。 ```java RateLimiter rateLimiter = RateLimiter.create(5.0); // 每秒最多5次导出请求 if (!rateLimiter.tryAcquire()) { // 请求被拒绝,记录日志或返回错误信息 } // 导出文件 ``` 4. **Spring AOP** 或其他框架的限流特性:如果你的应用使用了Spring框架,可以通过AOP(面向切面编程)来添加全局的限流规则,比如Hystrix或Resilience4j等库。 对于以上策略,你需要根据实际业务需求和系统的处理能力来调整限流阈值,以及设置合适的超时策略。记得在完成文件导出后关闭资源并释放对应的限流控制。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值