java接口下载zip,不生成中间文件,返回前端文件流

java接口下载zip,不生产中间文件,返回前端文件流

    • 程序设计:
    • 代码实现:

程序设计:

前端向后端请求zip文件,zip文件中有多个文件压缩而成,后端操作文件流,而不生成中间文件。最后把zip返回给前端。

代码实现:

    @ApiOperation(value = "下载Zip", notes = "")
    @PostMapping("/getDownLoadZip")
    public void getDownLoadZip(@RequestBody GClientManagementVo vo, HttpServletRequest request,HttpServletResponse response) throws Exception {
        SysUserEntityVo uc = (SysUserEntityVo) request.getAttribute("UC");
        GClientManagementService.getDownLoadZip(vo, uc,response);
    }

中间有业务代码,可酌情删减。代码中注意关闭流,避免影响内存。

 @Override
    public void getDownLoadZip(GClientManagementVo vo, SysUserEntityVo uc, HttpServletResponse response) throws Exception {
        String filename = vo.getProjectName();
        String encodeFileName = URLEncoder.encode(filename);
        ServletOutputStream out = response.getOutputStream();
        // 创建一个ByteArrayOutputStream来存放最终的ZIP流
        ByteArrayOutputStream zipOutputStream = new ByteArrayOutputStream();
        ZipOutputStream zipOut = new ZipOutputStream(zipOutputStream);
        try {
            //写入文件
            createFilePs(vo, zipOut);
            //结束写入
            zipOut.finish();
            // 最终ZIP流的内容
            byte[] zipBytes = zipOutputStream.toByteArray();

            //设置允许跨域的key
            response.setHeader("Access-Control-Expose-Headers", "Content-Disposition");
            //文件名有“,”等特殊字符发送到前端会报错,用""括起来解决
            response.addHeader("Content-Disposition", "attachment;filename=\"" + encodeFileName + "\"");
            //设置文件大小
            response.addHeader("Content-Length", "" + zipBytes.length);
            //设置文件名,避免问题,这个也用""括起来
            response.setHeader("filename,", "filename=\"" + encodeFileName + "\"");
            //设置文件类型
            response.setContentType("application/octet-stream");
            out.write(zipBytes);
            out.flush();

        } catch (Exception e) {
            throw e;
        } finally {
            try {
                out.close();
            } catch (Exception e) {
                throw e;
            }
            try {
                zipOutputStream.close();
            } catch (Exception e) {
                throw e;
            }

            try {
                zipOut.close();
            } catch (Exception e) {
                throw e;
            }

            try {
                out.close();
            } catch (Exception e) {
                throw e;
            }
        }
    }


    private void createFilePs(GClientManagementVo vo, ZipOutputStream zipOut) throws Exception {
        // 创建第一个文件install 或者uninstall
        ByteArrayOutputStream file1 = new ByteArrayOutputStream();
        //install  or uninstall
        String str1 = "";
        String fileName1 = "";
        QueryWrapper<GClientAdmin> queryWrapper = new QueryWrapper();
        if (vo.getFunction().equalsIgnoreCase(ParamsEnum.Install.getValue())) {
            queryWrapper.eq("sign", ParamsEnum.Install.getValue());
        } else {
            queryWrapper.eq("sign", ParamsEnum.Uninstall.getValue());
        }
        queryWrapper.last("limit 1");
        GClientAdmin gClientAdmin = gClientAdminDao.selectOne(queryWrapper);
        fileName1 = gClientAdmin.getScriptName();
        str1 = replaceScriptcode(gClientAdmin.getScriptCode(), vo);
        try {
            // 创建ByteArrayOutputStream来模拟文件流
            writeToFile(file1, str1);
            // 将每个文件流添加到ZIP流中
            addToZipFile(fileName1, file1.toByteArray(), zipOut);
        } catch (IOException ex) {
            ex.printStackTrace();
            throw ex;
        }

        //多个已经上传文件
        List<GClientManagementScriptVo> scriptList = vo.getScriptList();
        if (CollectionUtils.isNotEmpty(scriptList)) {
            scriptList.forEach(e -> {
                List<ParamsObject> paramslist = e.getParamslist();
                if (CollectionUtils.isNotEmpty(paramslist)) {
                    paramslist.forEach(m -> {
                        if (ParamsEnum.File.getValue().equalsIgnoreCase(m.getDataType())) {
                            String str = m.getFileContent();
                            String fileName = m.getFileName();
                            //创建ByteArrayOutputStream来模拟文件流
                            ByteArrayOutputStream file = new ByteArrayOutputStream();
                            try {
                                // 创建ByteArrayOutputStream来模拟文件流
                                writeToFile(file, str);
                                // 将每个文件流添加到ZIP流中
                                addToZipFile(fileName, file.toByteArray(), zipOut);
                            } catch (IOException ex) {
                                ex.printStackTrace();
                            }
                        }
                    });
                }
            });
        }
    }

    private String replaceScriptcode(String scriptcode, GClientManagementVo vo) throws Exception {
        //查找主参数的字符串
        String startParams = ParamsEnum.Params_Replace_Start.getValue();
        String endParams = ParamsEnum.Params_Replace_End.getValue();
        String resultParams = findSubstringBetween(scriptcode, startParams, endParams);
        if (resultParams != null) {
            //开始替换主参数

            String result = resultParams
                    //projectName
                    .replace(ParamsEnum.ProjectName.getValue(), "\"" + vo.getProjectName() + "\"")
                    //applicationName
                    .replace(ParamsEnum.MSIApplicationName.getValue(), "\"" + vo.getApplicationName() + "\"")
                    //cmdRcopt
                    .replace(ParamsEnum.CmdRCOpt.getValue(), vo.getCmdRcopt())
                    //maxRunTime
                    .replace(ParamsEnum.MaxRunTime.getValue(), vo.getMaxRunTime().toString())
                    //startStopService
                    .replace(ParamsEnum.StartStopService.getValue(), "\"" + vo.getStartStopService() + "\"")
                    //debugEnable
                    .replace(ParamsEnum.DebuggingEnabled.getValue(), "$" + vo.getDebugEnable())
                    //defaultMsicLine
                    .replace(ParamsEnum.DefaultMSICmdLine.getValue(), "\"" + vo.getDefaultMsicLine() + "\"")
                    //uDriveMap
                    .replace(ParamsEnum.UDriveMap.getValue(), "$" + vo.getuDriveMap())
                    //function
                    .replace(ParamsEnum.sFunction.getValue(), "\"" + vo.getFunction().toUpperCase(Locale.ROOT) + "\"")
                    //ifsScript
                    .replace(ParamsEnum.bIsIFSScript.getValue(), "$" + vo.getIfsScript());

            //替换主脚本字符串
            // 使用 replaceAll 方法和正则表达式来替换所有匹配的子字符串
            // 注意:这里使用了正则表达式,所以需要对特殊字符进行转义
            int startIndex = scriptcode.indexOf(startParams);
            if (startIndex != -1) {
                int endIndex = scriptcode.indexOf(endParams, startIndex);
                if (endIndex != -1) {
                    scriptcode = scriptcode.substring(0, startIndex) + result + scriptcode.substring(endIndex + endParams.length());
                } else {
                    throw new Exception("End tag not found.");
                }
            } else {
                throw new Exception("Start tag not found.");
            }

            //查找子参数的字符串
            String startCommand = ParamsEnum.Command_Replace_Start.getValue();
            String endCommand = ParamsEnum.Command_Replace_End.getValue();
            AtomicReference<String> codeStr = new AtomicReference<>("");
            if (CollectionUtils.isNotEmpty(vo.getScriptList())) {
                vo.getScriptList().forEach(e -> {

                    AtomicReference<String> finalCode = new AtomicReference<>(e.getScriptCode());
                    if (CollectionUtils.isNotEmpty(e.getParamslist())) {
                        e.getParamslist().forEach(m -> {
                            String value = "";
                            String name = "";
                            if (m.getDataType().equalsIgnoreCase(ParamsEnum.Boolean.getValue())) {
                                value = "$" + m.getValue();
                                name = "${" + m.getName() + "}";
                            } else if (m.getDataType().equalsIgnoreCase(ParamsEnum.String.getValue())) {
                                //CustomParameters特殊情况,全string参数
                                if(m.getName().equalsIgnoreCase(ParamsEnum.CustomParameters.getValue())){
                                    if(StringUtils.startsWith(m.getValue().toString(),"\"")&&StringUtils.endsWith(m.getValue().toString(),"\"")){
                                        value = m.getValue().toString();
                                    }else {
                                        value = "\"" + m.getValue() + "\"";
                                    }

                                }else {
                                    value = "\"" + m.getValue() + "\"";
                                }

                                name = "${" + m.getName() + "}";
                            } else if (m.getDataType().equalsIgnoreCase(ParamsEnum.Int.getValue())) {
                                value = m.getValue().toString();
                                name = "${" + m.getName() + "}";
                            } else if (m.getDataType().equalsIgnoreCase(ParamsEnum.File.getValue())) {
                                value = "\"\\" + m.getFileName() + "\"";
                                name = "${" + m.getName() + "}";
                            }
                            finalCode.set(finalCode.get().replace(name, value));
                        });
                    }
                    codeStr.set(codeStr.get() + "\n" + finalCode.get());
                });
            }
            //子参数替换
            int startIndex2 = scriptcode.indexOf(startCommand);
            if (startIndex2 != -1) {
                int endIndex2 = scriptcode.indexOf(endCommand, startIndex2);
                if (endIndex2 != -1) {
                    scriptcode = scriptcode.substring(0, startIndex2) + codeStr.get() + scriptcode.substring(endIndex2 + endCommand.length());
                } else {
                    throw new Exception("End tag not found.");
                }
            } else {
                throw new Exception("Start tag not found.");
            }

        } else {
            throw new Exception("error");
        }
        return scriptcode;

    }

    public static String findSubstringBetween(String source, String start, String end) {
        // 找到开始字符串的位置
        int startIndex = source.indexOf(start);
        // 开始字符串不存在
        if (startIndex == -1) {
            return null;
        }
        // 在开始字符串之后找到结束字符串的位置
        int endIndex = source.indexOf(end, startIndex + start.length());
        // 结束字符串不存在
        if (endIndex == -1) {
            return null;
        }
        // 提取并返回子字符串
        return source.substring(startIndex + start.length(), endIndex);
    }


    private void writeToFile(ByteArrayOutputStream out, String content) throws IOException {
        try (OutputStreamWriter writer = new OutputStreamWriter(out, "UTF-8")) {
            writer.write(content);
            writer.flush();
        } finally {
            try {
                out.close();
            } catch (Exception e) {
                throw e;
            }
        }
    }

    private void addToZipFile(String fileName, byte[] fileContent, ZipOutputStream zipOut) throws IOException {
        ZipEntry zipEntry = new ZipEntry(fileName);
        zipOut.putNextEntry(zipEntry);
        zipOut.write(fileContent);
        zipOut.closeEntry();
    }

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/948052.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Vue3 中的插槽

Vue3 中插槽的使用&#xff0c;插槽是 Vue 中的一个特别特性&#xff0c;插槽就是模版内容。例如<h1>标题 1</h1>标题 1 就是插槽&#xff0c;Vue 是无法识别模板内容的&#xff0c;只能通过属性进行传递。Slot 主要包括默认、具名和作用域。Slot开发起来难度不大&…

JAVA-制作小游戏期末实训

源码 import game.frame.Frame;public class App {public static void main(String[] args) {System.out.println("starting......");new Frame();} } package game.controller;import game.model.Enemy;public class EnemyController implements Runnable{private…

Linux Red Hat 7.9 Server安装GitLab

1、关闭防火墙 执行 systemctl disable firewalld 查看服务器状态 systemctl status firewalld 2、禁用selinux vi /etc/selinux/config 将SELINUX 的值改为 disabled 3、安装policycoreutils-python 执行 yum install policycoreutils-python 4、下载gitlab wget --co…

Windows 环境配置 HTTPS 服务实战

一、 环境准备 win10以上操作系统安装 Certbot申请阿里云\腾讯云域名安装 nginx 1.3以上版本 二、Certbot 安装及 SSL 证书生成 Certbot 是一个免费、开源工具&#xff0c;用于自动化在Web服务器上获取和更新SSL/TLS证书。它可以通过Let’s Encrypt服务获取免费的SSL/TLS证书…

普及组集训数据结构--并查集

P1551 亲戚 - 洛谷 | 计算机科学教育新生态 并查集就是把所有相关联的量串成一串珠子&#xff0c;抽象来说就是&#xff1a; 把此类相关联的量当作节点&#xff0c;两个节点之间连接一条无向边&#xff0c;所形成的图 例题算法流程&#xff1a; 在此定义“族长”就是一个树的…

windows编译llama.cpp GPU版本

Build 指南 https://github.com/ggerganov/llama.cpp/blob/master/docs/build.md 一、Prerequire 具体步骤&#xff08;以及遇到的坑&#xff09;&#xff1a; 如果你要使用CUDA&#xff0c;请确保已安装。 1.安装 最新的 cmake, git, anaconda&#xff0c; pip 配置pyt…

Android 性能优化:内存优化(实践篇)

1. 前言 前一篇文章Android性能优化&#xff1a;内存优化 &#xff08;思路篇&#xff09; 大概梳理了Android 内存原理和优化的必要性及应该如何优化&#xff0c;输出了一套短期和长期内存优化治理的SOP方案。 那么这一篇文章就总结下我最近在做内存优化如何实践的&#xff0…

「Mac畅玩鸿蒙与硬件53」UI互动应用篇30 - 打卡提醒小应用

本篇教程将实现一个打卡提醒小应用&#xff0c;通过用户输入时间进行提醒设置&#xff0c;并展示实时提醒状态&#xff0c;实现提醒设置和取消等功能。 关键词 打卡提醒状态管理定时任务输入校验UI交互 一、功能说明 打卡提醒小应用包含以下功能&#xff1a; 提醒时间输入与…

Nginx知识详解(理论+实战更易懂)

目录 一、Nginx架构和安装 1.1 Nginx 概述 1.1.1 nginx介绍 1.1.2?Nginx 功能介绍 1.1.3?基础特性 1.1.4?Web 服务相关的功能 1.2?Nginx 架构和进程 1.2.1?Nginx 进程结构 1.2.2?Nginx 进程间通信 1.2.3?Nginx 启动和 HTTP 连接建立 1.2.4?HTTP 处理过程 1…

Postgresql 命令还原数据库

因为PgAdmin打不开&#xff0c;但是数据库已经安装成功了&#xff0c;这里借助Pg命令来还原数据库 C:\Program Files\PostgreSQL\15\bin\psql.exe #链接数据库 psql -U postgres -p 5432#创建数据库 CREATE DATABASE "数据库名称"WITHOWNER postgresENCODING UTF8…

Vue 解决浏览器刷新路由参数丢失问题 全局统一配置无需修改组件

在路由跳转的时候,我们经常会传一些参数过去,然后通过传过来的参数调用接口获取相关数据,但是刷新浏览器的时候路由参数会丢失。此时页面报错误了,如何通过全局配置的方式,不需要修改任何组件 实现刷新浏览器保存参数? 实现方式如下: 首先在router/index.js里添加参数管…

【AIGC】电话录音转文字实践:基于Google Cloud Speech-to-Text-v1的技术方案Python

文章目录 引言技术原理技术方案设计系统架构关键技术要点 代码实现1. 环境准备2. 核心代码实现3. 音频预处理工具响应格式 性能优化实践经验应用场景未来展望总结 引言 在当今数字化时代&#xff0c;将语音内容转换为文字已经成为一个非常重要的技术需求。无论是客服通话记录、…

RabbitMQ-基本使用

RabbitMQ: One broker to queue them all | RabbitMQ 官方 安装到Docker中 docker run \-e RABBITMQ_DEFAULT_USERrabbit \-e RABBITMQ_DEFAULT_PASSrabbit \-v mq-plugins:/plugins \--name mq \--hostname mq \-p 15672:15672 \-p 5672:5672 \--network mynet\-d \rabbitmq:3…

Android Camera压力测试工具

背景描述&#xff1a; 随着系统的复杂化和业务的积累&#xff0c;日常的功能性测试已不足以满足我们对Android Camera相机系统的测试需求。为了确保Android Camera系统在高负载和多任务情况下的稳定性和性能优化&#xff0c;需要对Android Camera应用进行全面的压测。 对于压…

vscode中调用deepseek实现AI辅助编程

来自 Python大数据分析 费弗里 1 简介 大家好我是费老师&#xff0c;最近国产大模型Deepseek v3新版本凭借其优秀的模型推理能力&#xff0c;讨论度非常之高&#x1f525;&#xff0c;且其官网提供的相关大模型API接口服务价格一直走的“价格屠夫”路线&#xff0c;性价比很高…

基于 LMS 算法的离散傅里叶分析器

基于 LMS&#xff08;Least Mean Squares&#xff0c;最小均方&#xff09;算法的离散傅里叶分析器是一种结合自适应滤波和频域分析的工具&#xff0c;用于动态估计信号的频谱成分。它将 LMS 自适应算法与离散傅里叶变换&#xff08;DFT&#xff09;的频率分解能力结合&#xf…

2022浙江大学信号与系统笔记

原视频地址&#xff1a;2022浙江大学信号与系统&#xff08;含配套课件和代码&#xff09; - 胡浩基老师-哔哩哔哩 ⭐⭐⭐ 我的笔记&#xff1a;飞书链接 - 信号与系统 基于视频&#xff0c;记得笔记&#xff0c;加了点自己的补充&#xff08;有的是问 ChatGPT 的&#xff09;…

K8s高可用集群之Kubernetes集群管理平台、命令补全工具、资源监控工具部署、常用命令

K8s高可用集群之Kubernetes管理平台、补全命令工具、资源监控工具部署 1.Kuboard可视化管理平台2.kubectl命令tab补全工具3.MetricsServer资源监控工具4.Kubernetes常用命令 1.Kuboard可视化管理平台 可以选择安装k8s官网的管理平台&#xff1b;我这里是安装的其他开源平台Kub…

Gitlab-runner 修改默认的builds_dir并使用custom_build_dir配置

gitlab-runner 修改默认的builds_dir并使用custom_build_dir配置 1. 说明2. 实操&#xff08;以docker执行器为例&#xff09;2.1 修改默认的builds_dir2.1.1 调整gitlab-runner的配置文件2.1.2 CI文件 2.2 启用custom_build_dir2.2.1 调整gitlab-runner的配置文件2.2.2 CI文件…

网络IP协议

IP&#xff08;Internet Protocol&#xff0c;网际协议&#xff09;是TCP/IP协议族中重要的协议&#xff0c;主要负责将数据包发送给目标主机。IP相当于OSI&#xff08;图1&#xff09;的第三层网络层。网络层的主要作用是失陷终端节点之间的通信。这种终端节点之间的通信也叫点…