用chatgpt实现 java导出excel复杂表。

记录一次使用chatgpt解决实际问题的,需求是在页面添加一个订单导出excel的功能,订单编号、订单明细,相同订单编号合并单元格,模板如下请添加图片描述
表头表尾不用说,
主要是表格内容部分,左边是订单编号,右边是订单明细,**这边难就难在一个订单编号可能存在多条明细,所以订单编号相同时需要合并单元格,**我们将需求整理清楚,下面就可以去询问chatgpt了,
首先我们试探性的问他:
在这里插入图片描述
他给出了回答,可以看出,他是完全知道怎么写的,接着我们就可以一步步继续深入问他了,下面就要带入自己的需求:
在这里插入图片描述
可以看到,我们这时将表单数据直接带进去询问,他依旧很清楚的给出了答案接下来我们继续深入询问他,创建的表单是否为父子结构,订单编号为父数据,其余为子列数据:
在这里插入图片描述
这时他依旧很清晰的给出了回答,并且他知道应该将父数据订单编号合并单元格。
真的很神奇,下面我们继续深入:我发现,他所有的数据都是根据我给的表单结构,自己造的,所以就询问他,如果数据是从数据库中取出来的,应该怎么写
在这里插入图片描述
他很快给出了回答,还是很正确的,但是这只是一个void main 测试方法,并不是我们想要的controller、service方法,于是我继续询问:如果是由页面发起请求,进行导出应该怎么写呢
在这里插入图片描述
这时他居然直接给出了正确的controller方法,现在已经离我们想要的代码非常接近了。
我们继续询问:让他加上表头、表尾、金额统计等等
在这里插入图片描述

在这里插入图片描述
接下来我们基本就可以进行手动测试了,测试过程中会出现各种问题,没关系。我们把发现的问题直接告诉他,让他修改就行,比如说我发现他忘记给内容添加表头了,直接就问他:
在这里插入图片描述
他很快会给出修改后的代码,当然,如果你代码测试中出现报错,也可以直接把代码和错误信息发过去问他:在这里插入图片描述
他很快会解决,并进行说明,总之你出现问题都可以直接询问,他都是能看得懂的,最终代码如下:

 @Override
    public void exportBill(HttpServletResponse response, List<Integer> idList)  throws Exception{
        // 获取数据
        List<BillProductOderVO> orders = billProductService.getAllOrders(idList);

        // 创建工作簿
        Workbook workbook = new XSSFWorkbook();
        // 创建工作表
        Sheet sheet = workbook.createSheet("Sheet1");
        // 创建表头单元格样式
        CellStyle headerStyle = workbook.createCellStyle();
        headerStyle.setAlignment(HorizontalAlignment.CENTER);
        headerStyle.setVerticalAlignment(VerticalAlignment.CENTER);
        Font headerFont = workbook.createFont();
        headerFont.setBold(true);
        headerStyle.setFont(headerFont);

        // 创建正式内容单元格样式
        CellStyle contentStyle = workbook.createCellStyle();
        contentStyle.setAlignment(HorizontalAlignment.CENTER);
        contentStyle.setVerticalAlignment(VerticalAlignment.CENTER);
        contentStyle.setBorderTop(BorderStyle.THIN);
        contentStyle.setBorderBottom(BorderStyle.THIN);
        contentStyle.setBorderLeft(BorderStyle.THIN);
        contentStyle.setBorderRight(BorderStyle.THIN);

        // 添加大表头
        Row bigHeaderRow = sheet.createRow(0);
        bigHeaderRow.setHeightInPoints(30);
        Cell bigHeaderCell = bigHeaderRow.createCell(0);
        bigHeaderCell.setCellValue("销售出库单");
        bigHeaderCell.setCellStyle(headerStyle);
        sheet.addMergedRegion(new CellRangeAddress(0, 0, 0, 7));


        // 添加第一行表头
        Row firstHeaderRow = sheet.createRow(1);
        firstHeaderRow.setHeightInPoints(20);
        List<String> firstHeader = Arrays.asList("日期:", DateUtil.getNowDayForChinese(), "单据编号:", "", "客户:","",  "制单人:",JWTUtil.getCurrentUser().getTokenRealName());
        for (int i = 0; i < firstHeader.size(); i++) {
            Cell cell = firstHeaderRow.createCell(i);
            cell.setCellValue(firstHeader.get(i));
            cell.setCellStyle(headerStyle);
        }

        // 添加第二行表头
        Row secondHeaderRow = sheet.createRow(2);
        secondHeaderRow.setHeightInPoints(20);
        List<String> secondHeader = Arrays.asList("联系人:","", "联系电话:","", "部门:","", "仓库:","");
        for (int i = 0; i < secondHeader.size(); i++) {
            Cell cell = secondHeaderRow.createCell(i);
            cell.setCellValue(secondHeader.get(i));
            cell.setCellStyle(headerStyle);
        }

        // 添加正式数据表头
        Row headerRow = sheet.createRow(3);
        headerRow.setHeightInPoints(20);
        List<String> headers = Arrays.asList("订单编号","存货名称", "规格型号", "计量单位", "数量", "单价", "金额","");
        for (int i = 0; i < headers.size(); i++) {
            Cell cell = headerRow.createCell(i);
            cell.setCellValue(headers.get(i));
            cell.setCellStyle(headerStyle);
        }

        // 创建数据行,并设置内容
        int rowIndex = 4;
        BigDecimal totalAmount = BigDecimal.ZERO;
        String lastParentId = "";
        Map<String, Integer> startRows = new HashMap<>();
        Map<String, Integer> endRows = new HashMap<>();

        for (BillProductOderVO order : orders) {
            String parentId = order.getOrderNo();
            if (parentId.equals(lastParentId)) {
                // 当前行与上一行的订单编号相同,不需要合并单元格,更新结束行
                Row row = sheet.createRow(rowIndex++);
                for (int i = 0; i < headers.size(); i++) {
                    Cell subCell = row.createCell(i);
                    subCell.setCellValue(getCellValue(order, i));
                    subCell.setCellStyle(contentStyle);
                }
                endRows.put(parentId, rowIndex - 1);
                totalAmount = totalAmount.add(order.getAllMoney());
            } else {
                // 当前行与上一行的订单编号不同,更新起始行和结束行
                if (startRows.containsKey(lastParentId)) {
                    int startRow = startRows.get(lastParentId);
                    int endRow = endRows.get(lastParentId);
                    if (endRow - startRow > 0) {
                        // 只有当有两行及以上相同的订单编号时才需要合并单元格
                        CellRangeAddress mergedRegion = new CellRangeAddress(startRow, endRow, 0, 0);
                        sheet.addMergedRegion(mergedRegion);
                    }
                }
                startRows.put(parentId, rowIndex);
                endRows.put(parentId, rowIndex);
                Row row = sheet.createRow(rowIndex++);
                for (int i = 0; i < headers.size(); i++) {
                    Cell subCell = row.createCell(i);
                    subCell.setCellValue(getCellValue(order, i));
                    subCell.setCellStyle(contentStyle);
                }
                totalAmount = totalAmount.add(order.getAllMoney());
            }
            lastParentId = parentId;
        }

        // 对于最后一个订单号,需要进行合并单元格操作
        if (startRows.containsKey(lastParentId)) {
            int startRow = startRows.get(lastParentId);
            int endRow = endRows.get(lastParentId);
            if (endRow - startRow > 0) {
                // 只有当有两行及以上相同的订单编号时才需要合并单元格
                CellRangeAddress mergedRegion = new CellRangeAddress(startRow, endRow, 0, 0);
                sheet.addMergedRegion(mergedRegion);
            }
        }

        // 创建总金额行
        Row totalRow = sheet.createRow(rowIndex++);
        List<String> totalData = Arrays.asList("合计", "", "", "", "", "", String.valueOf(totalAmount),"");
        for (int i = 0; i < totalData.size(); i++) {
            Cell cell = totalRow.createCell(i);
            cell.setCellValue(totalData.get(i));
            cell.setCellStyle(contentStyle);
        }
        

        // 添加备注行
        Row remarkRow = sheet.createRow(rowIndex++);
        remarkRow.setHeightInPoints(20);
        Cell remarkCell = remarkRow.createCell(0);
        remarkCell.setCellValue("备注:");
        sheet.addMergedRegion(new CellRangeAddress(rowIndex - 1, rowIndex - 1, 0, headers.size() - 1));

        // 添加友情提示行
        Row promptRow = sheet.createRow(rowIndex++);
        promptRow.setHeightInPoints(20);
        Cell promptCell = promptRow.createCell(0);
        promptCell.setCellValue("友情提示:1、商品请当场验收,如有问题可即时更换或拒收2、易变质食品请妥善保管。");
        sheet.addMergedRegion(new CellRangeAddress(rowIndex - 1, rowIndex - 1, 0, headers.size() - 1));

        // 添加审核人、出库人、送货人、签收人行
        Row personRow = sheet.createRow(rowIndex++);
        personRow.setHeightInPoints(20);
        List<String> personData = Arrays.asList("审核人:", "出库人:", "送货人:", "签收人:");
        for (int i = 0; i < personData.size(); i++) {
            Cell cell = personRow.createCell(i * 2);
            cell.setCellValue(personData.get(i));
            sheet.addMergedRegion(new CellRangeAddress(rowIndex - 1, rowIndex - 1, i * 2, i * 2 + 1));
        }


        // 设置列宽
        for (int i = 0; i < headers.size(); i++) {
            if(i==1){
                sheet.setColumnWidth(i,60 * 256);
            }else {
                sheet.setColumnWidth(i, 20 * 256);
            }
        }

        // 设置响应头,告诉浏览器该文件为Excel格式
        response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
        response.setHeader("Content-Disposition", "attachment; filename=\"orders.xlsx\"");

        // 将工作簿写入输出流,导出Excel文件
        OutputStream outputStream = response.getOutputStream();
        workbook.write(outputStream);
        workbook.close();
        outputStream.close();
    }

    private String getCellValue(BillProductOderVO order, int columnIndex) {
        switch (columnIndex) {
            case 0:
                return order.getOrderNo();
            case 1:
                return order.getProductName();
            case 2:
                return order.getSpec();
            case 3:
                return order.getUnit();
            case 4:
                return String.valueOf(order.getProductQuantity());
            case 5:
                return String.valueOf(order.getProductPrice());
            case 6:
                return String.valueOf(order.getAllMoney());
            default:
                return "";
        }
    }

不得不说,ChatGPT还是很强大的,这样的小功能如果你之前做过类似的,相信很快就能写出来,但是如果没有,一般人还是要花点时间去琢磨的。
ChatGPT的出现,真的很nice,不过所谓的程序员成功将自己卷没了这句话也并不正确,因为如果你根本是小白,连基本的代码都看不懂,哪怕给你ChatGPT也没法写功能。ChatGPT的关键就在于如何提问,对于一些小功能,使用它还是非常节省时间的。

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

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

相关文章

ChatGPT常见问题及其解决方法汇总

好久没有更新过技术类的文章了&#xff0c;希望本篇文章能够对你有所帮助&#xff0c;今天这篇博客将会把ChatGPT注册中可能遇到的问题彻头彻尾的讲一下&#xff0c;创作不易&#xff0c;如果感觉有帮助的话就动动你发财的小手点个收藏点个赞吧。如有需要转载请附上原文链接&am…

微软骚操作恶心Win10用户,上网得先看广告

IE 浏览器在几个月前被彻底禁用&#xff0c;预装了快30年的老古董也确实到了退役的时候。 而微软也早有准备&#xff0c;2015年随着 Win10 发布推出了 Microsoft Edge 浏览器。 2020年迁移到 Chromium 内核让其成为了主流浏览器之一。 和 Chromium 系其他浏览器一样支持扩展插…

Portraiture4最新版滤镜P图一键磨皮插件

今天coco玛奇朵给大家带来了一款ps磨皮插件&#xff0c;超级简单好用。Portraiture 滤镜是一款 Photoshop&#xff0c;Lightroom 和 Aperture 插件&#xff0c;DobeLighttroom 的 Portraiture 消除了选择性掩蔽和逐像素处理的繁琐的手工劳动&#xff0c;以帮助您在肖像修整方面…

TiDB实战篇-PD调度常见问题处理方法

常见的问题 调度产生和执行 常见的调度类型 参数调度的速度 调度典型场景 Leader分布不均匀监控 leader分布算法&#xff0c;每一个leader的size作为总和&#xff0c;还有TiKV的剩余空间等等。 可以手动设置权重。 分布不均衡处理 TiKV节点下线速度慢 TiKV下线速度慢解决方法 …

IntelliJ IDEA修改背景颜色大全(护眼绿等)设置注释颜色

一.IDEA默认有3种背景颜色 路径为File->settings->Editor->Color Scheme可以设置软件默认颜色&#xff0c;旁边的小齿轮添加颜色名字 二.IDEA扩展颜色&#xff08;护眼绿&#xff09; 第一种方法&#xff1a; IDEA设置一张背景图片,路径&#xff1a;File->Setti…

Mysql 中left join时 on、and、where区别

1、准备两张表student与class表 student class 2、left join on左连接 select * from student s left join class c on s.classId c.id 左表数据全部显示&#xff0c;关联到的右表数据显示&#xff0c;没有显示null 3、left join on ... and对左表student进行条件筛选 …

深入理解Java Class文件格式 constant_UTF_info

首先&#xff0c; 让我们回顾一下关于class文件格式的之前两篇博客的主要内容。 在 深入理解Java Class文件格式&#xff08;一&#xff09; 中&#xff0c; 讲解了class文件在整个java体系结构中的位置和作用&#xff0c; 讲解了class文件中的魔数和版本号相关的信息&#xff…

Postman+Java springboot演示 get post put delete请求并携带(路径 路径问号后 json 表单)参数形式

我们先创建一个java的springboot工程 在项目中 找到启动类的位置目录 在项目创建一个类 叫 user 我是想将 user 当做一个属性类的 按规范来讲 我们可以创建一个entity包 然后在下面去创建属性类 但这里 我们不想搞那么麻烦了 毕竟只是练习一下 然后 user参考代码如下 package…

Django框架004:orm对mysql的增删改查

大家好&#xff0c;我是csdn的博主&#xff1a;lqj_本人 这是我的个人博客主页&#xff1a; lqj_本人的博客_CSDN博客-微信小程序,前端,python领域博主lqj_本人擅长微信小程序,前端,python,等方面的知识https://blog.csdn.net/lbcyllqj?spm1011.2415.3001.5343哔哩哔哩欢迎关注…

力扣---LeetCode141/142. 环形链表 (I)和(II) (代码详解+流程图+数学逻辑拓展)

文章目录 前言141. 环形链表 I1.1 链接&#xff1a;1.2 思路&#xff1a;1.3 代码&#xff1a;快慢指针1.4 流程图&#xff1a; 142. 环形链表 II2.1 链接&#xff1a;2.2 思路&#xff1a;2.3 代码&#xff1a;2.4 流程图&#xff1a; 拓展问题及证明(面试常问)&#xff1a;3.…

线上问题-CPU使用频率飙升

描述 中午收到群内人员反馈环境访问速度慢。登录验证码打不开等问题。通过查看日志发现是kafka出现问题&#xff0c;无法处理消息。联系运维解决。在排查的过程中使用mobaXterm连接服务器。左下角看到CPU使用频率非常高。于是记录一下通过CPU查看程序占用情况分析问题。 过程 …

Ansys Lumerical | CMOS - 光学仿真方法

通过使用更小的像素尺寸和更大的填充因子&#xff0c;基于CMOS图像传感器像素的数码相机系统的成本正在降低。但是&#xff0c;只有在不牺牲图像质量的情况下&#xff0c;CMOS像素尺寸减小才是可以接受的。随着CMOS像素尺寸的不断减小&#xff0c;图像信噪比降低&#xff0c;相…

2.1 Linux命令行

系列文章目录 第1章 Linux Shell简介 第2章 Shell基础 <本章所在位置> 第3章 Bash Shell基础命令 第4章 Bash Shell命令进阶 第5章 Linux Shell深度理解 第6章 Linux环境变量 第7章 Linux文件权限 第8章 Linux文件系统的管理 第9章 Linux软件安装 第10章 Linux文本编辑器…

记录一次docker容器引起的时间相差8h的问题

一、背景 系统打印日志时间小8h&#xff0c;部分插入mysql的日期却大8h&#xff0c;简直诡异。 测试时间是上午10:05 经过排查&#xff0c;mysql设置的时区&#xff0c;链接url设置的时区都是ok的。而且有其他服务时间正常&#xff0c;故排除MySQL的问题。 二、排查 2.1 查…

聚焦丨酷雷曼荣列XRMA联盟成员单位

自“元宇宙”概念兴起之初&#xff0c;酷雷曼VR所属北京同创蓝天云科技有限公司就积极布局、探索和实践。2022年12月&#xff0c;酷雷曼VR成功加入虚拟现实与元宇宙产业联盟&#xff08;XRMA&#xff09;&#xff0c;正式被接纳为联盟成员单位&#xff0c;意味着酷雷曼公司将进…

【电动车】基于双层凸优化的燃料电池混合动力汽车研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️❤️&#x1f4a5;&#x1f4a5;&#x1f4a5;&#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清…

rosbag相关进阶操作

一些很好用的网站 时间戳在线转换网页 旋转矩阵、四元数、绕轴旋转、欧拉角在线转换网页 四元数、欧拉角可视化在线转换网页 一、按时间截取bag 使用如下代码&#xff1a; rosbag filter 原始包名.bag 截取后的包名.bag "t.to_sec() > 开始时间 and t.to_sec() <…

初识Elasticsearch

初识Elasticsearch Elasticsearch 是一个分布式&#xff0c;RESTful 风格的搜索和数据分析引擎。它能够提供实时的搜索与数据分析功能&#xff0c;且能够将几乎所有类型的数据存储和搜索&#xff0c;包括结构化和非结构化数据。 在该博文中&#xff0c;我们将介绍 Elasticsea…

在CentOS上安装Jenkins并配置Docker

文章目录 步骤1 - 安装Java 11步骤2 - 安装Jenkins步骤3 - 安装Docker步骤4 - 配置Docker Cloud步骤 5 - 验证步骤 6 - 可能会遇到的问题 在本教程中&#xff0c;我们将展示如何在CentOS上安装Jenkins和Docker&#xff0c;并将它们配置在同一台机器上&#xff0c;使Jenkins能够…

开发必备,开源 or 免费的 AI 编程助手

AI 大模型的火热&#xff0c;让开发圈近来如虎添翼&#xff0c;各种各样基于 AI 技术的开发者工具和新范式不断涌现&#xff0c;尤其是 Github 和 OpenAI 共同推出的 Copilot X &#xff0c;更是一骑绝尘。本文推荐一些开源 or 免费的 AI 编程工具&#xff0c;不妨试着用起来。…