Java实现PDF模板填充的详细技术方案

在java中实现pdf模板填充的核心是使用openpdf或apache pdfbox等库操作acroforms。1. 引入openpdf依赖;2. 加载pdf模板;3. 获取并填充表单字段;4. 可选扁平化处理;5. 保存生成的新pdf。处理中文需嵌入字体,如使用basefont.createfont加载simsun.ttc并设置编码为identity_h和embedded。复杂字段如复选框、单选框、下拉列表等需按字段导出值准确填充。批量生成时应确保资源及时释放、避免重复加载模板、合理管理内存、适当使用多线程,并做好错误处理与日志记录,以提升性能和稳定性。

Java实现PDF模板填充的详细技术方案

在Java里实现PDF模板填充,核心要点就是利用成熟的库去识别并操作PDF文档中预设的表单字段(也就是AcroForms)。简单来说,就是你先准备一个带表单的PDF文件作为模板,然后用代码把数据“填”进去,最后保存成一份新的、数据完整的PDF。目前,业界比较常用且功能强大的库主要是OpenPDF(它是iText的开源分支,解决了iText的商业许可问题)和Apache PDFBox。它们都提供了相当全面的API来处理这类任务,让你能高效地把业务数据和PDF文档结合起来。

Java实现PDF模板填充的详细技术方案

解决方案

要用Java实现PDF模板填充,我们通常会选择OpenPDF或Apache PDFBox。这里以OpenPDF为例,它在处理AcroForms方面表现得相当出色,用起来也比较直观。

基本流程是这样的:

立即学习“Java免费学习笔记(深入)”;

Java实现PDF模板填充的详细技术方案

引入依赖: 在你的pom.xml(如果你用Maven的话)里加上OpenPDF的依赖。

    com.github.librepdf    openpdf    1.3.30 

加载PDF模板:PdfReader读取你的PDF模板文件。

Java实现PDF模板填充的详细技术方案

获取AcroForm: 通过PdfStamper获取PDF的表单对象(AcroFields)。

填充字段: 根据表单字段的名称,调用setField()方法填充数据。

扁平化(可选但推荐): 调用setFormFlattening(true)将表单字段“固化”到PDF内容中,这样用户就不能再编辑了,也能减少文件大小。

保存新PDF: 将填充后的PDF写入一个新的文件流。

这里是一个简单的代码示例:

import com.lowagie.text.pdf.AcroFields;import com.lowagie.text.pdf.PdfReader;import com.lowagie.text.pdf.PdfStamper;import com.lowagie.text.pdf.BaseFont; // 用于处理字体,尤其是中文import java.io.FileOutputStream;import java.io.IOException;import java.util.Map;import java.util.Set;public class PdfFormFiller {    public static void fillPdfTemplate(String templatePath, String outputPath, Map formData) throws IOException {        PdfReader reader = null;        PdfStamper stamper = null;        try {            reader = new PdfReader(templatePath);            // 创建一个PdfStamper,用于修改PDF。第二个参数是输出流。            stamper = new PdfStamper(reader, new FileOutputStream(outputPath));            AcroFields form = stamper.getAcroFields();            // 注册字体,这对于处理中文非常重要            // 通常需要引入一个支持中文的字体文件,比如Windows系统自带的simsun.ttc            // 如果你的服务器是Linux,需要自己上传字体文件到服务器的某个路径            BaseFont bf = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);            form.addSubstitutionFont(bf); // 备用字体,确保中文显示            // 遍历表单数据,填充字段            for (Map.Entry entry : formData.entrySet()) {                String fieldName = entry.getKey();                String fieldValue = entry.getValue();                // 检查字段是否存在                if (form.getField(fieldName) != null) {                    form.setField(fieldName, fieldValue);                    System.out.println("填充字段: " + fieldName + " -> " + fieldValue);                } else {                    System.err.println("警告: 模板中不存在字段 '" + fieldName + "'");                }            }            // 扁平化表单,使其内容成为PDF的一部分,不可编辑            stamper.setFormFlattening(true);            System.out.println("PDF模板填充完成,并已扁平化。");        } catch (Exception e) {            System.err.println("PDF填充过程中发生错误: " + e.getMessage());            e.printStackTrace();            throw new IOException("PDF填充失败", e);        } finally {            if (stamper != null) {                try {                    stamper.close(); // 关闭stamper会同时关闭reader                } catch (Exception e) {                    System.err.println("关闭PdfStamper失败: " + e.getMessage());                }            }            // reader在stamper关闭时会自动关闭,无需单独关闭        }    }    public static void main(String[] args) {        String template = "path/to/your/template.pdf"; // 你的PDF模板路径        String output = "path/to/output/filled_document.pdf"; // 输出文件路径        // 模拟要填充的数据        Map data = new java.util.HashMap();        data.put("nameField", "张三"); // 假设模板里有个叫 "nameField" 的文本框        data.put("ageField", "30");    // 假设有个叫 "ageField" 的文本框        data.put("genderRadio", "男"); // 假设有个单选框组叫 "genderRadio",值为 "男"        data.put("checkboxOption", "Yes"); // 假设有个复选框叫 "checkboxOption",值为 "Yes" (或 "Off" for uncheck)        data.put("descriptionArea", "这是一段很长的描述文本,用于测试多行文本区域的填充效果。确保内容能够正确地换行和显示。");        try {            fillPdfTemplate(template, output, data);            System.out.println("PDF已成功生成到: " + output);        } catch (IOException e) {            System.err.println("生成PDF失败: " + e.getMessage());        }    }}

重要提示BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED); 这行代码中的STSong-Light是一个iText内置的字体,但它并不嵌入字体本身,只是一个映射。如果你的PDF阅读器没有对应的字体,可能会显示乱码。更稳妥的做法是指定一个实际的字体文件路径,比如BaseFont.createFont("/path/to/simsun.ttc", BaseFont.IDENTITY_H, BaseFont.EMBEDDED);,这样字体会被嵌入到PDF中,确保在任何环境下都能正确显示。

PDF模板填充,究竟有哪些技术选型?

说到PDF模板填充,Java生态里可选择的方案其实不算少,但真正好用、成熟的,主要还是那几个。我个人觉得,这块的选择主要取决于你的具体需求、预算以及对开源社区的依赖程度。

首先是 OpenPDF。它实际上是iText 2.1.7的开源分支,继承了iText早期版本的强大功能,尤其是对AcroForms的支持相当到位。它的优点是:功能成熟稳定,API设计合理,社区活跃度也不错,而且最关键的是,它完全免费且基于LGPLv3许可,对于大多数商业项目来说,这是个非常稳妥且无许可风险的选择。如果你主要处理PDF表单的填充、合并、拆分等,OpenPDF通常是我的首选。

接着是 Apache PDFBox。这是一个由Apache基金会维护的开源项目,功能非常全面,远不止PDF填充。它能做文本提取、图像处理、PDF渲染、数字签名等等。PDFBox的优点在于其Apache许可证,非常宽松,而且社区庞大,文档丰富。不过,在处理AcroForms的某些复杂场景下,我个人感觉它的API可能不如OpenPDF那么直观,或者说,需要更多的代码来达到同样的效果。但如果你需要一个全能型的PDF处理库,PDFBox绝对值得考虑。

当然,还有 iText 本身。iText现在已经发展到很高版本,功能非常强大,支持XFA表单、数字签名、PDF/A归档等高级特性。但是,iText从版本5开始转向AGPLv3许可,这意味着如果你在商业应用中使用它(特别是作为网络服务提供给用户),可能需要购买商业许可证,否则就必须开源你的整个应用代码。这是一个非常重要的考量点,很多公司因此选择避开iText的最新版本,转而使用OpenPDF或PDFBox。

此外,市面上也有一些商业的PDF SDK,比如Aspose.PDF for Java、Docmosis等。它们通常提供更高级的功能、更好的技术支持,以及可能更简单的API,但代价就是高昂的授权费用。对于大多数内部系统或预算有限的项目,开源方案通常是更优的选择。

总结一下,如果你的核心需求是PDF表单填充,并且希望避免商业许可问题,OpenPDF无疑是性价比最高的选择。如果你需要更广泛的PDF处理能力,并且不介意API可能略显繁琐,Apache PDFBox则是一个强大的替代品。

如何处理PDF模板中的复杂字段类型和中文乱码问题?

PDF模板填充过程中,处理复杂字段和中文乱码是两个非常常见的“坑”。说实话,一开始踩到这些问题的时候,确实会让人有点抓狂,但好在都有成熟的解决方案。

复杂字段类型处理

PDF表单字段不仅仅是简单的文本框。你可能会遇到:

复选框(Checkbox)和单选框(Radio Button):

复选框: 通常,复选框的值不是true/false,而是Yes/Off或者1/0之类的字符串。你需要知道模板中该复选框被选中时的具体值。比如,如果模板里选中是Yes,那么form.setField("checkboxName", "Yes");就是选中,form.setField("checkboxName", "Off");就是取消选中。单选框: 单选框组中的每个选项都有一个相同的字段名,但每个选项有不同的导出值。当你设置这个字段的值时,实际上是选择了该组中对应导出值的那个单选框。比如,一个名为gender的单选框组,可能有MaleFemale两个选项,分别对应导出值MF。那么form.setField("gender", "M");就会选中“Male”选项。

下拉列表(Dropdown/Combobox):

下拉列表的值通常是你希望显示的文本,但有时也可能是其内部的导出值。你需要根据模板的具体设置来填充。setField()方法通常会尝试匹配文本或导出值。

多行文本区域(Multi-line Text Area):

setField()可以直接填充多行文本。PDF库会自动处理文本的换行和溢出。不过,如果文本过长,超出字段区域,多余的部分就会被截断或隐藏,这是需要注意的。所以,在设计模板时,最好预留足够的空间。

日期字段:

PDF的日期字段通常是文本框,你只需要按照特定格式(比如YYYY-MM-DD)填充字符串即可。

处理这些复杂字段的关键在于:了解你的PDF模板。你可以用Adobe Acrobat Pro或其他PDF编辑器打开模板,查看每个字段的属性(特别是名称、类型和导出值),这能帮助你准确地进行编程填充。

中文乱码问题

中文乱码是PDF处理中一个老大难问题,主要原因在于PDF文件本身需要嵌入或引用正确的字体来显示非ASCII字符。如果PDF阅读器没有对应的字体,或者PDF中没有嵌入字体信息,就会出现乱码。

解决方案通常是:

嵌入字体(推荐): 这是最稳妥的方法。通过BaseFont.createFont()方法加载一个支持中文的TrueType字体(.ttf.ttc)文件,并将其嵌入到PDF中。

// 假设simsun.ttc字体文件在你的项目资源路径下或者某个已知路径// 如果是Linux服务器,确保字体文件已上传String fontPath = "path/to/fonts/simsun.ttc"; // 或者其他中文字体,如微软雅黑等BaseFont bf = BaseFont.createFont(fontPath, BaseFont.IDENTITY_H, BaseFont.EMBEDDED);form.setDefaultInsertFont(bf); // 设置默认插入字体form.addSubstitutionFont(bf); // 添加为备用字体,确保旧字段也能用

fontPath:字体文件的绝对路径或相对路径。BaseFont.IDENTITY_H:这是处理中文(或其他Unicode字符)的关键编码,它表示水平写入的Unicode字符。BaseFont.EMBEDDED:这个参数至关重要,它告诉PDF库将字体文件数据嵌入到生成的PDF中。这样,无论用户在哪个系统上打开PDF,只要PDF阅读器支持,中文就能正常显示,而不会依赖用户系统是否安装了该字体。

使用iText内置的CJK字体(不推荐用于生产环境):STSong-LightMHei-Light这些是iText/OpenPDF内置的CID-keyed字体,它们不嵌入字体本身,而是依赖阅读器系统中的字体映射。虽然在某些环境下能显示中文,但兼容性较差,如果阅读器没有对应的字体,就可能显示乱码。所以,除非你对目标环境有严格控制,否则不建议在生产环境中使用这种方式。

确保PDF模板本身已嵌入字体:如果你的PDF模板本身在设计时就考虑了中文,并且已经嵌入了中文字体,那么在填充时,通常不需要额外处理字体,库会沿用模板的字体信息。但这种情况比较少见,因为为了减小模板文件大小,设计者往往不会在模板中嵌入完整的字体。

我个人在实际项目中,每次遇到中文乱码,几乎都是字体没有正确嵌入或者编码设置不对。所以,只要把字体文件路径、编码(IDENTITY_H)和嵌入(EMBEDDED)这三点搞定,中文乱码问题基本就能迎刃而解了。

批量生成PDF时,性能和资源消耗该如何优化?

当需要批量生成PDF文件时,性能和资源消耗就成了绕不开的话题。单个PDF的生成可能感觉不到什么,但一旦数量达到几百、几千甚至上万,不当的处理方式很容易导致内存溢出(OOM)或者CPU飙升,让整个服务变得不稳定。我记得有一次,因为没有注意资源释放,导致一个定时任务跑了几千份PDF后直接把服务器内存吃光了,那次排查真是让人头疼。

这里有几个我总结的优化点:

资源及时关闭与释放: 这是最最基础也最重要的。每次生成一个PDF,都会涉及到文件流的读写。PdfReaderPdfStamperFileOutputStream等对象使用完毕后,一定要在finally块中确保它们被正确关闭。PdfStamperclose()方法通常会负责关闭其内部的PdfReader和输出流,所以你只需要确保stamper.close()被调用即可。如果忘记关闭,文件句柄会持续占用,内存也会不断累积,最终导致资源耗尽。

// 示例中已包含finally块关闭资源,这是最佳实践// ...finally {    if (stamper != null) {        try {            stamper.close(); // 确保关闭        } catch (Exception e) {            // 记录日志,但不要阻止程序继续        }    }}

避免重复加载模板: 如果你的模板文件是固定的,并且每次填充的数据不同,理论上你可以尝试复用PdfReader对象。但实际操作中,PdfStamper在关闭时通常会关闭它所关联的PdfReader,所以每次生成一份新的PDF,重新创建PdfReaderPdfStamper实例反而是更安全、更常见的做法。因为PdfReader内部可能持有文件句柄和缓存,强制复用可能导致状态混乱或资源泄露。对于批量处理,每次迭代都完整地走一遍“读取-填充-写入-关闭”的流程,虽然看起来有点重复,但它能保证每个PDF生成过程的独立性和资源的及时释放。

内存管理:

大对象处理: 如果你的PDF模板非常大,或者填充的数据量非常大(比如嵌入大量图片),要特别留意内存使用。尽量避免一次性将所有数据加载到内存中,可以考虑分批处理。JVM参数调优: 对于Java应用,适当调整JVM的堆内存大小(-Xmx)和垃圾回收器(GC)策略,能有效缓解内存压力。比如,在生产环境中,可以考虑使用G1GC。

并发与线程池:

多线程处理: 如果你的服务器CPU核心数足够,并且I/O不是瓶颈,可以考虑使用线程池(ExecutorService)来并行生成PDF。将每个PDF的生成任务封装成一个CallableRunnable提交给线程池。线程安全: 大多数PDF库(如OpenPDF、PDFBox)的核心操作对象(如PdfReaderPdfStamper)不是线程安全的。这意味着每个线程需要有自己独立的PdfReaderPdfStamper实例。不要尝试共享这些对象,否则会引发不可预测的并发问题。

中间文件处理:

如果PDF生成过程中涉及到临时文件,确保这些文件在任务完成后被清理。对于最终的PDF文件,如果生成后需要立即上传到文件存储服务(如S3),可以考虑直接将输出流连接到上传接口,避免先写入本地磁盘再读取上传,减少一次I/O操作。

错误处理与重试机制:在批量处理中,单个PDF生成失败不应该影响整个批次。为每个生成任务添加健壮的异常捕获,记录详细日志,并考虑实现简单的重试机制,以提高整体的成功率。

总的来说,批量生成PDF,最核心的还是“用完即扔”的资源管理哲学。确保每个PDF生成过程都是独立的、资源可控的,并且在完成后立即释放所有资源,这样才能保证系统的稳定性和可扩展性。

以上就是Java实现PDF模板填充的详细技术方案的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/135348.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月29日 17:20:37
下一篇 2025年11月29日 17:32:06

相关推荐

  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    600
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • CSS元素设置em和transition后,为何载入页面无放大效果?

    css元素设置em和transition后,为何载入无放大效果 很多开发者在设置了em和transition后,却发现元素载入页面时无放大效果。本文将解答这一问题。 原问题:在视频演示中,将元素设置如下,载入页面会有放大效果。然而,在个人尝试中,并未出现该效果。这是由于macos和windows系统…

    2025年12月24日
    200
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 如何用HTML/JS实现Windows 10设置界面鼠标移动探照灯效果?

    Win10设置界面中的鼠标移动探照灯效果实现指南 想要在前端开发中实现类似于Windows 10设置界面的鼠标移动探照灯效果,有两种解决方案:CSS 和 HTML/JS 组合。 CSS 实现 不幸的是,仅使用CSS无法完全实现该效果。 立即学习“前端免费学习笔记(深入)”; HTML/JS 实现 要…

    2025年12月24日
    000
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000
  • 如何用前端实现 Windows 10 设置界面的鼠标移动探照灯效果?

    如何在前端实现 Windows 10 设置界面中的鼠标移动探照灯效果 想要在前端开发中实现 Windows 10 设置界面中类似的鼠标移动探照灯效果,可以通过以下途径: CSS 解决方案 DEMO 1: Windows 10 网格悬停效果:https://codepen.io/tr4553r7/pe…

    2025年12月24日
    000
  • 如何用前端技术实现Windows 10 设置界面鼠标移动时的探照灯效果?

    探索在前端中实现 Windows 10 设置界面鼠标移动时的探照灯效果 在前端开发中,鼠标悬停在元素上时需要呈现类似于 Windows 10 设置界面所展示的探照灯效果,这其中涉及到了元素外围显示光圈效果的技术实现。 CSS 实现 虽然 CSS 无法直接实现探照灯效果,但可以通过以下技巧营造出类似效…

    2025年12月24日
    000
  • 使用CSS mask属性指定图片URL时,为什么浏览器无法加载图片?

    css mask属性未能加载图片的解决方法 使用css mask属性指定图片url时,如示例中所示: mask: url(“https://api.iconify.design/mdi:apple-icloud.svg”) center / contain no-repeat; 但是,在网络面板中却…

    2025年12月24日
    000
  • 如何用CSS Paint API为网页元素添加时尚的斑马线边框?

    为元素添加时尚的斑马线边框 在网页设计中,有时我们需要添加时尚的边框来提升元素的视觉效果。其中,斑马线边框是一种既醒目又别致的设计元素。 实现斜向斑马线边框 要实现斜向斑马线间隔圆环,我们可以使用css paint api。该api提供了强大的功能,可以让我们在元素上绘制复杂的图形。 立即学习“前端…

    2025年12月24日
    000
  • 图片如何不撑高父容器?

    如何让图片不撑高父容器? 当父容器包含不同高度的子元素时,父容器的高度通常会被最高元素撑开。如果你希望父容器的高度由文本内容撑开,避免图片对其产生影响,可以通过以下 css 解决方法: 绝对定位元素: .child-image { position: absolute; top: 0; left: …

    2025年12月24日
    000
  • CSS 帮助

    我正在尝试将文本附加到棕色框的左侧。我不能。我不知道代码有什么问题。请帮助我。 css .hero { position: relative; bottom: 80px; display: flex; justify-content: left; align-items: start; color:…

    2025年12月24日 好文分享
    200

发表回复

登录后才能评论
关注微信