在若依框架中,使用easyExcel完成动态列导出

一、表结构

例如,现在有两张数据库表:

sql 复制代码
CREATE TABLE `product` (
  `id` bigint NOT NULL AUTO_INCREMENT,
  `product_name` varchar(100) NOT NULL COMMENT '商品名称',
  `price` decimal(10,2) DEFAULT NULL COMMENT '价格',
  `category` varchar(50) DEFAULT NULL COMMENT '分类',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=16 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='商品主表';
sql 复制代码
CREATE TABLE `product_attr` (
  `id` bigint NOT NULL AUTO_INCREMENT,
  `product_id` bigint NOT NULL COMMENT '商品ID',
  `attr_name` varchar(50) NOT NULL COMMENT '属性名(动态列名)',
  `attr_value` varchar(500) DEFAULT NULL COMMENT '属性值(列对应的值)',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=24 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='商品动态属性表';

二、代码实现

1、列表查询代码

controller

java 复制代码
    @GetMapping("/list")
    public TableDataInfo list(Product product)
    {
        startPage();
        List<Product> list = productService.selectProductList(product);
        return getDataTable(list);
    }

impl+service

java 复制代码
/**
     * 查询商品主列表
     *
     * @param product 商品主
     * @return 商品主
     */
    @Override
    public List<Product> selectProductList(Product product) {
        // 1. 查询所有商品
        List<Product> productList = productMapper.selectProductList(product);

        // 2. 遍历每个商品,给它设置动态属性
        for (Product p : productList) {
            // 查询当前商品对应的所有属性
            ProductAttr productAttr = new ProductAttr();
            productAttr.setProductId(p.getId());
            List<ProductAttr> attrList = productAttrMapper.selectProductAttrList(productAttr);

            Map<String, String> extMap = new HashMap<>();
            // 循环属性列表,一个个放进map里
            for (ProductAttr attr : attrList) {
                String attrName = attr.getAttrName();   // 颜色/尺寸/功率
                String attrValue = attr.getAttrValue(); // 黑色/XL/500W
                extMap.put(attrName, attrValue);
            }

            // 把Map设置到商品里
            p.setExtMap(extMap);
        }

        return productList;
    }

/**
     * 查询商品主列表
     * 
     * @param product 商品主
     * @return 商品主集合
     */
    public List<Product> selectProductList(Product product);

主表.xml

java 复制代码
 /**
     * 查询商品主列表
     * 
     * @param product 商品主
     * @return 商品主集合
     */
    public List<Product> selectProductList(Product product);

  <sql id="selectProductVo">
        select id, product_name, price, category from product
    </sql>

    <select id="selectProductList" parameterType="Product" resultMap="ProductResult">
        <include refid="selectProductVo"/>
        <where>  
            <if test="productName != null  and productName != ''"> and product_name like concat('%', #{productName}, '%')</if>
            <if test="price != null "> and price = #{price}</if>
            <if test="category != null  and category != ''"> and category = #{category}</if>
        </where>
    </select>

子表xml

java 复制代码
  /**
     * 查询商品动态属性列表
     * 
     * @param productAttr 商品动态属性
     * @return 商品动态属性集合
     */
    public List<ProductAttr> selectProductAttrList(ProductAttr productAttr);

   <sql id="selectProductAttrVo">
        select id, product_id, attr_name, attr_value from product_attr
    </sql>

    <select id="selectProductAttrList" parameterType="ProductAttr" resultMap="ProductAttrResult">
        <include refid="selectProductAttrVo"/>
        <where>  
            <if test="productId != null "> and product_id = #{productId}</if>
            <if test="attrName != null  and attrName != ''"> and attr_name like concat('%', #{attrName}, '%')</if>
            <if test="attrValue != null  and attrValue != ''"> and attr_value = #{attrValue}</if>
        </where>
    </select>

2、导出代码实现

导入依赖:


controller

java 复制代码
   @GetMapping("/export")
    public void export(Product product, HttpServletResponse response) {
        try {
            productService.exportExcel(response, product);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

impl

java 复制代码
    
    public void exportExcel(HttpServletResponse response, Product product) throws IOException {
        List<Product> list = selectProductList(product);

        // ========== 第一步:获取所有动态列名(颜色、尺寸、功率...)==========
        Set<String> allAttrNames = new LinkedHashSet<>();
        for (Product p : list) {
            if (p.getExtMap() != null) {
                allAttrNames.addAll(p.getExtMap().keySet());
            }
        }
        List<String> dynamicHeaders = new ArrayList<>(allAttrNames);

        // ========== 第二步:构建 Excel 表头 ==========
        List<List<String>> head = new ArrayList<>();
        // 固定列
        head.add(Collections.singletonList("商品ID"));
        head.add(Collections.singletonList("商品名称"));
        head.add(Collections.singletonList("价格"));
        head.add(Collections.singletonList("分类"));
        // 动态列
        for (String attrName : dynamicHeaders) {
            head.add(Collections.singletonList(attrName));
        }

        // ========== 第三步:构建 Excel 行数据 ==========
        List<List<Object>> dataList = new ArrayList<>();
        for (Product p : list) {
            List<Object> row = new ArrayList<>();
            // 固定数据
            row.add(p.getId());
            row.add(p.getProductName());
            row.add(p.getPrice());
            row.add(p.getCategory());
            // 动态数据(按表头顺序)
            Map<String, String> ext = p.getExtMap() == null ? new HashMap<>() : p.getExtMap();
            for (String key : dynamicHeaders) {
                row.add(ext.getOrDefault(key, ""));
            }
            dataList.add(row);
        }

        // ========== 第四步:EasyExcel 写出 ==========
        response.setContentType("application/vnd.openxmlformats");
        response.setCharacterEncoding("utf-8");
        response.setHeader("Content-Disposition", "attachment;filename=product.xlsx");

        EasyExcel.write(response.getOutputStream())
                .head(head)
                .sheet("商品数据")
                .doWrite(dataList);
    }
相关推荐
碳基硅坊5 小时前
Spring AI:把大模型接进 Spring 应用
java·人工智能·spring ai
黄毛火烧雪下5 小时前
Java 核心知识点总结(一)
java·开发语言
技术小结-李爽6 小时前
【工具】Maven的下载、安装、使用
java·maven
极创信息6 小时前
Linux挖矿病毒深度清理实战教程,从进程隐藏、Rootkit驻留到彻底根除
java·大数据·linux·运维·安全·tomcat·健康医疗
努力成为AK大王6 小时前
并发编程的核心挑战、优化方案与核心知识点总结
java·开发语言·数据库
云烟成雨TD6 小时前
Agent Scope Java 2.x 系列【10】技能(Skill)
java·人工智能·agent
摇滚侠6 小时前
SpringMVC 入门到实战 DispatcherServlet 源码解读 92-95
java·后端·spring·maven·intellij-idea
键盘歌唱家6 小时前
Spring AI 入门分享:它和“直接调 API“到底差在哪
java·人工智能·spring
宸丶一7 小时前
Day 10:LangGraph - Agent 的图执行引擎
java·windows·python
hikktn7 小时前
Excel 导出 OOM 预防实战:30 万行从堆溢出到 50MB 的演进
java·excel·easyexcel