pdf导出实例(itestpdf)
依赖
<!-- https://mvnrepository.com/artifact/com.itextpdf/itextpdf --><dependency><groupId>com.itextpdf</groupId><artifactId>itextpdf</artifactId><version>5.5.10</version></dependency>
工具类
package org.jeecg.utils;import com.itextpdf.text.*;
import com.itextpdf.text.pdf.*;
import io.swagger.annotations.ApiOperation;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.Resource;
import org.springframework.stereotype.Component;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;import javax.servlet.http.HttpServletResponse;
import java.io.ByteArrayOutputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.URLEncoder;
import java.util.*;
import java.util.List;public class PDFUtil {/*** pdf模板导出** @param map 导出结果* @param filePath 空pdf路径,作物前端访问路径* @throws Exception*/public static void creatPdf(Map<String, Object> map, String filePath) throws Exception {try {FileOutputStream fos = new FileOutputStream(filePath);;BaseFont bf = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);// 读取pdf模板路径Resource resource = new ClassPathResource(String.valueOf(map.get("tempPath")));PdfReader reader = new PdfReader(resource.getURL());ByteArrayOutputStream bos = new ByteArrayOutputStream();PdfStamper stamper = new PdfStamper(reader, bos);stamper.setFormFlattening(true);AcroFields form = stamper.getAcroFields();// 文字类的内容处理Map<String, String> datemap = (Map<String, String>) map.get("dataMap");form.addSubstitutionFont(bf);for (String key : datemap.keySet()) {String value = datemap.get(key);form.setField(key, value);form.setFieldProperty(key,"textsize",9f,null);}// 图片类的内容处理Map<String, String> imgmap = (Map<String, String>) map.get("imgMap");for (String key : imgmap.keySet()) {String value = imgmap.get(key);String imgpath = value;int pageNo = form.getFieldPositions(key).get(0).page;Rectangle signRect = form.getFieldPositions(key).get(0).position;float x = signRect.getLeft();float y = signRect.getBottom();// 根据路径读取图片Image image = Image.getInstance(imgpath);// 获取图片页面PdfContentByte under = stamper.getOverContent(pageNo);// 图片大小自适应image.scaleToFit(signRect.getWidth(), signRect.getHeight());// 添加图片image.setAbsolutePosition(x, y);under.addImage(image);}// 如果为false,生成的PDF文件可以编辑,如果为true,生成的PDF文件不可以编辑stamper.setFormFlattening(false);stamper.close();Document doc = new Document();PdfCopy copy = new PdfCopy(doc, fos);doc.open();int pageNum = reader.getNumberOfPages();for (int i = 1; i <= pageNum; i++) {PdfImportedPage importPage = copy.getImportedPage(new PdfReader(bos.toByteArray()), i);copy.addPage(importPage);}doc.close();fos.close();} catch (IOException e) {System.out.println(e);} catch (DocumentException e) {System.out.println(e);}}}
导出结果数据实例(仅供参考)
package com.fiftheconomic.dataCalculate.service.impl;import com.fiftheconomic.dataCalculate.domain.TjDataCalculateBase;
import com.fiftheconomic.dataCalculate.domain.TjDataCalculateDetail;
import com.fiftheconomic.dataCalculate.domain.resCompenyIndex;
import com.fiftheconomic.dataCalculate.service.IPdfService;
import org.springframework.stereotype.Service;import java.text.SimpleDateFormat;
import java.time.LocalDate;
import java.util.*;/*** @Author: 苏子龙* @Date: 2023/08/01/17:00* @Description:*/
@Service
public class PdfServiceImpl implements IPdfService {/*** 123 工业单位填报 124 批发和零售业单位填报 125 住宿和餐饮业单位填报 126 制造业,信息传输、软件和信息技术服务业,科学研究和技术服务业单位填报** @param resMap* @param list* @return*/@Overridepublic Map<String, Object> oneThree(List<resCompenyIndex> resMap, TjDataCalculateBase base, List<TjDataCalculateDetail> list) {List<List<String>> lists = new ArrayList<>();//表格 (一行数据一个list)SimpleDateFormat dft = new SimpleDateFormat("yyyy-MM-dd");Map<String, String> map = new HashMap<>();//文本域for (int i = 0; i < resMap.size(); i++) {if (resMap.get(i).getIndexCode() != null) {if (!map.containsKey(resMap.get(i).getIndexCode())) {map.put(resMap.get(i).getIndexCode(), resMap.get(i).getIndexValue());}}}//添加tj_data_calculate_index基础信息表数据for (int j = 0; j < list.size(); j++) {System.out.println("++++++++" + j);TjDataCalculateDetail data = list.get(j);map.put("name" + j, data.getIndexName());map.put("unit" + j, data.getUnitof());map.put("value" + j, data.getIndexValue());}
// map.put("name", "无"); //添加tj_data_calculate_index头信息
// map.put("nameInfo", "无"); //填表人姓名
// map.put("phoen", "无");//填表人电话
// map.put("nameBy", "无");//单位负责人
// map.put("USCI", base.getUSCI());//企业社会信用统一代码
// map.put("comName", base.getComName());//单位详细名称
// //添加日期
// LocalDate now = LocalDate.now();
// map.put("y", String.valueOf(now.getYear()));
// map.put("m", String.valueOf(now.getMonthValue()));
// map.put("d", String.valueOf(now.getDayOfMonth()));
// map.put("time", String.valueOf(now));//填表人电话Map<String, Object> o = new HashMap<>();String s = base.getIndustryCode();if (s.equals("1")) {o.put("tempPath", "static/pdf/123.pdf");} else if (s.equals("2") || s.equals("3")) {o.put("tempPath", "static/pdf/124.pdf");} else if (s.equals("4") || s.equals("5")) {o.put("tempPath", "static/pdf/125.pdf");} else {o.put("tempPath", "static/pdf/126.pdf");}o.put("dataMap", map);o.put("imgMap", new HashMap<>());Map<String, List<List<String>>> listMap = new HashMap<>();listMap.put("table", lists);o.put("tableList", listMap);return o;}public static void main(String[] args) {LocalDate now = LocalDate.now();System.out.println(now.getYear());System.out.println(now.getMonthValue());System.out.println(now.getDayOfMonth());}}
优化工具包
public static String creatPdf1(Map<String, Object> map ,OutputStream out,float width[]) throws Exception {BaseFont bf = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
// // 输出流
// String outPutPath = "E:/excel/files/pdf模板导出.pdf";
// FileOutputStream out = new FileOutputStream(filePath);// 读取pdf模板路径Resource resource = new ClassPathResource(String.valueOf(map.get("tempPath")));PdfReader reader = new PdfReader(resource.getURL());ByteArrayOutputStream bos = new ByteArrayOutputStream();PdfStamper stamper = new PdfStamper(reader, bos);stamper.setFormFlattening(true);AcroFields form = stamper.getAcroFields();// 文字类的内容处理Map<String, String> datemap = (Map<String, String>) map.get("dataMap");form.addSubstitutionFont(bf);for (String key : datemap.keySet()) {String value = datemap.get(key);form.setField(key, value);form.setFieldProperty(key, "textsize", 9f, null);}// 图片类的内容处理Map<String, String> imgmap = (Map<String, String>) map.get("imgMap");for (String key : imgmap.keySet()) {String value = imgmap.get(key);String imgpath = value;int pageNo = form.getFieldPositions(key).get(0).page;Rectangle signRect = form.getFieldPositions(key).get(0).position;float x = signRect.getLeft();float y = signRect.getBottom();// 根据路径读取图片Image image = Image.getInstance(imgpath);// 获取图片页面PdfContentByte under = stamper.getOverContent(pageNo);// 图片大小自适应image.scaleToFit(signRect.getWidth(), signRect.getHeight());// 添加图片image.setAbsolutePosition(x, y);under.addImage(image);}// 表格类Map<String, List<List<String>>> listMap = (Map<String, List<List<String>>>) map.get("tableList");if (listMap != null) {Object[] arr = new Object[]{};if (listMap.containsKey("table1")) {List<List<String>> lists1 = listMap.get("table1");if (lists1.size() > 0) {arr = new Object[]{"table", "table1"};} else {arr = new Object[]{"table"};}}else {arr = new Object[]{"table"};arr = new Object[]{"tableInfo"};}
// for (Object key : arr) {List<List<String>> lists = listMap.get("table");List<List<String>> listInfo = listMap.get("tableInfo");List<List<String>> listBy = listMap.get("table1");//判断if (listBy!= null && listBy.size()>0) {//遍历数据添加序号for (int i = 0; i < listBy.size(); i++) {List<String> list = listBy.get(i);
// List<String> list = listBy.get(0);String data = list.get(0);String on = i + 1 + " " + data;
// String on = 0 + 1 + " " + data;list.set(0, on);}int pageNo = 0;if (form.getFieldPositions("table1") == null) {pageNo = 0;} else {pageNo = form.getFieldPositions("table1").get(0).page;}PdfContentByte pcb = stamper.getOverContent(pageNo);List<AcroFields.FieldPosition> listAf = form.getFieldPositions("table1");Rectangle signRect = listAf.get(0).position;//表格位置int columnInfo = listBy.get(0).size();int rowInfo = listBy.size();PdfPTable tableInfo = new PdfPTable(columnInfo);
// float tatalWidth = signRect.getRight() - signRect.getLeft() - 1;tableInfo.setTotalWidth(width);tableInfo.setLockedWidth(true);tableInfo.setKeepTogether(true);tableInfo.setSplitLate(false);tableInfo.setSplitRows(true);Font FontProve = new Font(bf, 8, 0);//表格数据填写for (int k = 0; k < rowInfo; k++) {List<String> listTo = listBy.get(k);
// List<String> listTo = listBy.get(i);for (int j = 0; j < columnInfo; j++) {Paragraph paragraph = new Paragraph(String.valueOf(listTo.get(j)), FontProve);PdfPCell cell = new PdfPCell(paragraph);cell.setBorderWidth(1);cell.setMinimumHeight(1);if (j == 0) {cell.setVerticalAlignment(Element.ALIGN_LEFT);cell.setHorizontalAlignment(Element.ALIGN_LEFT);} else {cell.setVerticalAlignment(Element.ALIGN_CENTER);cell.setHorizontalAlignment(Element.ALIGN_CENTER);}cell.setLeading(0, (float) 1.4);cell.disableBorderSide(15);//隐藏tableInfo.addCell(cell);}}tableInfo.writeSelectedRows(0, -1, signRect.getLeft(), signRect.getTop(), pcb);
// }}if (listInfo!= null && listInfo.size()>0) {//遍历数据添加序号for (int i = 0; i < listInfo.size(); i++) {List<String> list = listInfo.get(i);String data = list.get(0);String on = i + 1 +" " + data ;list.set(0,on);}int pageNo = 0;if (form.getFieldPositions("tableInfo") == null) {pageNo = 0;} else {pageNo = form.getFieldPositions("tableInfo").get(0).page;}PdfContentByte pcb = stamper.getOverContent(pageNo);List<AcroFields.FieldPosition> listAf = form.getFieldPositions("tableInfo");Rectangle signRect = listAf.get(0).position;//表格位置int columnInfo = listInfo.get(0).size();int rowInfo = listInfo.size();PdfPTable tableInfo = new PdfPTable(columnInfo);float tatalWidth = signRect.getRight() - signRect.getLeft() - 1;
//tableInfo.setTotalWidth(tatalWidth);tableInfo.setLockedWidth(true);tableInfo.setKeepTogether(true);tableInfo.setSplitLate(false);tableInfo.setSplitRows(true);Font FontProve = new Font(bf, 8, 0);//表格数据填写for (int i = 0; i < rowInfo; i++) {List<String> list = listInfo.get(i);for (int j = 0; j < columnInfo; j++) {Paragraph paragraph = new Paragraph(String.valueOf(list.get(j)), FontProve);PdfPCell cell = new PdfPCell(paragraph);cell.setBorderWidth(1);cell.setMinimumHeight(1);if (j == 0) {cell.setVerticalAlignment(Element.ALIGN_LEFT);cell.setHorizontalAlignment(Element.ALIGN_LEFT);} else {cell.setVerticalAlignment(Element.ALIGN_CENTER);cell.setHorizontalAlignment(Element.ALIGN_CENTER);}cell.setLeading(0, (float) 1.4);cell.disableBorderSide(15);//隐藏tableInfo.addCell(cell);}}System.out.println(signRect.getLeft());System.out.println(signRect.getTop());System.out.println(pcb);tableInfo.writeSelectedRows(0, -1, signRect.getLeft(), signRect.getTop(), pcb);}if (lists!= null && lists.size()>0) {//遍历数据添加序号for (int i = 0; i < lists.size(); i++) {List<String> list = lists.get(i);String data = list.get(0);String on = i + 1 +" "+ data ;list.set(0,on);}int pageNo = 0;if (form.getFieldPositions("table") == null) {pageNo = 0;} else {pageNo = form.getFieldPositions("table").get(0).page;}PdfContentByte pcb = stamper.getOverContent(pageNo);List<AcroFields.FieldPosition> listAf = form.getFieldPositions("table");Rectangle signRect = listAf.get(0).position;//表格位置int column = lists.get(0).size();int row = lists.size();PdfPTable table = new PdfPTable(column);
// float tatalWidth = signRect.getRight() - signRect.getLeft() - 1;// table.setTotalWidth(width);table.setLockedWidth(true);table.setKeepTogether(true);table.setSplitLate(false);table.setSplitRows(true);Font FontProve = new Font(bf, 8, 0);//表格数据填写for (int i = 0; i < row; i++) {List<String> list = lists.get(i);for (int j = 0; j < column; j++) {Paragraph paragraph = new Paragraph(String.valueOf(list.get(j)), FontProve);PdfPCell cell = new PdfPCell(paragraph);cell.setBorderWidth(1);cell.setMinimumHeight(1);if (j == 0) {cell.setVerticalAlignment(Element.ALIGN_LEFT);cell.setHorizontalAlignment(Element.ALIGN_LEFT);} else {cell.setVerticalAlignment(Element.ALIGN_CENTER);cell.setHorizontalAlignment(Element.ALIGN_CENTER);}cell.setLeading(0, (float) 1.4);cell.disableBorderSide(15);//隐藏table.addCell(cell);}}table.writeSelectedRows(1, -1, signRect.getLeft(), signRect.getTop(), pcb);}}// 如果为false,生成的PDF文件可以编辑,如果为true,生成的PDF文件不可以编辑stamper.setFormFlattening(true);stamper.close();Document doc = new Document();PdfCopy copy = new PdfCopy(doc, out);doc.open();
// File file = new File("ruoyi-admin/src/main/resources/static/pdf/111.pdf");
// File file = new File("D:\\feishu\\yihe\\111.pdf");//本地路径File file = new File("E:\\item\\yihe\\ruoyi-admin\\src\\main\\resources\\static\\pdf\\111.pdf");//线上路径
// File file = new File("D:\\feishu\\yihe\\111.pdf");PdfCopy copy1 = new PdfCopy(doc, new FileOutputStream(file));doc.open();int pageNum = reader.getNumberOfPages();for (int i = 1; i <= pageNum; i++) {PdfImportedPage importPage = copy.getImportedPage(new PdfReader(bos.toByteArray()), i);copy.addPage(importPage);copy1.addPage(importPage);}
// String url = AliyunOSSUtil.upload(getMultipartFile(file));
// System.out.println("url = " + url);doc.close();return "url";}
相关文章:
pdf导出实例(itestpdf)
依赖 <!-- https://mvnrepository.com/artifact/com.itextpdf/itextpdf --><dependency><groupId>com.itextpdf</groupId><artifactId>itextpdf</artifactId><version>5.5.10</version></dependency>工具类 package or…...
关于CW32单片机pack包安装 KEIL IAR
CW32 系列微控制器软件开发工具入门 芯片包 1. 下载芯片包 官方下载链接:武汉鑫源半导体 2. 安装芯片包 双击芯片包.pack文件 支持 CW32F 系列的 IDE 支持 CW32F 系列的工具链: • • EWARM v7.70 或更高版本 MDK-ARM v5.17 或更高版本 2.1 EW…...
memcpy内存拷贝函数
目录 一、memcpy内存拷贝函数 注意事项 二、memcpy与strcpy对比 三、模拟实现memcpy函数 四、memcpy函数不能进行两块存在内存重叠的空间的内存拷贝 五、改进my_memcpy函数 一、memcpy内存拷贝函数 头文件:string.h 函数原型:void* memcpy(void* …...
数组模拟堆
文章目录 QuestionIdeasCode Question 维护一个集合,初始时集合为空,支持如下几种操作: I x,插入一个数 x ; PM,输出当前集合中的最小值; DM,删除当前集合中的最小值(数…...
【深度学习基础知识(一):卷积神经网络CNN基础知识】
深度学习基础知识 深度学习基础知识(一):卷积神经网络CNN基础知识 卷积神经网络CNN基础知识 0、目录 1. CNN卷积神经网络的特点 2. 卷积操作基础知识 2.1 卷积操作的概念2.2 卷积操作的种类2.3 卷积操作后特征图谱大小计算公式 3. 池化操…...
Git使用入门
一、Git简介 Git 是一个开源的分布式版本控制系统。 Git版本控制的功能为保存不同版本的代码,保存代码的地方叫做仓库。 每个仓库中有多个分支,每个分支上又有很多节点,每个节点代表一个版本,不同的分支可以进行合并࿰…...
电机矢量控制算法和例程
电机矢量控制算法是一种高级的电机控制方法,它通过将电机转子空间矢量转换到旋转坐标系中,并在该坐标系中进行控制来实现对电机的精确控制。下面是对电机矢量控制算法的详细解释: 坐标系变换:电机矢量控制首先将电机转子空间矢量变…...
std::string_view概念原理及应用
概念 使用const string&作为参数是先使用字符串字面量编译器会创建一个临时字符串对象然后创建std::string。 或者一个函数提供char*和const string&参数的两个版本函数,不是优雅的解决方案。 于是需要一个只使用内存不维护内存的类。 原理 在visual s…...
lodash库_.chunk、_.pick、_.omit、_.cloneDeep、_.debounce方法
lodash 模块化、高性能的 JavaScript 实用工具库。官方文档:https://www.lodashjs.com 1.对数组进行分组 _.chunk(array, [size1]) 使用场景,如移动端页面一行能放5个元素,总共7条数据,将一维数组转为二维数组,让一个…...
Java使用FFmpeg对视频文件打标记
免安装 FFmpeg <dependency><groupId>ws.schild</groupId><artifactId>jave-all-deps</artifactId><version>3.0.1</version><exclusions><!-- 排除windows 32位系统 --><exclusion><groupId>ws.sch…...
Redux 学习笔记
在使用 React Redux 前,我们首先了解一下 Redux 的一些基础知识。 Redux 是 JavaScript 应用程序中用于状态管理的容器。它不依赖于任何框架,可以与任何 UI 库和框架一起使用。在应用程序中使用 Redux 时,Redux 是以可预测的方式管理状态。 …...
【Bug】8086汇编学习
文章目录 随笔Bug1、masm编译报错:Illegal use of register2、debug中使用段前缀3、[idata]在编译器中的处理4、push立即数报错5、报错:improper operand type6、程序莫名跳转到未知位置 (doing)7、DOSBox失去响应8、程序运行显示乱码9、程序运行导致DOS…...
JetBrains系列IDE全家桶激活
jetbrains全家桶 正版授权,这里有账号授权的渠道: https://www.mano100.cn/thread-1942-1-1.html 附加授权后的一张图片...
洛谷p1618三连击
import java.util.Scanner; //将 1-9 共9个数分成3组,分别组成3个三位数,且使这3个三位数构成A:B:C的比例,试求出所有满足条件的3个三位数。不满足输出“No!!!”。 public class Main {public static void main(String[] args) {Scanner sc …...
微信公众号h5写一个全局调用微信分享功能
1. 首先先安装依赖 npm install weixin-js-sdk --save 2. app.vue文件 <script> export default { onLaunch: function(e) {}, onShow: function(e) { console.log(App Show页面初始); // 路由参数存缓存的 这是为了防止他…...
聊聊精益需求的产生过程
这是鼎叔的第七十八篇原创文章。行业大牛和刚毕业的小白,都可以进来聊聊。 欢迎关注本公众号《敏捷测试转型》,星标收藏,大量原创思考文章陆续推出。本人新书《无测试组织-测试团队的敏捷转型》已出版ÿ…...
Linux - 还不懂 gdb 调试器?(调试软件)
前言 当前,我们可以使用 make/makefile 来程序化执行代码文件;可以使用 gcc/g 等编译器来编译代码;可以使用 vim 编辑器来编写代码;其实在 Linux 当中还有一个工具,可以实现调试工作,这个工具就是 -- gdb。…...
Linux:程序地址空间/虚拟地址等相关概念理解
文章目录 程序地址空间虚拟地址和物理地址地址的转换地址空间是什么? 程序地址空间 在C和C程序中,一直有一个观点是,程序中的各个变量等都会有一定的地址空间,因此才会有诸如取地址,通过地址访问等操作,那…...
Python之爬虫
目录 HTTP请求HTTP响应获得页面响应伪装用户访问打包数据爬取豆瓣top250 HTTP请求 HTTP:HypertextTransferProtcol 超文本传输协议 1、请求行 POST/user/info?new_usertrue HTTP/1.1#资源了路径user/info 查询参数new_usertrue 协议版本HTTP/1.1 2、请求头 Ho…...
打造自己的前端组件库(奶妈版,超详细)
打造自己的前端组件库 demo是开源的,自己上npm 或者 github 上都能搜到 新建vue项目(sass js vue2) vue create yt-ui 修改文件目录(如下) 修改: 1.src 更名 examples; 2. src/components移动到项目最外层;3.vue.config.js更改入口文件 /…...
零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?
一、核心优势:专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发,是一款收费低廉但功能全面的Windows NAS工具,主打“无学习成本部署” 。与其他NAS软件相比,其优势在于: 无需硬件改造:将任意W…...
ubuntu搭建nfs服务centos挂载访问
在Ubuntu上设置NFS服务器 在Ubuntu上,你可以使用apt包管理器来安装NFS服务器。打开终端并运行: sudo apt update sudo apt install nfs-kernel-server创建共享目录 创建一个目录用于共享,例如/shared: sudo mkdir /shared sud…...
Appium+python自动化(十六)- ADB命令
简介 Android 调试桥(adb)是多种用途的工具,该工具可以帮助你你管理设备或模拟器 的状态。 adb ( Android Debug Bridge)是一个通用命令行工具,其允许您与模拟器实例或连接的 Android 设备进行通信。它可为各种设备操作提供便利,如安装和调试…...
土地利用/土地覆盖遥感解译与基于CLUE模型未来变化情景预测;从基础到高级,涵盖ArcGIS数据处理、ENVI遥感解译与CLUE模型情景模拟等
🔍 土地利用/土地覆盖数据是生态、环境和气象等诸多领域模型的关键输入参数。通过遥感影像解译技术,可以精准获取历史或当前任何一个区域的土地利用/土地覆盖情况。这些数据不仅能够用于评估区域生态环境的变化趋势,还能有效评价重大生态工程…...
C# SqlSugar:依赖注入与仓储模式实践
C# SqlSugar:依赖注入与仓储模式实践 在 C# 的应用开发中,数据库操作是必不可少的环节。为了让数据访问层更加简洁、高效且易于维护,许多开发者会选择成熟的 ORM(对象关系映射)框架,SqlSugar 就是其中备受…...
管理学院权限管理系统开发总结
文章目录 🎓 管理学院权限管理系统开发总结 - 现代化Web应用实践之路📝 项目概述🏗️ 技术架构设计后端技术栈前端技术栈 💡 核心功能特性1. 用户管理模块2. 权限管理系统3. 统计报表功能4. 用户体验优化 🗄️ 数据库设…...
用机器学习破解新能源领域的“弃风”难题
音乐发烧友深有体会,玩音乐的本质就是玩电网。火电声音偏暖,水电偏冷,风电偏空旷。至于太阳能发的电,则略显朦胧和单薄。 不知你是否有感觉,近两年家里的音响声音越来越冷,听起来越来越单薄? —…...
使用Spring AI和MCP协议构建图片搜索服务
目录 使用Spring AI和MCP协议构建图片搜索服务 引言 技术栈概览 项目架构设计 架构图 服务端开发 1. 创建Spring Boot项目 2. 实现图片搜索工具 3. 配置传输模式 Stdio模式(本地调用) SSE模式(远程调用) 4. 注册工具提…...
计算机基础知识解析:从应用到架构的全面拆解
目录 前言 1、 计算机的应用领域:无处不在的数字助手 2、 计算机的进化史:从算盘到量子计算 3、计算机的分类:不止 “台式机和笔记本” 4、计算机的组件:硬件与软件的协同 4.1 硬件:五大核心部件 4.2 软件&#…...
Caliper 负载(Workload)详细解析
Caliper 负载(Workload)详细解析 负载(Workload)是 Caliper 性能测试的核心部分,它定义了测试期间要执行的具体合约调用行为和交易模式。下面我将全面深入地讲解负载的各个方面。 一、负载模块基本结构 一个典型的负载模块(如 workload.js)包含以下基本结构: use strict;/…...
