企业级-PDF水印签字

作者:fyupeng
技术专栏:☞ https://github.com/fyupeng
项目地址:☞ https://github.com/fyupeng/distributed-blog-system-api


留给读者

一、介绍

根据关键字偏移量水印签字。

允许重复调用,文件安全性高,自动备份并恢复。

二、代码

测试 Demo
PdfUtilTest .java


package com.gwssi.common.service;import com.gwssi.common.util.PDFSigner;
import com.gwssi.torch.v1.dao.TorchDaoManager;
import com.gwssi.torch.v1.dao.api.IBaseDao;
import org.apache.commons.lang.StringUtils;import java.awt.*;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;/*** @Auther: fyp* @Date: 2024/6/24* @Description:* @Package: com.gwssi.common.service* @Version: 1.0*/
public class PdfUtilTest {public String signPdf(String applyno) {String[] keywords = {"领取人签字", "发照日期", "领取日期"};Float[] signatureTextOffsetX = {0f ,100f, -20f};Float[] signatureTextOffsetY = {-50f, 0f, -50f};//String[] signatureTexts = {"小明", "2024-04-26", "2024-04-28"};List<String> signatureTexts = new ArrayList<>();String filepath = "C:/output.pdf";Map<String,Object> signatureParams = new HashMap<>();signatureParams.put("keywords", keywords);signatureParams.put("signatureTexts", signatureTexts);signatureParams.put("signatureTextOffsetX", signatureTextOffsetX);signatureParams.put("signatureTextOffsetY", signatureTextOffsetY);PDFSigner.signPdf(filepath, signatureParams, Color.gray, 18);return filepath;}}

封装类 PDFSigner.java

package com.gwssi.common.util;/*** @Auther: fyp* @Date: 2024/4/28* @Description:* @Package: com.gwssi.common.util* @Version: 1.0*/
import java.awt.*;
import java.io.*;
import java.util.*;
import java.util.List;import org.apache.pdfbox.pdmodel.PDDocument;
import org.apache.pdfbox.pdmodel.PDPage;
import org.apache.pdfbox.pdmodel.PDPageContentStream;
import org.apache.pdfbox.pdmodel.font.PDFont;
import org.apache.pdfbox.pdmodel.font.PDType0Font;
import org.apache.pdfbox.text.PDFTextStripper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;public class PDFSigner {private static final Logger LOG = LoggerFactory.getLogger(PDFSigner.class);private static String filePath;private static ResourceBundle resourceBundle;static {resourceBundle = ResourceBundle.getBundle("torch-v1");filePath = resourceBundle.getString("project.filePath");}//private static final Properties properties = System.getProperties();// 获取当前用户的工作目录//private static final String userDir = properties.getProperty("user.dir");//private static final String rootDirPath = userDir + "/WebContent/";public static byte[] readFileToBytes(File file) throws IOException {FileInputStream fis = new FileInputStream(file);byte[] bytes = new byte[(int) file.length()];fis.read(bytes);fis.close();return bytes;}public static void backPdf(String pdfPath) {// 文件 备份String backPdfPath = pdfPath + ".bak";try {File file = new File(pdfPath);byte[] fileBytes = null;try {if (checkFileSign(pdfPath, backPdfPath)) {fileBytes =  readFileToBytes(file);backFile(fileBytes, backPdfPath);}} catch (Exception e) {LOG.error("文件备份异常,失败原因:{}", e.getMessage(), e);throw new RuntimeException("文件备份异常,失败原因:" + e.getMessage());}} catch (Exception e) {LOG.error("文件备份异常,失败原因:{}", e.getMessage(), e);throw new RuntimeException("文件备份异常,失败原因:" + e.getMessage());}}public static void signPdf(String pdfPath, Map<String ,Object> signParams, Color signatureColor, int signatureFontSize) {String[] keywords = (String[]) signParams.get("keywords");List<String> signatureTexts = (List<String>) signParams.get("signatureTexts");Float[] signatureTextOffsetX = (Float[]) signParams.get("signatureTextOffsetX");Float[] signatureTextOffsetY = (Float[]) signParams.get("signatureTextOffsetY");// 文件 备份String backPdfPath = pdfPath + ".bak";File file = new File(pdfPath);byte[] fileBytes = null;try {if (checkFileSign(pdfPath, backPdfPath)) {fileBytes = readFileToBytes(file);backFile(fileBytes, backPdfPath);} else {fileBytes = readFileToBytes(file);}PDDocument document = PDDocument.load(new File(pdfPath));for (int idx = 0; idx < keywords.length; idx++) {List<KeyWordPosition> keyWordsByByte = PdfMatchKeyword.getKeyWordsByByte(fileBytes, keywords[idx]);System.out.println(keyWordsByByte);float[] keyWordsByByte2 = PdfHelper.getKeyWordsByByte(fileBytes, keywords[idx]);float signatureX = keyWordsByByte2[0]; // 使用关键字的 x 坐标作为签名的 x 坐标float signatureY = keyWordsByByte2[1] ; // 在关键字的 y 坐标上移 20 个单位作为签名的 y 坐标int pageNum = findKeywordPage(document, keywords[idx]);if (pageNum != -1) {signPage(document, pageNum, signatureTexts.get(idx), signatureX + signatureTextOffsetX[idx], signatureY + signatureTextOffsetY[idx], signatureColor, signatureFontSize);// PDF 覆盖 (原文件备份)document.save(pdfPath);System.out.println("PDF 文件签名成功!");} else {System.out.println("未找到关键字,无法签名。");}}document.close();//deleteFile(pdfPath);} catch (Exception e) {LOG.error("文件备份异常,失败原因:{}", e.getMessage(), e);throw new RuntimeException("文件备份异常,失败原因:" + e.getMessage());}}public static byte[] fileToBytes(File file) throws IOException {// 创建字节输入流FileInputStream fis = new FileInputStream(file);try {// 获取文件大小long fileSize = file.length();// 创建一个与文件大小相同的字节数组byte[] bytesArray = new byte[(int) fileSize];// 将文件内容读取到字节数组fis.read(bytesArray);// 返回字节数组return bytesArray;} finally {// 关闭文件输入流fis.close();}}/**** 检查是否第一次生成 第一次 true* @param pdfPath* @param backPdfPath* @return* @throws IOException*/private static boolean checkFileSign(String pdfPath, String backPdfPath) throws IOException {File pdfFile = new File(pdfPath);File backPDFFile = new File(backPdfPath);// 恢复文件 重新生成 (存在备份文件,非第一次生成)if (backPDFFile.exists()) {if (pdfFile.exists()) {pdfFile.delete();}byte[] bytes = fileToBytes(backPDFFile);FileOutputStream fos = new FileOutputStream(pdfFile);fos.write(bytes);fos.close();return false;}// 第一次生成return true;}private static void backFile(byte[] fileBytes, String backPdfPath) throws IOException {File file = new File(backPdfPath);FileOutputStream fos = new FileOutputStream(file);fos.write(fileBytes);fos.close();}private static void deleteFile(String pdfPath) throws IOException {File file = new File(pdfPath);file.delete();}public static void main(String[] args) throws FileNotFoundException {String[] keywords = {"领取人签字", "发照日期", "领取日期"};Float[] signatureTextOffsetX = {0f ,100f, -20f};Float[] signatureTextOffsetY = {-50f, 0f, -50f};String[] signatureTexts = {"小明", "2024-04-26", "2024-04-28"};Map<String,Object> signatureParams = new HashMap<>();signatureParams.put("keywords", keywords);signatureParams.put("signatureTexts", signatureTexts);signatureParams.put("signatureTextOffsetX", signatureTextOffsetX);signatureParams.put("signatureTextOffsetY", signatureTextOffsetY);signPdf("D:/doc/2023-10-13/14-37-37/珠海陈文路走走杨服装零售店(440003A2300006440).pdf", signatureParams, Color.gray, 18);}public static int findKeywordPage(PDDocument document, String keyword) throws IOException {PDFTextStripper stripper = new PDFTextStripper();int pageNum = 0;for (PDPage page : document.getPages()) {stripper.setStartPage(pageNum + 1);stripper.setEndPage(pageNum + 1);String text = stripper.getText(document);if (text.contains(keyword)) {return pageNum;}pageNum++;}return -1;}public static void signPage(PDDocument document, int pageNum, String signatureText, float signatureX, float signatureY, Color signatureColor, int fontSize) throws IOException {PDPage page = document.getPage(pageNum);try (PDPageContentStream contentStream = new PDPageContentStream(document, page, PDPageContentStream.AppendMode.APPEND, true)) {contentStream.beginText();//PDFont font = PDType0Font.load(document, new File(filePath, "/app/font/simsun.ttf")); // 替换成实际的字体文件路径//PDFont font = PDType0Font.load(document, new File(filePath, "/app/font/STSong.ttf")); // 替换成实际的字体文件路径//PDFont font = PDType0Font.load(document, new File(filePath, "/app/font/simhei.ttf")); // 替换成实际的字体文件路径PDFont font = PDType0Font.load(document, new File( filePath + "/app/font/hanyizhonghei.ttf")); // 替换成实际的字体文件路径contentStream.setFont(font, fontSize);contentStream.setNonStrokingColor(signatureColor); // 设置文本颜色contentStream.newLineAtOffset(signatureX, signatureY); // 垂直偏移量减少,签字下移contentStream.showText(signatureText);contentStream.endText();} catch (Exception e) {LOG.error("签名异常,异常原因:{}", e.getMessage(), e);throw new RuntimeException("签名异常,异常原因:" + e.getMessage());}}}

PdfMatchKeyword.java


package com.gwssi.common.util;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;import com.itextpdf.text.BaseColor;
import com.itextpdf.text.DocumentException;
import com.itextpdf.text.Rectangle;
import com.itextpdf.text.pdf.PdfContentByte;
import com.itextpdf.text.pdf.PdfReader;
import com.itextpdf.text.pdf.PdfStamper;
import com.itextpdf.text.pdf.parser.ImageRenderInfo;
import com.itextpdf.text.pdf.parser.PdfTextExtractor;
import com.itextpdf.text.pdf.parser.TextExtractionStrategy;
import com.itextpdf.text.pdf.parser.TextRenderInfo;
import com.itextpdf.text.pdf.parser.Vector;public class PdfMatchKeyword {/*** 用于供外部类调用获取关键字所在PDF文件坐标* @param filepath* @param keyWords* @return*/public static List<KeyWordPosition> getKeyWordsByPath(String filepath, String keyWords) {List<KeyWordPosition> matchItems = null;try{PdfReader pdfReader = new PdfReader(filepath);matchItems = getKeyWords(pdfReader, keyWords);} catch (IOException e) {e.printStackTrace();}return matchItems;}/*** 用于供外部类调用获取关键字所在PDF文件坐标* @param filepath* @param keyWords* @return*/public static List<KeyWordPosition> getKeyWordsByByte(byte[] pdfIn, String keyWords) {List<KeyWordPosition> matchItems = null;try{PdfReader pdfReader = new PdfReader(pdfIn);matchItems = getKeyWords(pdfReader, keyWords);} catch (IOException e) {e.printStackTrace();}return matchItems;}/*** 获取关键字所在PDF坐标* @param pdfReader* @param keyWords* @return*/private static List<KeyWordPosition> getKeyWords(PdfReader pdfReader, String keyWords) {int page = 0;List<KeyWordPosition> matchItems = new ArrayList<>();try{int pageNum = pdfReader.getNumberOfPages();StringBuilder allText = null;//遍历页for (page = 1; page <= pageNum; page++) {//只记录当页的所有内容,需要记录全部页放在循环外面List<ItemPosition> allItems = new ArrayList<>();//扫描内容MyTextExtractionStrategy myTextExtractionStrategy = new MyTextExtractionStrategy(allItems, page);PdfTextExtractor.getTextFromPage(pdfReader, page, myTextExtractionStrategy);//当页的文字内容,用于关键词匹配allText = new StringBuilder();//一个字一个字的遍历for (int i=0; i<allItems.size(); i++) {ItemPosition item = allItems.get(i);allText.append(item.getText());//关键字存在连续多个块中if(allText.indexOf(keyWords) != -1) {KeyWordPosition keyWordPosition = new KeyWordPosition();//记录关键词每个字的位置,只记录开始结束标记时会有问题List<ItemPosition> listItem = new ArrayList<>();for(int j=i-keyWords.length()+1; j<=i; j++) {listItem.add(allItems.get(j));}keyWordPosition.setListItem(listItem);keyWordPosition.setText(keyWords);matchItems.add(keyWordPosition);allText.setLength(0);}}}} catch (Exception e) {e.printStackTrace();}return matchItems;}/*** 添加矩形标记* @param oldPath* @param newPath* @param matchItems 关键词* @param color 标记颜色* @param lineWidth 线条粗细* @param padding 边框内边距* @throws DocumentException* @throws IOException*/public static void andRectangleMark(String oldPath, String newPath, List<KeyWordPosition> matchItems, BaseColor color, int lineWidth, int padding) throws DocumentException, IOException{// 待加水印的文件PdfReader reader = new PdfReader(oldPath);// 加完水印的文件PdfStamper stamper = new PdfStamper(reader, new FileOutputStream(newPath));PdfContentByte content;// 设置字体// 循环对每页插入水印for (KeyWordPosition keyWordPosition:matchItems){//一个关键词的所有字坐标List<ItemPosition> oneKeywordItems = keyWordPosition.getListItem();for(int i=0; i<oneKeywordItems.size(); i++) {ItemPosition item = oneKeywordItems.get(i);ItemPosition preItem = i==0?null:oneKeywordItems.get(i-1);// 水印的起始content = stamper.getOverContent(item.getPage());// 开始写入水印content.setLineWidth(lineWidth);content.setColorStroke(color);//底线content.moveTo(item.getRectangle().getLeft()-padding, item.getRectangle().getBottom()-padding);content.lineTo(item.getRectangle().getRight()+padding, item.getRectangle().getBottom()-padding);if(i!=0 && preItem!=null && (preItem.getRectangle().getBottom()-padding)==(item.getRectangle().getBottom()-padding) && (preItem.getRectangle().getRight()+padding)!=(item.getRectangle().getLeft()-padding)) {content.moveTo(preItem.getRectangle().getRight()+padding, preItem.getRectangle().getBottom()-padding);content.lineTo(item.getRectangle().getLeft()-padding, item.getRectangle().getBottom()-padding);}//上线content.moveTo(item.getRectangle().getLeft()-padding, item.getRectangle().getTop()+padding);content.lineTo(item.getRectangle().getRight()+padding, item.getRectangle().getTop()+padding);if(i!=0 && preItem!=null && (preItem.getRectangle().getTop()+padding)==(item.getRectangle().getTop()+padding) && (preItem.getRectangle().getRight()+padding)!=(item.getRectangle().getLeft()-padding)) {content.moveTo(preItem.getRectangle().getRight()+padding, preItem.getRectangle().getTop()+padding);content.lineTo(item.getRectangle().getLeft()-padding, item.getRectangle().getTop()+padding);}//左线if(i==0) {content.moveTo(item.getRectangle().getLeft()-padding, item.getRectangle().getBottom()-padding);content.lineTo(item.getRectangle().getLeft()-padding, item.getRectangle().getTop()+padding);}//右线if(i==(oneKeywordItems.size()-1)) {content.moveTo(item.getRectangle().getRight()+padding, item.getRectangle().getBottom()-padding);content.lineTo(item.getRectangle().getRight()+padding, item.getRectangle().getTop()+padding);}content.stroke();}}stamper.close();}public static void main(String[] args) throws Exception {String keyword = "陈雪英";String sourcePdf = "D:\\SoftPackage\\InterDowloads\\东莞市浩影塑胶模具科技有限公司-登记审核表.pdf";String watermarkPdf = "D:\\SoftPackage\\InterDowloads\\东莞市浩影塑胶模具科技有限公司-登记审核表2.pdf";Long start = System.currentTimeMillis();System.out.println("开始扫描....");List<KeyWordPosition> matchItems = getKeyWordsByPath(sourcePdf, keyword);System.out.println("扫描结束["+(System.currentTimeMillis()-start)+"ms],共找到关键字["+keyword+"]出现["+matchItems.size()+"]次");start = System.currentTimeMillis();System.out.println("开始添加标记....");andRectangleMark(sourcePdf, watermarkPdf, matchItems, BaseColor.RED, 2, 2);System.out.println("标记添加完成["+(System.currentTimeMillis()-start)+"ms]");}
}/*** @ClassName: MyTextExtractionStrategy* @Description: 记录所有位置+字体信息,这种方式获取坐标信息和字体信息方便一点* @author chenyang-054* @date 2021-04-09 11:00:31*/
class MyTextExtractionStrategy implements TextExtractionStrategy{private List<ItemPosition> positions;private Integer page;public MyTextExtractionStrategy() {}public MyTextExtractionStrategy(List<ItemPosition> positions, Integer page) {this.positions = positions;this.page = page;}@Overridepublic void beginTextBlock() {// TODO Auto-generated method stub}@Overridepublic void renderText(TextRenderInfo renderInfo) {ItemPosition ItemPosition = new ItemPosition();Vector bottomLeftPoint = renderInfo.getDescentLine().getStartPoint();Vector topRightPoint = renderInfo.getAscentLine().getEndPoint();//记录矩形坐标Rectangle rectangle = new Rectangle(bottomLeftPoint.get(Vector.I1), bottomLeftPoint.get(Vector.I2),topRightPoint.get(Vector.I1), topRightPoint.get(Vector.I2));ItemPosition.setPage(page);ItemPosition.setRectangle(rectangle);ItemPosition.setText(renderInfo.getText());positions.add(ItemPosition);}@Overridepublic void endTextBlock() {// TODO Auto-generated method stub}@Overridepublic void renderImage(ImageRenderInfo renderInfo) {// TODO Auto-generated method stub}@Overridepublic String getResultantText() {// TODO Auto-generated method stubreturn null;}
}

PdfHelper .java

package com.gwssi.common.util;import com.itextpdf.text.pdf.PdfReader;
import com.itextpdf.text.pdf.parser.PdfReaderContentParser;import java.io.IOException;/*** @ClassName PdfHelper* @Description Pdf帮助类* @Author AlphaJunS* @Date 2020/3/7 17:40* @Version 1.0*/
public class PdfHelper {/*** @Author AlphaJunS* @Date 18:24 2020/3/7* @Description 用于供外部类调用获取关键字所在PDF文件坐标* @param filepath* @param keyWords* @return float[]*/public static float[] getKeyWordsByPath(String filepath, String keyWords) {float[] coordinate = null;try{PdfReader pdfReader = new PdfReader(filepath);coordinate = getKeyWords(pdfReader, keyWords);} catch (IOException e) {e.printStackTrace();}return coordinate;}public static float[] getKeyWordsByByte(byte[] pdfin, String keyWords) {float[] coordinate = null;try{PdfReader pdfReader = new PdfReader(pdfin);coordinate = getKeyWords(pdfReader, keyWords);} catch (IOException e) {e.printStackTrace();}return coordinate;}/*** @Author AlphaJunS* @Date 18:26 2020/3/7* @Description 获取关键字所在PDF坐标* @param pdfReader* @param keyWords* @return float[]*/private static float[] getKeyWords(PdfReader pdfReader, String keyWords) {float[] coordinate = null;int page = 0;try{int pageNum = pdfReader.getNumberOfPages();PdfReaderContentParser pdfReaderContentParser = new PdfReaderContentParser(pdfReader);CustomRenderListener renderListener = new CustomRenderListener();renderListener.setKeyWord(keyWords);for (page = 1; page <= pageNum; page++) {renderListener.setPage(page);pdfReaderContentParser.processContent(page, renderListener);coordinate = renderListener.getPcoordinate();if (coordinate != null) break;}} catch (IOException e) {e.printStackTrace();}return coordinate;}public static void main(String[] args) {float[] keyWordsByPath = getKeyWordsByPath("D:\\SoftPackage\\InterDowloads\\优选(东莞)汽车服务有限公司-登记审核表.pdf", "东莞市");for(float f : keyWordsByPath){System.out.println(f);}
//		238.35
//		141.6
//		1.0}
}

三、总结

高效、便携、易上手!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

win10系统管理员账号怎么切换

1、按住“windowsx”&#xff0c;选择“计算机管理” 2、在页面左侧&#xff0c;找到“计算机管理(本地)”&#xff0c;展开“系统工具”&#xff0c;点击“本地用户和组”下面的“用户”&#xff0c;在右侧找到“Administrator”&#xff0c;双击打开。 3、在打开页面选择常规…

Docker(六)-本地镜像发布到私有库

1.下载镜像Docker Registry 用于搭建私人版本Docker Hub docker pull registry2.运行私有库Registry 运行私有库Registry&#xff0c;相当于本地有个私有Docker hubdocker run -d -p hostPort:containerPort -v 【宿主机目录】:【容器目录】 --privilegedtrue 【私有库镜像】…

泛微E9开发 根据判断条件,控制字段的编辑/必填属性

根据判断条件&#xff0c;控制字段的编辑/必填属性 1、需求说明2、实现方法3、扩展知识点1. 注册钩子事件&#xff0c;指定动作完成后触发1.1 接口名称及参数说明1.2 案例 2. 改变单个字段显示属性(只读/必填等)2.1 参数说明2.2 案例 1、需求说明 当字段“填报人”和字段“姓名…

android-aidl4

转&#xff1a;Android Aidl的使用_android aidl使用-CSDN博客 一.准备 Parcelable&#xff0c;可以理解成只是把car整个对象在aidl中进行传递&#xff0c;就理解成一个car的一个类吧&#xff0c;和其他类使用一样就行了&#xff0c;回调&#xff1a;把接口作为参数放在函数参…

Spring Boot集成Redisson

文章目录 Spring Boot集成Redisson1. Redisson概述2. Redission作用3. 集成Redission前提&#xff1a;步骤 1: 添加依赖步骤 2: 配置Redisson 4. 结论 Spring Boot集成Redisson 1. Redisson概述 Redisson是一个在Redis基础上实现的Java驻内存数据网格&#xff08;In-Memory D…

从广州到上海|荣载光的智慧 与SSHT共同探索智能照明更多想象空间

随着生活水平的提高&#xff0c;大众对高品质生活的追求脚步逐步加快&#xff0c;人们对智能照明的需求日益多样化&#xff0c;不再仅仅满足于传统的照明功能&#xff0c;而是转向智能照明系统&#xff0c;提出更高的需求。 展望未来&#xff0c;中国智能照明市场预计将迎来全…

文旅景区科技馆增加5D影院项目VR游乐场

王屋山风景名胜区增加20座5D影院设备&#xff0c;5D影院不像普通电影院&#xff0c;5D影院座椅可以根据影片播放内容&#xff0c;进行实时摆动和各种特效&#xff0c;闪电&#xff0c;振臀振臂&#xff0c;泡泡等个种特效。影片内容可以进行定制&#xff0c;根据景区文化风光&a…

AI交互及爬虫【数据分析】

各位大佬好 &#xff0c;这里是阿川的博客&#xff0c;祝您变得更强 个人主页&#xff1a;在线OJ的阿川 大佬的支持和鼓励&#xff0c;将是我成长路上最大的动力 阿川水平有限&#xff0c;如有错误&#xff0c;欢迎大佬指正 Python 初阶 Python–语言基础与由来介绍 Python–…

k8s学习--chart包开发(创建chart包)

文章目录 chart包应用环境一、安装helm客户端工具二、chart包目录结构三、创建不可配置的chart1.创建目录和chart.yaml2.创建deployment.yaml3.创建service.yaml4.使用chart安装应用5.查看和验证 四、创建可配置的Chart1.官方的预定义变量2.新增values.yaml文件3.配置deploy引用…

网络安全协议

1. 概述 1.1 网络安全需求 五种需求&#xff1a; 机密性&#xff1a;防止数据未授权公开&#xff0c;让消息对无关听众保密 完整性&#xff1a;防止数据被篡改 可控性&#xff1a;限制对网络资源&#xff08;硬件和软件&#xff09;和数据&#xff08;存储和通信&#xff0…

【C++:哈希】

目录 哈希概念 哈希冲突 如何解决哈希冲突&#xff1f; 闭散列 开散列/哈希桶&#xff08;链地址法&#xff09; 负载因子 哈希概念 哈希&#xff1a; 一种高效用来搜索的数据结构。哈希利用某一种函数使元素的存储位置与它的关键码之间建立一个映射关系&#xff0c;方便…

SpringBoot+Vue 数据的批量导入和导出

导出 总结了三种导出数据为表格的情况&#xff0c;直接上代码&#xff01;&#xff01; 整体导出 后端接口&#xff1a; // 批量导出数据GetMapping("/export")public void exportDate(HttpServletResponse response) throws IOException {ExcelWriter writer…

前端axios发送请求如何导出excel文件?

// 导出方法 function down(){let reqUrl import.meta.env.BASE_URL /api/AboutView/downLoad?aboutTypeadmin_temp;let _token sessionStorage.get(token).replace(/\"/g, "");let xhr new XMLHttpRequest();xhr.open("get", reqUrl, true);xhr…

计算机网络 交换机的安全配置

一、理论知识 1.交换机端口安全功能介绍 交换机端口安全功能是针对交换机端口进行安全属性的配置&#xff0c;以控制用户的安全接入。主要包括以下两种配置项&#xff1a; ①限制交换机端口的最大连接数&#xff1a;控制交换机端口连接的主机数量&#xff1b;防止用户进行恶…

Java常用类--String类--上

String类 String类介绍 &#xff08;1&#xff09;创建的每个字符串实际上都是String类的对象。即使是字符串字面值实际上也是String对象。 &#xff08;2&#xff09;String类型的对象是不可变的&#xff1b;一旦创建了一个String对象&#xff0c;其内容就不能再改变。即&a…

【华为OD机试】Linux发行版的数量(C++ Java JavaScript Python )

题目 题目描述 Linux操作系统有多个发行版,distrowatch.com提供了各个发行版的资料。这些发行版互相存在关联,例如Ubuntu基于Debian开发,而Mint又基于Ubuntu开发,那么我们认为Mint同Debian也存在关联。 发行版集是一个或多个相关存在关联的操作系统发行版,集合内不包含没有…

kafka基础概念

目录 1、kafka简介 2、kafka使用场景 3、kafka基础概念 3.1、消息 3.1.1、消息构成详解 3.1.2、消息存储设计 3.2、topic 3.3、partition 3.4、offset 3.5、replication 3.5.1、replication简介 3.5.2、副本角色 3.5.3、副本类型 3.5.3.1、副本类型简介 3.5.3.2、…

30分钟学习如何搭建扩散模型的运行环境【pytorch版】【B站免费视频教程!】【解决环境搭建问题】

30分钟学习如何搭建扩散模型的运行环境【B站免费视频教程&#xff01;】【解决环境搭建问题】 动手学习扩散模型 点击以下链接即可进入学习&#xff1a; B站免费视频教程环境配置安装&#xff08;配套讲解文档&#xff09; 视频 讲解主要内容 一、环境设置 1.本地安装&…

用Python设置Excel工作表网格线的隐藏与显示

Excel表格界面的直观性很大程度上得益于表格中的网格线设计&#xff0c;这些线条帮助用户精确对齐数据&#xff0c;清晰划分单元格。网格线是Excel界面中默认显示的辅助线&#xff0c;用于辅助定位&#xff0c;与单元格边框不痛&#xff0c;不影响打印输出。然而&#xff0c;在…

fyne的对话框

对话框 import "fyne.io/fyne/v2/dialog"dialog包 定义了应用程序GUI的标准对话框窗口。 NewError NewError()为应用程序错误在指定的窗口上创建一个对话框。该消息是从提供的错误中提取的&#xff08;不应为nil&#xff09;。创建后&#xff0c;您应该调用Show()…