基于Spring的三方平台接口对接方法(OkHttp/RestTemplate/视图)

      本文介绍了三方平台接口对接方法,一是基于OkHttp请求工具及dom4j报文封装解析xml的方法,二是采用RestTemplate方法封装请求,三是采用建立视图和从库数据源的方式查询。

一、OkHttp请求工具及dom4j报文封装解析

1、 依赖引入

<!-- okhttp3包 --><dependency><groupId>com.squareup.okhttp3</groupId><artifactId>okhttp</artifactId><version>4.9.3</version><scope>compile</scope></dependency>
<dependency><groupId>org.dom4j</groupId><artifactId>dom4j</artifactId><version>2.1.4</version></dependency>

2、OkHttp请求工具类

package com.inspur.common.utils.okhttp;import okhttp3.*;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import java.io.File;import java.io.IOException;import java.util.Map;import java.util.concurrent.TimeUnit;/*** OkHttp封装,调用示例OkHttpUtil.postJson(...)*/public class OkHttpUtil {private static final Logger logger = LoggerFactory.getLogger(OkHttpUtil.class);private static final byte[] LOCKER = new byte[0];private static volatile OkHttpClient okHttpClient;private static final MediaType JSON = MediaType.parse("application/json; charset=utf-8");private static final MediaType XML = MediaType.parse("application/xml; charset=utf-8");private OkHttpUtil() {}private static OkHttpClient getOkHttpClient() {if (okHttpClient == null) {synchronized (LOCKER) {if (okHttpClient == null) {OkHttpClient.Builder ClientBuilder = new OkHttpClient.Builder();ClientBuilder.readTimeout(30, TimeUnit.SECONDS);//读取超时ClientBuilder.connectTimeout(30, TimeUnit.SECONDS);//连接超时ClientBuilder.writeTimeout(60, TimeUnit.SECONDS);//写入超时okHttpClient = ClientBuilder.build();}}}return okHttpClient;}/*** 发送get请求** @param url* @param json* @return* @throws IOException*/public static Result<String> get(String url) throws IOException {logger.debug("请求url:{}", url);Request request = new Request.Builder().url(url).get().build();return execute(request);}/*** 发送get请求,带参数** @param url* @param json* @return* @throws IOException*/public static Result<String> get(String url, Map<String, String> params) throws IOException {logger.debug("请求url:{}", url);HttpUrl.Builder urlBuilder = HttpUrl.parse(url).newBuilder();for (Map.Entry<String, String> entry : params.entrySet()) {//追加表单信息urlBuilder.addQueryParameter(entry.getKey(), entry.getValue());}Request request = new Request.Builder().url(urlBuilder.build()).build();return execute(request);}/*** 发送json请求** @param url* @param json* @return* @throws IOException*/public static Result<String> postJson(String url, String json) throws IOException {logger.debug("请求url:{},请求体:{}", url, json);RequestBody requestBody = RequestBody.create(json,JSON);Request request = new Request.Builder().url(url).post(requestBody).build();return execute(request);}public static Result<String> postXml(String url, String xml) throws IOException {RequestBody requestBody = RequestBody.create(xml, XML);Request request = new Request.Builder().url(url).post(requestBody).build();return execute(request);}/*** 发送form表单post请求** @param url* @param params* @return* @throws IOException*/public static Result<String> post(String url, Map<String, String> params) throws IOException {logger.debug("请求地址:{};", url);//创建一个FormBody.BuilderFormBody.Builder builder = new FormBody.Builder();for (Map.Entry<String, String> entry : params.entrySet()) {//追加表单信息builder.add(entry.getKey(), entry.getValue());}//生成表单实体对象RequestBody formBody = builder.build();Request request = new Request.Builder().url(url).post(formBody).build();return execute(request);}/*** 发送form表单post请求** @param url* @param params* @return* @throws IOException*/public static Result<String> postWithHeader(String url, Map<String, String> params, String... headers) throws IOException {logger.debug("请求url:{};", url);//创建一个FormBody.BuilderFormBody.Builder builder = new FormBody.Builder();for (Map.Entry<String, String> entry : params.entrySet()) {//追加表单信息builder.add(entry.getKey(), entry.getValue());}//生成表单实体对象RequestBody formBody = builder.build();Request request = new Request.Builder().url(url).headers(Headers.of(headers)).post(formBody).build();return execute(request);}/*** 发送form表单post请求** @param url* @param params* @return* @throws IOException*/public static Result<String> postFile(String url, String fileName, File file, Map<String, String> params) throws IOException {logger.debug("请求url:{};", url);MultipartBody.Builder builder = new MultipartBody.Builder();builder.addFormDataPart(fileName, fileName, RequestBody.create(MediaType.parse("application/octet-stream"), file));if (params != null) {for (Map.Entry<String, String> entry : params.entrySet()) {//追加表单信息builder.addFormDataPart(entry.getKey(), entry.getValue());}}MultipartBody multipartBody = builder.build();Request request = new Request.Builder().url(url).post(multipartBody).build();return execute(request);}private static Result<String> execute(Request request) {Result<String> result = new Result();try (Response response = OkHttpUtil.getOkHttpClient().newCall(request).execute()) {logger.debug("{} 请求结果:{}", request.url().url(), response);if (response.isSuccessful()) {ResponseBody body = response.body();if (body != null) {result = Result.ok(response.message(), body.string());}else {result = Result.ok(response.message());}}else {result = Result.fail(response.code(),response.message());}} catch (Exception e) {result = Result.fail(e.getMessage());}return result;}}

3、基于dom4j解析和封装xml

(1)xml文件报文示例

/admin-module/src/main/resources/nc/classStudent.xml

<?xml version="1.0" encoding='UTF-8'?>
<ufinterface account="tpy63_wzx0731" billtype="4D" filename="" groupcode="" isexchange="Y" replace="Y" roottag="" sender="gyhlw">
<classInfo><school>市第一中学</school><city>济南</city><location>2楼</location>
</classInfo>
<students><student name="zhangsan" age="18"></student><student name="lisi" age="19"></student>
</students></ufinterface>
(2)构建封装xml报文文件
public String buildXml(){// 创建saxReader对象SAXReader reader = new SAXReader();try (InputStream classInfo = new ClassPathResource("nc/classStudent.xml").getInputStream()) {// 通过read方法读取一个文件 转换成Document对象Document document = reader.read(classInfo);// 获取根节点元素对象Element node = document.getRootElement();// 根节点下的一级子节点Element classInfo = node.element("classInfo");   // 根节点下的子节点 classInfoElement students = node.element("students");   // 根节点下的子节点 students// 一级子节点classInfo 下属二级节点students.element("school").setText("市第一中学");students.element("city").setText("济南");students.element("location").setText("2楼");// 一级子节点students 下属二级节点// 以此添加成List并设置属性List<Student> st = ArrayList<T> obj = new ArrayList<Student>() {{add(new Student("zhangsan",18));add(new Student("lisi",19));}};students.stream().forEach(student -> {root.addElement("student").addAttribute("name", student.getName()).addAttribute("age", String.valueOf(student.getAge()))});// 生成xml对象String xml = document.asXML();logger.info("生成XML:\n{} ", xml);}}

(3)okHttp三方接口请求
(4)xml报文读取解析
public Object handlerReturn() {try {/*   直接读取文件解析// 创建SAXReaderSAXReader saxReader = SAXReader.createDefault();// 将xml解析为树Document document = saxReader.read("xml/students.xml");*/String xml = this.buildXml();// 三方接口请求地址String url = "http://110.10.1.11:8888/service/xxx"// 1、请求发送Result<String>  result = OkHttpUtil.postXml(url, xml);if (result.isOk()) {String resMess = result.getData();logger.info("调用结果:{}", resMess);// resMess实际为classStudent.xml文件格式内容Document respXml = DocumentHelper.parseText(resMess);Element rootElement = respXml.getRootElement();// 2、获取根节点下的一级子节点Element classInfo = rootElement.element("classInfo");   // 根节点下的子节点 classInfoElement students = rootElement.element("students");   // 根节点下的子节点 students// 3、获取一级子节点classInfo 下属二级节点属性String school = classInfo.elementText("school");String city = classInfo.elementText("city");String location = classInfo.elementText("location");// 4、获取一级子节点students,遍历下属二级节点student列表并获取属性值List<Element> eles = students.elements();List<Student> students = new ArrayList<>();// 遍历子节点列表for(Element ele : eles){// 读取节点属性String name = element.attributeValue("name");Interger age = element.attributeValue("age");Student stu = new Student(name,age);students.add(stu)}}} catch (DocumentException | IOException e) {logger.error("调用接口异常", e);}// 5、解析结果值构建对应实体类或mapHashMap resObj = new HashMap<>();resObj.put("students",students);resObj.put("school",school);return resObj;}

二、RestTemplate

OutInterfaceDomain 实体类根据接口规范自行组装定义

1、依赖

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency>

2、使用

import org.springframework.http.HttpEntity;
import org.springframework.http.HttpHeaders;
import org.springframework.http.MediaType;import org.springframework.web.client.RestTemplate;public class OutInterfaceService {@Autowired
private RestTemplate restTemplate;private String url = "http://110.1.1.129:8087/DataStandard/*****";HttpHeaders httpHeaders = new HttpHeaders();
httpHeaders.setContentType(MediaType.APPLICATION_JSON_UTF8);// 要传递的业务内容Map<String, Object> columnMap = new HashMap<>();columnMap.put("DEVICE_NO", factoryDevice.getDeviceNo());// 按接口规范组装OutInterfaceDomain outInterfaceDomain = new OutInterfaceDomain();outInterfaceDomain.setColumns(columnMap);String contentJson = JSON.toJSONString(outInterfaceDomain, SerializerFeature.WriteMapNullValue);
HttpEntity<String> httpEntity = new HttpEntity<>(contentJson, httpHeaders);// 调用三方接口restTemplate.postForEntity(url, httpEntity, String.class);}

三、sql视图

在原表建立视图,并创建新用户,赋予只读权限。使用@DataSource来切换数据库查询。

1、创建视图和用户并赋予权限

grant select on [view] to [username];//查询视图权限grant connect to [username];//连接数据库权限   # 创建设备视图create or replace view factory_device_view asselect * from factory_device;comment on table factory_device_view is '设备视图';SELECT * FROM all_tab_cols WHERE table_name ='FACTORY_DEVICE_VIEW';# 创建用户create user jtgViewer identified by password123$# 赋予权限GRANT SELECT ON jtg.factory_device_view TO jtgViewer; grant connect to jtgViewer

2、配置文件

# 从库数据源
slave:# 从数据源开关/默认关闭enabled: trueurl: jdbc:mysql://192.168.100.200:3306/esis_tjzg?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8&rewriteBatchedStatements=trueusername: rootpassword: Inspur@2023

3、查询使用

@DataSource(DataSourceType.SLAVE)public List<Map<String,Object>> selectOutAmountByMaterial(TImMaterialrequestbill tImMaterialrequestbill);

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

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

相关文章

CleanClip for Mac v2.2.0 剪贴板历史管理软件正式激活版

CleanClip 是一款专为 Mac 用户设计的强大剪贴板历史管理工具。它能够自动保存您复制的内容,让您轻松访问和管理剪贴板历史记录,大大提高工作效率。 下载地址&#xff1a;CleanClip for Mac v2.2.0 剪贴板历史管理软件正式激活版 主要特点 自动保存剪贴板历史 CleanClip 会自…

SOMEIP_ETS_074: Wrong_Interface_Version

测试目的&#xff1a; 验证当设备&#xff08;DUT&#xff09;接收到一个包含错误接口版本的SOME/IP请求时&#xff0c;是否能够返回错误消息或忽略该请求。 描述 本测试用例旨在检查DUT在处理一个echoUINT8方法的SOME/IP消息时&#xff0c;如果消息中包含的接口版本不正确&…

[ACP云计算]易错题(原题)

ECS 1、 2、 3、 4、 5、 6、 7、 8、 9、 10、 11、 12、 13、 14、 15、 16、 17、 18、 19、 20、 21、 22、 23、 24、 25、 26、 27、 28、 29、 30、 31、 32、 33、 34、 35、 36、 37、 对象存储OSS 1、 2、 3、 4、 5、 6、 重点&#xff01;&#xff01;&#xff…

javaee、ssm(maven)、springboot(maven)项目目录结构以及编译后文件目录存放路径

javaee项目目录结构&#xff1a; src下的文件或者是源码编译后都会放在WebRoot&#xff08;项目根目录&#xff09;文件夹\WebRoot\WEB-INF\classes目录中。 编译后的文件夹目录如下&#xff1a; 以上为普通的javaee项目目录结构&#xff0c;同maven工程目录结构是不一样的。…

Electron 项目实战 03: 实现一个截图功能

实现效果 实现思路 创建两个window&#xff0c;一个叫mainWindow&#xff0c;一个叫cutWindowmainWindow&#xff1a;主界面用来展示截图结果cutWindow&#xff1a;截图窗口&#xff0c;加载截图页面和截图交互逻辑mainWindow 页面点击截图&#xff0c;让cutWIndow 来实现具体…

实习项目|苍穹外卖|day1

碎碎念 眨眼间&#xff0c;留给自己的时间不多了。想要去好的公司实习&#xff0c;现在是八股不会背&#xff0c;算法题全忘&#xff0c;跟了好多教程&#xff0c;也没有能写上简历的项目。因此&#xff0c;我决定用两个月的时间学习两个能够写到简历上的项目&#xff08;的确…

Android Studio:模拟器页面闪烁,手机模拟器输入画面闪烁 android studio闪屏

主要解决&#xff0c;android studio 启动app测试&#xff0c;输入数据时&#xff0c;手机画面就会闪烁&#xff0c;闪屏 1. 如图所示&#xff0c;依照顺序找到Edit &#xff0c;并点击Edit 2. 找到Graphics 选择为SoftWare &#xff0c;并保存修改即可 3. 如果此处不能选择S…

国内AI工具分类大盘点,这些神器你都用过了吗?

AI爆发到现成已经快2年了&#xff0c;基本上我自己也使用了近2年的AI产品。国内、外的AI产品体验了很多。 从最初文本聊天类的gpt、new bing、文心一言、通义千问&#xff0c;到后面绘图类Midjourney、Stable Diffusion、文心一格、通义万相等等。 在这里来分享我自己使用的一…

jmeter连接mysql数据库以及常规用法

1、在jmeter中新建一个测试计划&#xff0c;在测试计划界面中点击浏览&#xff0c;选择连接mysql数据库的jar包 如果没有jar包可以去网上下载&#xff0c;也可以通过如下链接进行下载 链接: https://pan.baidu.com/s/1BI6f19KSzXGlkSOwbnequw 提取码: gn8e 2、然后创建线程组&a…

SpringBoot日常:Spring之@PostConstruct解析

简介 spring的Bean在创建的时候会进行初始化&#xff0c;而初始化过程会解析出PostConstruct注解的方法&#xff0c;并反射调用该方法。 PostConstruct 的使用和特点 只有一个非静态方法能使用此注解&#xff1b;被注解的方法不得有任何参数&#xff1b;被注解的方法返回值必…

npm登录

npm 登录问题 npm login --auth-typelegacy报错 原因 npm源非npm本身源&#xff0c;需要切换&#xff1a; 查看源 nrm ls切换为npm源 nrm use npm重新登录 输入OTP验证后登录成功&#xff1a;

python从入门到精通:数据可视化-图形开发

1、json数据格式 • Json是一种轻量级的数据交互格式&#xff0c;可以按照Json指定的格式去组织和封装数据 • Json本质上是一种带有特殊格式的字符串 主要功能&#xff1a;json就是一种在各个编程语言中流通的数据格式&#xff0c;负责不同编程语言中的数据传递和交互&#xf…

【DSP+FPGA】基于2 个TMS320C6678+ XC7VX690T FPGA 的6U VPX 总线架构的高性能实时信号处理平台

6U VPX架构&#xff0c;符合VITA46规范板载 2 个TMS320C6678 多核DSP处理节点板载 1 片 XC7VX690T FPGA处理节点板载 2 个FMC 接口背板之间具有 4 路 x4 高速 GTH 互联&#xff0c;支持RapidIO、PCI ExpressFPGA 与 DSP 之间采用高速Rapid IO互联 基于6U VPX架构的高性能实时信…

BERT:Pre-training of Deep Bidirectional Transformers forLanguage Understanding

个人觉着BERT是一篇读起来很爽的论文 摘要 我们引入了一种新的语言表示模型BERT&#xff0c;它代表Bidirectional Encoder Representations from Transformers。与最近的语言表示模型不同(Peters et al.&#xff0c; 2018a;Radford et al.&#xff0c; 2018)&#xff0c; BER…

组合式API-reactive和ref函数,computed计算属性,watch函数

一.reactive&#xff08;&#xff09;接收一个对象类型的数据&#xff0c;返回一个响应式的对象&#xff1a; <script setup> import {reactive} from vue const state reactive({count:100 }) const setCount () > {state.count } </script> <template>…

书生大模型实战营第三期进阶岛第三课——LMDeploy 量化部署实践

LMDeploy 量化部署实践 任务一&#xff1a;创建虚拟环境创建文件夹LMDEPLOY用于存放课程相关的文件创建模型软连接LMDeploy验证启动模型文件LMDeploy API部署InternLM2.5-1.8b以命令行形式连接API服务器以Gradio网页形式连接API服务器LMDeploy KV量化W4A16 模型量化和部署W4A16…

create-react-app 移除 ESLint 语法检查

ESLint 的作用&#xff1a; ESLint 是一个流行的 JavaScript 代码静态检查工具&#xff0c;旨在帮助开发者识别和修复代码中的问题。以下是关于 ESLint 的一些关键信息&#xff1a; 主要功能&#xff1a; 1.代码风格检查&#xff1a;ESLint 可以检查代码是否符合特定的编码风…

经典算法之链表篇(二)

目录 一&#xff1a;重排链表&#xff08;LeetCode.143&#xff09; 二&#xff1a;删除链表的节点&#xff08;LCR 136. 删除链表的节点&#xff09; 三&#xff1a;K个一组反转链表&#xff08;LeetCode.25&#xff09; 有关经典算法链表的第一篇内容&#xff0c;可以查看我…

ESXi服务器无法安装Windows11:“不符合此版本的Windows所需最低系统要求“

目录 一、问题描述1.使用环境2.问题截图3.问题解析 二、解决方法Ⅰ1.按 ShiftF10 弹出命令提示符2.在弹出的Dos框中输入regedit&#xff0c;回车&#xff0c;进入注册表。3.打开HKEY_LOCAL_MACHINE\SYSTEM\Setup&#xff0c;并新建 LabConfig 的项&#xff0c;在 LabConfig 下创…

使用预训练的 ONNX 格式的 YOLOv8n 模型进行目标检测,并在图像上绘制检测结果

目录 __init__方法&#xff1a; pre_process方法&#xff1a; run方法&#xff1a; filter_boxes方法&#xff1a; view_img方法&#xff1a; __init__方法&#xff1a; 初始化类的实例时&#xff0c;创建一个onnxruntime的推理会话&#xff0c;加载名为yolov8n.onnx的模型…