在Java里实现基础报表生成功能_Java字符串拼接项目说明
发布时间 - 2025-12-30 00:00:00 点击率:次用 StringBuilder 拼接 HTML 表格最稳,需预估容量、分离固定结构、仅最后调用 toString();用户输入须 HTML 转义防 XSS;金额用 DecimalFormat 显式指定 US Locale 格式化;CSV 导出应使用 OpenCSV 或 Commons CSV 库并添加 UTF-8 BOM。
用 StringBuilder 拼接报表 HTML 表格最稳
直接用 + 拼接多行 HTML 报表字符串,在循环中极易触发频繁对象创建,导致 GC 压力大、生成慢。尤其当数据量超 500 行时,String 拼接耗时可能翻倍。
实操建议:
- 初始化
StringBuilder时预估容量,比如每行约 120 字符 × 1000 行 → 设为new StringBuilder(120_000) - 表格头、分页信息、合计行等固定结构,单独拼一次,别塞进主循环
- 避免在循环内调用
toString(),只在最后生成完成时调用一次
StringBuilder report = new StringBuilder(120_000);
report.append("| ID | 姓名 | 金额 |
|---|---|---|
| ") .append(order.getId()) .append(" | ") .append(escapeHtml(order.getName())) // 防 XSS,见下节 .append(" | ") .append(String.format("%.2f", order.getAmount())) .append(" |
必须对用户输入做 HTML 转义,否则会崩报表页面
报表字段若含 、&、" 等字符(比如姓名是 "张三"),不转义会导致 HTML 结构错乱,甚至执行脚本。
不要自己写正则替换 —— 容易漏掉 ' 或 Unicode 变体。用成熟工具更可靠:
- JDK 9+ 可用
StringEscapeUtils.escapeHtml4()(来自 Apache Commons Text) - 若项目已引入 Spring,优先用
org.springframework.web.util.HtmlUtils.htmlEscape() - 纯 JDK 方案:手写一个最小化方法,只处理 4 个关键字符,比正则快且确定
private static String escapeHtml(String s) {
if (s == null) return "";
return s.replace("&", "&")
.replace("<", "zuojiankuohaophpcn")
.replace(">", "youjiankuohaophpcn")
.replace("\"", """);
}
DecimalFormat 格式化金额比 String.format 更可控
报表里金额字段要求统一显示为“千分位 + 两位小数”,比如 1234567.89 → 1,234,567.89。用 String.format("%,.2f", d) 看似简单,但受 Locale 影响大:在德国环境会输出 1.234.567,89,和中文报表预期不符。
实操建议:
- 显式指定
Locale.US,或直接用DecimalFormat控制符号 - 复用
DecimalFormat实例(线程安全),避免每次 new - 注意
Double.NaN或Infinity会抛异常,需提前判空
private static final DecimalFormat AMOUNT_FORMAT = new DecimalFormat("#,##0.00");
static {
AMOUNT_FORMAT.setDecimalFormatSymbols(new DecimalFormatSymbols(Locale.US));
}
// 使用时:
String amtStr = AMOUNT_FORMAT.format(order.getAmount());
导出 CSV 时别用字符串拼接,用 OpenCSV 或 Apache Commons CSV
报表常需同时支持 HTML 和 CSV 导出。用 StringBuilder 手拼 CSV 很危险:字段含逗号、换行、双引号时,必须按 RFC 4180 规则加引号并转义,手写极易出错(比如漏转义嵌套双引号 "a""b")。
直接上轻量库更省心:
- OpenCSV:API 直观,
CsvWriter支持自动转义,适合单线程导出 - Apache Commons CSV:更现代,支持流式写入、自定义分隔符,且
QuoteMode.ALL_NON_NULL可控性更强 - 禁用
BufferedWriter.write(line + "\n")这类裸写法 —— 换行符在 Windows/Linux 下不一致,库会自动适配
容易被忽略的一点:CSV 文件的 BOM 头。中文 Excel 打开乱码,大概率是因为没加 \uFEFF。用 Commons CSV 时需手动在输出流开头写入:
try (OutputStream os = response.getOutputStream()) {
os.write(0xEF); os.write(0xBB); os.write(0xBF); // UTF-8 BOM
try (CSVPrinter printer = new CSVPrinter(new OutputStreamWriter(os, StandardCharsets.UTF_8),
CSVForma
t.DEFAULT.withQuoteMode(QuoteMode.ALL_NON_NULL))) {
printer.printRecord("ID", "姓名", "金额");
for (Order o : orders) {
printer.printRecord(o.getId(), o.getName(), o.getAmount());
}
}
}
# linux
# excel
# java
# html
# windows
# apache
# app
# 工具
# csv
# win
# stream
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
如何用PHP工具快速搭建高效网站?
如何在Windows环境下新建FTP站点并设置权限?
如何在沈阳梯子盘古建站优化SEO排名与功能模块?
高配服务器限时抢购:企业级配置与回收服务一站式优惠方案
jQuery 常见小例汇总
Laravel怎么调用外部API_Laravel Http Client客户端使用
Laravel Seeder填充数据教程_Laravel模型工厂Factory使用
装修招标网站设计制作流程,装修招标流程?
中山网站推广排名,中山信息港登录入口?
Laravel怎么实现一对多关联查询_Laravel Eloquent模型关系定义与预加载【实战】
网站制作软件有哪些,制图软件有哪些?
Android滚轮选择时间控件使用详解
晋江文学城电脑版官网 晋江文学城网页版直接进入
购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?
javascript基本数据类型及类型检测常用方法小结
专业型网站制作公司有哪些,我设计专业的,谁给推荐几个设计师兼职类的网站?
青岛网站建设如何选择本地服务器?
如何在浏览器中启用Flash_2025年继续使用Flash Player的方法【过时】
iOS UIView常见属性方法小结
小视频制作网站有哪些,有什么看国内小视频的网站,求推荐?
Laravel怎么实现搜索功能_Laravel使用Eloquent实现模糊查询与多条件搜索【实例】
Laravel如何实现数据导出到PDF_Laravel使用snappy生成网页快照PDF【方案】
Laravel如何生成和使用数据填充?(Seeder和Factory示例)
绝密ChatGPT指令:手把手教你生成HR无法拒绝的求职信
Laravel怎么实现前端Toast弹窗提示_Laravel Session闪存数据Flash传递给前端【方法】
HTML5建模怎么导出为FBX格式_FBX格式兼容性及导出步骤【指南】
高端建站三要素:定制模板、企业官网与响应式设计优化
如何基于PHP生成高效IDC网络公司建站源码?
千问怎样用提示词获取健康建议_千问健康类提示词注意事项【指南】
Windows家庭版如何开启组策略(gpedit.msc)?(安装方法)
如何在橙子建站中快速调整背景颜色?
mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?
linux写shell需要注意的问题(必看)
js实现获取鼠标当前的位置
如何登录建站主机?访问步骤全解析
Laravel观察者模式如何使用_Laravel Model Observer配置
米侠浏览器网页图片不显示怎么办 米侠图片加载修复
如何在IIS服务器上快速部署高效网站?
Laravel怎么上传文件_Laravel图片上传及存储配置
如何在服务器上配置二级域名建站?
node.js报错:Cannot find module 'ejs'的解决办法
如何在阿里云购买域名并搭建网站?
如何使用 jQuery 正确渲染 Instagram 风格的标签列表
长沙企业网站制作哪家好,长沙水业集团官方网站?
如何为不同团队 ID 动态生成多个非值班状态按钮
Android实现代码画虚线边框背景效果
详解MySQL数据库的安装与密码配置
网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?
*服务器网站为何频现安全漏洞?
奇安信“盘古石”团队突破 iOS 26.1 提权


t.DEFAULT.withQuoteMode(QuoteMode.ALL_NON_NULL))) {
printer.printRecord("ID", "姓名", "金额");
for (Order o : orders) {
printer.printRecord(o.getId(), o.getName(), o.getAmount());
}
}
}