Java导出MySQL数据成csv文件并压缩成zip

Background

业务需求:提供数据导出功能。数据为传感器监测数据,存储在MySQL。需要用户传入传感器的IDs,起始时间和结束时间,由于数据量较大,所以这里通过定时任务实现,然后把每个传感器数据写一个csv文件,最终把所有的文件压缩成一个zip,再把压缩包的存储路径写入数据库,用户请求下载数据时从数据库查询获取压缩包的存储路径,最终把压缩包返回给用户。下面是主要代码(springboot+swagger2)

Ctroller

package com.cloudansys.monitor.solution.export.controller;

import com.cloudansys.monitor.base.BaseController;
import com.cloudansys.monitor.common.CSVUtils;
import com.cloudansys.monitor.solution.export.entity.TaskExportDTO;
import com.cloudansys.monitor.solution.export.entity.TaskInitDTO;
import com.cloudansys.monitor.solution.export.service.ExportService;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiImplicitParam;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.compress.utils.IOUtils;
import org.apache.commons.io.FileUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;

import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.OutputStream;
import java.util.List;

@Slf4j
@Api(tags = "数据导出")
@RestController
@RequestMapping("/export")
public class ExportController extends BaseController {

    @Autowired
    private ExportService service;

    @ApiOperation("提交数据导出任务")
    @PostMapping("/submitExportJob")
    public Integer submitExportJob(@RequestBody TaskExportDTO exportDTO) {
        log.info("exportDTO: {}", exportDTO);
        return this.service.submitJob(exportDTO);
    }

    @ApiOperation("根据任务初始化ID进行下载")
    @ApiImplicitParam(name = "init_id", value = "任务初始化ID")
    @GetMapping("/downloadFile/{init_id}")
    public void downloadFile(@PathVariable Integer init_id, HttpServletResponse response) {
        String filePath = this.service.getPathByInitID(init_id);
        log.info("filePath: {}", filePath);
        CSVUtils.downloadZipFile(filePath, response);
    }

    @ApiOperation("获取用户数据导出任务")
    @PostMapping("/getExportJob/{user_id}")
    public List<TaskInitDTO> getExportJob(@PathVariable Integer user_id) {
        log.info("user_id: {}", user_id);
        return this.service.getExportJob(user_id);
    }

}

定时任务

package com.cloudansys.monitor.solution.export.schedule;

import com.cloudansys.monitor.common.CSVUtils;
import com.cloudansys.monitor.common.CacheHandler;
import com.cloudansys.monitor.common.ZipUtils;
import com.cloudansys.monitor.entity.FileBean;
import com.cloudansys.monitor.solution.data.entity.TargetPrimaryData;
import com.cloudansys.monitor.solution.data.service.TargetPrimaryDataService;
import com.cloudansys.monitor.solution.export.entity.TaskExportDTO;
import com.cloudansys.monitor.solution.export.entity.TaskInitDTO;
import com.cloudansys.monitor.solution.export.service.ExportService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;

import java.util.ArrayList;
import java.util.Date;
import java.util.List;

@Slf4j
@Component
@EnableScheduling
public class ScheduleExportJob implements ScheduleJob, CommandLineRunner {

    @Autowired
    private JdbcTemplate jdbcTemplate;

    @Autowired
    private ExportService exportService;

    @Autowired
    private TargetPrimaryDataService targetService;

    @Autowired
    private CacheHandler cacheHandler;

    @Override
    public void run(String... args) {
//        log.info("定时任务开始 . . .");
        exec();
    }

    // {秒} {分} {时} {日} {月} {星期} {年份(可为空)}
    @Scheduled(cron = "* */10 * * * ?")
    private void exec() {

        // 压缩目录和生成的压缩文件
        String srcFile = "data/data-targets";
        String zipFile = "data/data-targets.zip";

        String sql_1 = "SELECT id FROM t_task_init WHERE status = 0;";
        String sql_2 = "SELECT target_ids, start_time, end_time FROM t_task_export WHERE init_id = ?;";
//        log.debug("sql :{} ", sql_1);
//        log.debug("sql_2 :{} ", sql_2);

        // 获取需要执行的任务的 init_ids
        List<Integer> init_ids = this.jdbcTemplate.queryForList(sql_1, Integer.class);

        // 首先判断有没有需要执行的数据导出任务
        if (null == init_ids || init_ids.size() == 0) {
            return;
        }

        // 遍历 init_ids 一个一个执行任务
        init_ids.forEach(init_id -> {
            List<TaskExportDTO> exportDTOs = this.jdbcTemplate.query(sql_2, new Object[]{init_id},
                    (resultSet, i) -> {
                        TaskExportDTO exportDTO = new TaskExportDTO();
                        exportDTO.setInit_id(init_id);
                        exportDTO.setTarget_ids(resultSet.getString(++i));
                        exportDTO.setSTime(resultSet.getDate(++i));
                        exportDTO.setETime(resultSet.getDate(++i));
                        return exportDTO;
                    });

            // sql_2 查询出来的总是只有一个 TaskExportDTO
            TaskExportDTO exportDTO = exportDTOs.get(0);

            // 开始执行 init_id 的任务,并更新任务状态
            TaskInitDTO initDTO = new TaskInitDTO();
            initDTO.setId(init_id);
            initDTO.setStatus(1);
            this.exportService.updateTaskInit(initDTO);

            // 遍历测点ID,查询数据,每个测点的数据生成一个以测点编号命名的 csv 文件,都放在 data/data-targets 目录
            String[] target_ids = exportDTO.getTarget_ids().split(",");
            Date sTime = exportDTO.getSTime();
            Date eTime = exportDTO.getETime();
            for (String target_id : target_ids) {
                Integer targetId = Integer.valueOf(target_id);
                List<TargetPrimaryData> targetPrimaryData = this.targetService.getByTargetId(targetId, sTime, eTime);

                // 构建该测点数据生成 csv 文件的文件头 [参数1,参数2,数据时间,. . .]
                String[] param = targetPrimaryData.get(0).getParam();
                List<Object> head = new ArrayList<>();
                for (String p : param) {
                    head.add(p);
                }
                head.add("数据时间");
                List<List<Object>> dataList = new ArrayList<>();
                for (TargetPrimaryData target : targetPrimaryData) {
                    Double[] data = target.getData();
                    List<Object> list = new ArrayList<>();
                    for (Double d : data) {
                        list.add(d);
                    }
                    list.add(target.getTime());
                    dataList.add(list);
                }

                FileBean fileBean = new FileBean();
                fileBean.setFileID(targetId);

                // 从缓存中获取该 targetId 对应的 targetCode
                String targetCode = cacheHandler.getTargetCode(targetId);
                fileBean.setFileName(targetCode);
                fileBean.setFilePath("data/data-targets");

                // 将该 targetId 测点数据写入 csv 文件,文件名为 targetCode
                CSVUtils.createCSVFile(head, dataList, fileBean);
//                log.info("================csv 文件创建结束!");
            }

            // 一个数据导出任务所需的数据处理结束后,生成的 csv 数据文件都在 data/data-targets 目录下
            // 然后 把 data-targets 目录压缩成 zip
            ZipUtils.doCompress(srcFile, zipFile);
            log.info("================csv 文件压缩结束!");

            // 最终把压缩包的路径存储到数据库中,并更新任务状态
            initDTO.setStatus(2);
            initDTO.setETime(new Date());
            initDTO.setPath(zipFile);
            this.exportService.updateTaskInit(initDTO);
            log.info("================csv 文件路径存储结束!");
        });
    }

}

CSV工具类

经测试三个字段,1千万条数据写成csv文件仅3秒

package com.cloudansys.monitor.common;

import com.cloudansys.monitor.entity.FileBean;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.compress.utils.IOUtils;
import org.apache.commons.csv.CSVFormat;
import org.apache.commons.csv.CSVPrinter;
import org.apache.commons.io.FileUtils;

import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.io.OutputStream;
import java.time.Instant;
import java.time.temporal.ChronoUnit;
import java.util.List;

@SuppressWarnings("ResultOfMethodCallIgnored")
@Slf4j
public class CSVUtils {

    /**
     * @param head     csv 文件头
     * @param dataList 要写入 csv 文件中的数据
     * @param fileBean 文件实体类
     * @return 返回生成的 csv 文件的路径
     */
    public static String createCSVFile(List<Object> head, List<List<Object>> dataList, FileBean fileBean) {

        Instant start_time = Instant.now();

        File csvFile = null;
        try {
            csvFile = new File(fileBean.getFilePath() + File.separator + fileBean.getFileName() + ".csv");
            File parent = csvFile.getParentFile();
            if (parent != null && !parent.exists()) {
                parent.mkdirs();
            }
            csvFile.createNewFile();
            FileWriter writer = new FileWriter(csvFile);
            CSVPrinter printer = CSVFormat.EXCEL.print(writer);

            // 写入文件头部
            printer.printRecord(head);

            // 写入文件内容
            for (List<Object> row : dataList) {
                printer.printRecord(row);
            }
            printer.flush();
            printer.close();
        } catch (IOException e) {
            e.printStackTrace();
        }

        Instant end_time = Instant.now();
        long seconds = ChronoUnit.SECONDS.between(start_time, end_time);
//        log.info("耗时:{}", seconds);
//        System.out.println("写入成功!");

        return csvFile.getPath();
    }

    /**
     * 压缩 csv 文件成 zip,并提供下载
     *
     * @param filePath 要进行压缩的文件目录
     * @param response      下载请求响应对象
     */
    public static void downloadZipFile(String filePath, HttpServletResponse response) {
        File file = new File(filePath);
        if (!file.exists()) {
            return;
        }
        OutputStream os = null;
        try {
            os = response.getOutputStream();
            response.reset();
            response.setHeader("Content-Disposition", "attachment;filename=" + file.getName());
            response.setContentType("application/octet-stream");
            os.write(FileUtils.readFileToByteArray(file));
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            IOUtils.closeQuietly(os);
        }
    }

}

ZIP压缩工具

package com.cloudansys.monitor.common;

import lombok.extern.slf4j.Slf4j;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;

@Slf4j
public class ZipUtils {

    private ZipUtils() {
    }

    public static void doCompress(String srcFile, String zipFile) {
        try {
            doCompress(new File(srcFile), new File(zipFile));
        } catch (IOException e) {
            log.info("doCompress——文件压缩出错了!");
            e.printStackTrace();
        }
    }

    /**
     * 文件压缩
     *
     * @param srcFile 目录或者单个文件
     * @param zipFile 压缩后的ZIP文件
     */
    private static void doCompress(File srcFile, File zipFile) throws IOException {
        ZipOutputStream out = null;
        try {
            out = new ZipOutputStream(new FileOutputStream(zipFile));
            doCompress(srcFile, out);
        } catch (Exception e) {
            throw e;
        } finally {
            out.close();//记得关闭资源
        }
    }

    static void doCompress(String fileName, ZipOutputStream out) throws IOException {
        doCompress(new File(fileName), out);
    }

    private static void doCompress(File file, ZipOutputStream out) throws IOException {
        doCompress(file, out, "");
    }

    private static void doCompress(File inFile, ZipOutputStream out, String dir) throws IOException {
        if (inFile.isDirectory()) {
            File[] files = inFile.listFiles();
            if (files != null && files.length > 0) {
                for (File file : files) {
                    String name = inFile.getName();
                    if (!"".equals(dir)) {
                        name = dir + "/" + name;
                    }
                    ZipUtils.doCompress(file, out, name);
                }
            }
        } else {
            ZipUtils.doZip(inFile, out, dir);
        }
    }

    private static void doZip(File inFile, ZipOutputStream out, String dir) throws IOException {
        String entryName;
        if (!"".equals(dir)) {
            entryName = dir + "/" + inFile.getName();
        } else {
            entryName = inFile.getName();
        }
        ZipEntry entry = new ZipEntry(entryName);
        out.putNextEntry(entry);
        int len;
        byte[] buffer = new byte[1024];
        FileInputStream fis = new FileInputStream(inFile);
        while ((len = fis.read(buffer)) > 0) {
            out.write(buffer, 0, len);
            out.flush();
        }
        out.closeEntry();
        fis.close();
    }

}
  • 1
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
### 回答1: 使用java语言查询数据导出csv文件可以使用Java语言中提供的JDBC API和ResultSet类。首先,连接数据库,然后构建查询语句,接着使用ResultSet类检索查询结果,最后将结果写入CSV文件中。 ### 回答2: 使用Java语言查询数据导出CSV文件可以分为以下几个步骤: 1. 连接数据库:首先,需要在Java代码中连接到数据库。可以使用Java提供的JDBC(Java Database Connectivity) API来连接到各种类型的数据库,如MySQL、Oracle等。需要设置连接URL、用户名和密码等数据库连接信息。 2. 执行查询语句:通过使用Java的JDBC API,可以执行SQL查询语句从表中获取数据。可以使用预编译的语句来参数化查询语句,以便更安全和高效地执行查询操作。 3. 将查询结果写入CSV文件:在获取查询结果后,通过Java中的文件操作API,可以将查询结果写入CSV文件。可以使用Java中的File类和相关的输入/输出流来创建、写入和关闭CSV文件。 4. CSV文件格式:CSV文件是以逗号分隔的文本文件,每行代表表中的一条数据记录,每个字段之间使用逗号分隔。可以使用Java中的字符串操作API来格式化数据并写入CSV文件中。 5. 异常处理:在代码中需要进行适当的异常处理,例如数据库连接失败、SQL语句执行错误或文件操作错误等。可以使用Java中的异常处理机制(try-catch-finally)来捕获和处理异常,并在出现异常时进行适当的处理,如记录日志或回滚事务。 总结起来,使用Java语言查询数据导出CSV文件,主要涉及到数据库连接、SQL查询文件操作和异常处理等方面的知识。通过合理应用Java中的API和相关技术,可以实现对表数据查询导出操作,生符合CSV格式的文件。 ### 回答3: 在使用Java语言查询数据导出CSV文件之前,首先需要使用Java数据库连接(JDBC)驱动连接到数据库。可以选择合适的JDBC驱动程序来连接不同类型的数据库,比如MySQL、Oracle等。 然后,通过编写Java代码来执行SQL查询语句来获取表中的数据。可以使用PreparedStatement或Statement对象来执行查询,并通过ResultSet对象来获取查询结果的数据。 接下来,需要将查询结果的数据按照CSV文件的格式进行处理,即将每一行数据转换为以逗号分隔的字段。可以使用StringBuilder或StringBuffer来构建CSV文件的内容。 最后,将CSV文件内容写入到磁盘上的文件中。可以使用FileWriter或BufferedWriter等类来实现文件的写入操作。在写入过程中,需要注意处理异常和关闭文件资源。 以下是一种可能的实现方式: 1. 导入JDBC驱动: ```java import java.sql.*; ``` 2. 连接到数据库: ```java String url = "jdbc:mysql://localhost:3306/数据库名"; String User = "用户名"; String Password = "密码"; Connection conn = null; try { conn = DriverManager.getConnection(url, user, password); } catch (SQLException e) { e.printStackTrace(); } ``` 3. 执行SQL查询语句并获取结果: ```java String sql = "SELECT * FROM 表名"; Statement stmt = null; ResultSet rs = null; try { stmt = conn.createStatement(); rs = stmt.executeQuery(sql); // 处理查询结果,将数据转换为CSV格式 StringBuilder csvContent = new StringBuilder(); ResultSetMetaData metaData = rs.getMetaData(); int columnCount = metaData.getColumnCount(); // 写入CSV列名 for (int i = 1; i <= columnCount; i++) { csvContent.append(metaData.getColumnLabel(i)); if (i < columnCount) { csvContent.append(","); } } csvContent.append("\n"); // 写入CSV数据 while (rs.next()) { for (int i = 1; i <= columnCount; i++) { csvContent.append(rs.getString(i)); if (i < columnCount) { csvContent.append(","); } } csvContent.append("\n"); } // 将CSV内容写入文件 String csvFilePath = "导出文件路径"; BufferedWriter writer = null; try { writer = new BufferedWriter(new FileWriter(csvFilePath)); writer.write(csvContent.toString()); } catch (IOException e) { e.printStackTrace(); } finally { if (writer != null) { try { writer.close(); } catch (IOException e) { e.printStackTrace(); } } } } catch (SQLException e) { e.printStackTrace(); } finally { try { if (rs != null) { rs.close(); } if (stmt != null) { stmt.close(); } if (conn != null) { conn.close(); } } catch (SQLException e) { e.printStackTrace(); } } ``` 以上代码示例展示了如何使用Java语言查询数据并将其导出CSV文件。需要注意的是,代码中的数据库连接信息、SQL语句、导出文件路径等都需要根据实际情况进行修改。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

WaiSaa

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值