帆软(FineReport)程序集动态获得表列名(应该是全网免费首发)

本文描述了作者在使用帆软报表时,如何通过Java代码解决复杂数据获取的问题,包括从数据库动态获取列名和处理表结构变化带来的挑战。作者分享了自定义的代码示例以供参考。

1.业务场景

帆软报表,有些复杂得数据需要使用到下面的程序(java代码)获得数据集,来实现复杂的数据展示(不能通过sql语句展示出来的业务数据),本人是图片问题

2 官方文档 

这里不得不吐槽一下帆软官方文档上的demo过于简单;

数据获得

example/SimpleParamTableDataDemo.java at release/10.0 - example - 帆软第三方插件仓库

 列名获得

example/SimpleArrayTableDataDemo.java at release/10.0 - example - 帆软第三方插件仓库

 

 按照官方demo就这样写死的,
要是表结构改变(增加字段),就需要每次手动去在这个String[]增加之后,在打包部署,非常麻烦

3.咨询客服,也是仅供参考,具体自己写逻辑,不提供业务技术支持

 

        7月份最初版就是写死的,10月份在不忙的时候自己研究,(一个月的成果啊~,改的至少40次,都快吐槽.因为每次打包部署), 多的不说直接上代码

/** 可以自己拿过去使用 有注解应该能看懂,,多多点赞支持分享 */
public class ryyV8Demo15Day02 extends SimpleTableData {

    Connection con = null;

    //定义的是类的成员变量;初始化方式不对 ,就会报错;
//    private ArrayList<Object[]> valueList = null; //方式一
//    private ArrayList<Object[]> valueList ;  //方式二

    //定义的是类的成员变量;初始化一定要new (可以不使用)
    private ArrayList<Object[]> valueList = new ArrayList<>();

    private static  String[] nameList = null;
    private static  String id ;


    /** 本地main方法测试  打开*/
//    public static void main(String[] args) {
//
//        ryyV8Demo15Day02 demo15Day02 = new ryyV8Demo15Day02();
//        List<Object[]> list = demo15Day02.loadData();
//        for (int i = 0; i < demo15Day02.nameList.length; i++) {
////            System.out.println(nameList[i]);
//            System.out.println(nameList[i] +": "+list.get(0)[i]);
//        }
//    }

    /**
     * 获取列名(动态)
     * @param
     * @return
     * @version 3.0
     * @date 2023/10/15
     */
    public final void getColumnNames (){
        FineLoggerFactory.getLogger().info("nameList开始");
        System.out.println("nameList开始");
        if (nameList != null){
            System.out.println("nameList已经初始化有值");
             return;
        }
        System.out.println("提示nameList空值");
        //这里我固定写死参数 000
        String sql = "SELECT * FROM   数据库表名 A WHERE (1=1) and ID = '000'";// + tableName;
        FineLoggerFactory.getLogger().info("Query SQL of ParamTableDataDemo: \n" + sql);


        // 下面开始建立数据库连接,按照刚才的SQL语句进行查询
        // 根据连接名获取FR数据连接定义的数据连接,如果没有定义,也可以参考getConnection方法自己创建连接

        // 用String[]保存 k 初始化
//        String[] ColumnNames = null;
        try {
            con = getConnection();
            Statement stmt = con.createStatement();
            ResultSet rs = stmt.executeQuery(sql);
            // 获得记录的详细信息,然后获得总列数
            ResultSetMetaData rsmd = rs.getMetaData();
            System.out.println("获得记录的详细信息");
//            System.out.println(rs.next());
            int colNum = rsmd.getColumnCount();
            FineLoggerFactory.getLogger().info("rsmd.getColumnCount():k的数量:" + colNum);

//            rs.next();
            // 列名 赋值 String[]
            nameList = new String[colNum];
            for (int i = 0; i < colNum; i++) {
                nameList[i] = rsmd.getColumnName(i + 1);
            }
            // 释放数据库资源
            rs.close();
            stmt.close();
        } catch (Exception e) {
            FineLoggerFactory.getLogger().error(e.getMessage(), e);
        }
        System.out.println("初始化类加载的时候运行--获取列名方法(getColumnNames)结束-------"+nameList.length);
    }


    /**
     * 程序集的列名(K)
     *
     * @param
     * @return
     * @version 2.0
     * @date 2023/9/18
     */
    @Override
    public String[] initColumnNames() {
        FineLoggerFactory.getLogger().info("demo02-开始--列名15ryy");
        getColumnNames();
    
        return nameList;
    }


    @Override
    public List<Object[]> loadData() {
        id = ((ParameterProvider) (parameters.get().toArray())[0]).getValue().toString();
        //测试使用 打开
//        id = "08b2e6cbcf1e429baf30fff6fab7ccf2";
        FineLoggerFactory.getLogger().info("loadData开始--首先查看参数"+id);

        //获取图片 自己的业务代码
        HashMap<String, Object> mp = getPicInfo(id);
        List<Object[]> list = assembleData(mp, id);
        try {
            if (con != null) {
                con.close();
            }
        } catch (SQLException e) {
            FineLoggerFactory.getLogger().error(e.getMessage(), e);
        }
        return list;
    }

    /**
     * 获取数据库连接 driverName和 url 可以换成您需要的
     *
     * @return Connection
     */
    private Connection getConnection() throws Exception {

        com.fr.data.impl.Connection conn = ConnectionConfig.getInstance().getConnection("帆软编辑器自自己绑定的数据库命名"); //参考上面官方文档
        con = conn.createConnection();

        //本地连接测试打开
//        String driverName = "oracle.jdbc.driver.OracleDriver";
////        String url = "自己写"; //测试
//        String username = "111";
//        String password = "123456";
//        Connection con = null;
//        try {
//            Class.forName(driverName);
//            con = DriverManager.getConnection(url, username, password);
//        } catch (Exception e) {
//            FineLoggerFactory.getLogger().error(e.getMessage(), e);
//        }

        return con;
    }

}

小弟第二次分享文章,多多点赞支持分享.

ps:有注解应该能看懂,不懂得可以留言,可能遇到部署问题,加油小伙伴们,

### 如何在 FineReport 中设置条件属性动态列 #### 1. 条件属性的概念 在 FineReport设计中,条件属性是一种用于控制单元格显示样式的机制。它可以根据特定的逻辑达式改变单元格的颜色、字体或其他样式[^1]。 #### 2. 动态列的基础概念 动态列是指报中的某些列会根据用户的输入或者后台数据的变化而自动调整其数量或内容。通常可以通过复选框控件或者其他交互方式实现动态列的选择和展示[^2]。 #### 3. 设置条件属性动态列的具体方法 以下是具体的操作流程: ##### (1)创建复选框控件并配置数据字典 在报设计器界面中新增一个 **复选框控件** `clos`,为其设置数据字典。此步骤的关键在于定义页面显示值与实际对应值的关系,这些值应来源于数据库(如 `tb_user`),并与后续使用的变量 `${cols}` 的值保持一致[^1]。 ##### (2)编写模板脚本以处理动态列 为了使动态列生效,需在报模板中加入相应的脚本代码。以下是一个简单的示例代码片段,展示了如何通过 Java 脚本来动态生成列结构: ```java String[] cols = parameter.clos.split(","); // 将复选框返回的结果按逗号分割成数组 for (int i = 0; i < cols.length; i++) { String colName = cols[i]; report.setCellText(1, i + 1, "${" + colName + "}"); // 假设第1行为标题行 } ``` ##### (3)应用条件属性到动态列 进入单元格的“条件属性”设置窗口,添加新的条件规则。例如,当某一列数值大于某个阈值时更改背景颜色为红色;反之则保留默认样式。具体的判断语句可以写成类似于 `"${cellValue} > threshold"` 这样的形式[^1]。 ##### (4)测试与优化 完成以上操作之后,请务必保存所做的修改,并利用内置的预览功能验证最终的效果是否满足预期。如果发现任何问题,则及时回溯至相应环节进行修正[^1]。 #### 4. 高级技巧——结合父子格实现更复杂的布局 如果有涉及层级关系的数据需要呈现出来的话,比如按照地区划分销售人员名单的情况,那么除了单纯依靠条件属性外还可以考虑引入父格子与子格子相结合的方式来增强视觉现力[^3]。 ```python # 示例 Python 版本伪代码示如何遍历树状结构填充格 def fill_table(data_list): row_index = 0 for region_data in data_list: table.addCell(region_data['region'], row_index, 0) # 插入区域名作为首列 salesmen_count = len(region_data['salesmen']) rowspan_value = max(salesmen_count, 1) if salesmen_count > 0: table.mergeCells(row_index, 0, rowspan_value, 1) # 合并相同地区的多行 current_row = row_index for salesman_name in region_data['salesmen']: table.addCell(salesman_name, current_row, 1) current_row += 1 row_index += rowspan_value ```
评论 3
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值