讯飞星火认知大模型Java后端接口

文章目录

  • 1.免费申请星火大模型套餐
  • 2.Java后端接口说明
    • 2.1 项目地址
    • 2.2 项目说明
    • 2.3 项目结构
    • 2.4 项目代码
      • 🍀 maven 依赖
      • 🍀 application.yml 配置文件
      • 🍀 config 包
        • 📌 XfXhConfig
      • 🍀 dto 包
        • 📌 MsgDTO
        • 📌 RequestDTO
        • 📌 ResponseDTO
      • 🍀 listener 包
        • 📌 XfXhWebSocketListener
      • 🍀 component 包
        • 📌 XfXhStreamClient
      • 🍀 controller 包
        • 📌 TestController
      • 🍀 XfXhApplication 启动类

1.免费申请星火大模型套餐

🏠 讯飞星火认知大模型-AI大语言模型-产品套餐

image-20230918131323194

我们选择 “免费试用”。

image-20230918131434278

点击 “确认下单”即可,注意这个免费套餐一个用户只能购买一次。

购买完毕,我们进入控制台界面查看应用信息:讯飞应用控制台

image-20230918131738113

image-20230918132120461

这个“服务接口认证信息”中的 APPIDAPISecretAPIKey 就是我们需要在 Java 后端配置文件 application.yml 中配置中的信息。

2.Java后端接口说明

2.1 项目地址

🏠 Mr-Write/SpringbootDemo: 各种demo案例 (github.com)

我已经写了一个比较完整的 SpringBoot Demo 项目,为了方便理解对代码做了详细注释,已开源在 GitHub。

下载后只需要找到 xfxh-web-simple-demo 模块,在其 application.yml 文件配置你自己的 APPIDAPISecretAPIKey 信息,再以 GET 方式访问接口:http://localhost:8080/test/sendQuestion?question=hello

2.2 项目说明

该后端接口的大致实现逻辑:

  1. 以 GET 方式访问 SpringBoot 后端接口;
  2. 根据你的配置信息生成通用鉴权URL,并携带 question 建立 websocket 连接
  3. 星火大模型流式返回生成的回答;
  4. 当大模型返回给后端的响应中出现 已返回全部回答的标识status 后,后端关闭 websocket 连接;
  5. 后端将生成的完整回答响应给接口调用者。

如果你想了解更详细的与星火大模型之间的参数说明,请参考 星火认知大模型Web文档

该项目后端接口的实现功能:

  • 能回答单个问题,但不支持上下文;
  • 对星火大模型限制的 QPS 做了处理;
  • 通过配置文件可以规定大模型回复问题的最大响应时长;

如果想要使用支持上下文的接口,只需要找到 xfxh-web-support-context-demo 模块,它在 xfxh-web-simple-demo 模块基础上实现了基于上下文的回答,该增强模块的后端接口说明:

  • 将上下文内容信息保存到了内存中,可以通过配置文件设置保存的上下文内容条数以及用户信息数;
  • 一份交互记录指的是两条上下文内容,分别是用户的问题和大模型的回答;
  • 支持了唯一标识的用户必须先等他的上一条问题的回答生成才能发送新的问题;
  • 由于信息存储在内存中,因此设置了定时任务检查用户是否过期并移除,这个过期时间可以在配置文件中设置。

代码还是易懂的,如果想了解如何实现的,建议先看完有完整注释的 xfxh-web-simple-demo 模块,再去看 xfxh-web-support-context-demo 模块。xfxh-web-support-context-demo 模块只是在 xfxh-web-simple-demo 模块进行了补充/增强。

2.3 项目结构

SpringBoot Demo 项目的结构:

image-20230921225551637

2.4 项目代码

🍀 maven 依赖

<parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.7.0</version><relativePath/> <!-- lookup parent from repository -->
</parent>
<properties><maven.compiler.source>8</maven.compiler.source><maven.compiler.target>8</maven.compiler.target><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-configuration-processor</artifactId><optional>true</optional></dependency><dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId><version>5.8.18</version></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>1.2.67</version></dependency><dependency><groupId>org.java-websocket</groupId><artifactId>Java-WebSocket</artifactId><version>1.3.8</version></dependency><dependency><groupId>com.squareup.okhttp3</groupId><artifactId>okhttp</artifactId><version>4.10.0</version></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId></dependency>
</dependencies>

🍀 application.yml 配置文件

xfxh:# 服务引擎使用 讯飞星火认知大模型V2.0,如果使用 V1.5 需要将 hostUrl 修改为 https://spark-api.xf-yun.com/v1.1/chathostUrl: https://spark-api.xf-yun.com/v2.1/chat# 发送请求时指定的访问领域,如果是 V1.5版本 设置为 general,如果是 V2版本 设置为 generalv2domain: generalv2# 核采样阈值。用于决定结果随机性,取值越高随机性越强即相同的问题得到的不同答案的可能性越高。取值 [0,1]temperature: 0.5# 模型回答的tokens的最大长度,V1.5取值为[1,4096],V2.0取值为[1,8192]。maxTokens: 2048# 大模型回复问题的最大响应时长,单位 smaxResponseTime: 30# 允许同时连接大模型的 websocket 数,如果是普通(免费)用户为 2,超过这个数连接响应会报错,具体参考官网。QPS: 2# 用于权限验证,从服务接口认证信息中获取appId:# 用于权限验证,从服务接口认证信息中获取apiKey:# 用于权限验证,从服务接口认证信息中获取apiSecret:

🍀 config 包

📌 XfXhConfig
/*** @author 狐狸半面添* @create 2023-09-15 0:46*/
@Configuration
@ConfigurationProperties(prefix = "xfxh")
@Data
public class XfXhConfig {/*** 服务引擎使用 讯飞星火认知大模型V2.0,如果使用 V1.5 需要将 hostUrl 修改为 https://spark-api.xf-yun.com/v1.1/chat*/private String hostUrl;/*** 发送请求时指定的访问领域,如果是 V1.5版本 设置为 general,如果是 V2版本 设置为 generalv2*/private String domain;/*** 核采样阈值。用于决定结果随机性,取值越高随机性越强即相同的问题得到的不同答案的可能性越高。取值 [0,1]*/private Float temperature;/*** 模型回答的tokens的最大长度,V1.5取值为[1,4096],V2.0取值为[1,8192]。*/private Integer maxTokens;/*** 大模型回复问题的最大响应时长,单位 s*/private Integer maxResponseTime;/*** 用于权限验证,从服务接口认证信息中获取*/private String appId;/*** 用于权限验证,从服务接口认证信息中获取*/private String apiKey;/*** 用于权限验证,从服务接口认证信息中获取*/private String apiSecret;}

🍀 dto 包

📌 MsgDTO
/*** 消息对象** @author 狐狸半面添* @create 2023-09-15 0:42*/
@Data
@AllArgsConstructor
@NoArgsConstructor
@JsonInclude(JsonInclude.Include.NON_NULL)
public class MsgDTO {/*** 角色*/private String role;/*** 消息内容*/private String content;/*** 响应结果字段:结果序号,取值为[0,10]; 当前为保留字段,开发者可忽略*/private Integer index;public static final String ROLE_USER = "user";public static final String ROLE_ASSISTANT = "assistant";public static MsgDTO createUserMsg(String content) {return new MsgDTO(ROLE_USER, content, null);}public static MsgDTO createAssistantMsg(String content) {return new MsgDTO(ROLE_ASSISTANT, content, null);}}
📌 RequestDTO
/*** 请求参数* 对应生成的 JSON 结构参考 resources/demo-json/request.json** @author 狐狸半面添* @create 2023-09-15 0:42*/
@NoArgsConstructor
@Data
public class RequestDTO {@JsonProperty("header")private HeaderDTO header;@JsonProperty("parameter")private ParameterDTO parameter;@JsonProperty("payload")private PayloadDTO payload;@NoArgsConstructor@Data@AllArgsConstructorpublic static class HeaderDTO {/*** 应用appid,从开放平台控制台创建的应用中获取*/@JSONField(name = "app_id")private String appId;/*** 每个用户的id,用于区分不同用户,最大长度32*/@JSONField(name = "uid")private String uid;}@NoArgsConstructor@Data@AllArgsConstructorpublic static class ParameterDTO {private ChatDTO chat;@NoArgsConstructor@Data@AllArgsConstructorpublic static class ChatDTO {/*** 指定访问的领域,general指向V1.5版本 generalv2指向V2版本。注意:不同的取值对应的url也不一样!*/@JsonProperty("domain")private String domain;/*** 核采样阈值。用于决定结果随机性,取值越高随机性越强即相同的问题得到的不同答案的可能性越高*/@JsonProperty("temperature")private Float temperature;/*** 模型回答的tokens的最大长度*/@JSONField(name = "max_tokens")private Integer maxTokens;}}@NoArgsConstructor@Data@AllArgsConstructorpublic static class PayloadDTO {@JsonProperty("message")private MessageDTO message;@NoArgsConstructor@Data@AllArgsConstructorpublic static class MessageDTO {@JsonProperty("text")private List<MsgDTO> text;}}
}
📌 ResponseDTO
/*** 返回参数* 对应生成的 JSON 结构参考 resources/demo-json/response.json** @author 狐狸半面添* @create 2023-09-15 0:42*/
@NoArgsConstructor
@Data
public class ResponseDTO {@JsonProperty("header")private HeaderDTO header;@JsonProperty("payload")private PayloadDTO payload;@NoArgsConstructor@Datapublic static class HeaderDTO {/*** 错误码,0表示正常,非0表示出错*/@JsonProperty("code")private Integer code;/*** 会话是否成功的描述信息*/@JsonProperty("message")private String message;/*** 会话的唯一id,用于讯飞技术人员查询服务端会话日志使用,出现调用错误时建议留存该字段*/@JsonProperty("sid")private String sid;/*** 会话状态,取值为[0,1,2];0代表首次结果;1代表中间结果;2代表最后一个结果*/@JsonProperty("status")private Integer status;}@NoArgsConstructor@Datapublic static class PayloadDTO {@JsonProperty("choices")private ChoicesDTO choices;/*** 在最后一次结果返回*/@JsonProperty("usage")private UsageDTO usage;@NoArgsConstructor@Datapublic static class ChoicesDTO {/*** 文本响应状态,取值为[0,1,2]; 0代表首个文本结果;1代表中间文本结果;2代表最后一个文本结果*/@JsonProperty("status")private Integer status;/*** 返回的数据序号,取值为[0,9999999]*/@JsonProperty("seq")private Integer seq;/*** 响应文本*/@JsonProperty("text")private List<MsgDTO> text;}@NoArgsConstructor@Datapublic static class UsageDTO {@JsonProperty("text")private TextDTO text;@NoArgsConstructor@Datapublic static class TextDTO {/*** 保留字段,可忽略*/@JsonProperty("question_tokens")private Integer questionTokens;/*** 包含历史问题的总tokens大小*/@JsonProperty("prompt_tokens")private Integer promptTokens;/*** 回答的tokens大小*/@JsonProperty("completion_tokens")private Integer completionTokens;/*** prompt_tokens和completion_tokens的和,也是本次交互计费的tokens大小*/@JsonProperty("total_tokens")private Integer totalTokens;}}}
}

🍀 listener 包

📌 XfXhWebSocketListener
/*** @author 狐狸半面添* @create 2023-09-15 1:11*/
@Slf4j
public class XfXhWebSocketListener extends WebSocketListener {private StringBuilder answer = new StringBuilder();private boolean wsCloseFlag = false;public StringBuilder getAnswer() {return answer;}public boolean isWsCloseFlag() {return wsCloseFlag;}@Overridepublic void onOpen(@NotNull WebSocket webSocket, @NotNull Response response) {super.onOpen(webSocket, response);}@Overridepublic void onMessage(@NotNull WebSocket webSocket, @NotNull String text) {super.onMessage(webSocket, text);// 将大模型回复的 JSON 文本转为 ResponseDTO 对象ResponseDTO responseData = JSONObject.parseObject(text, ResponseDTO.class);// 如果响应数据中的 header 的 code 值不为 0,则表示响应错误if (responseData.getHeader().getCode() != 0) {// 日志记录log.error("发生错误,错误码为:" + responseData.getHeader().getCode() + "; " + "信息:" + responseData.getHeader().getMessage());// 设置回答this.answer = new StringBuilder("大模型响应错误,请稍后再试");// 关闭连接标识wsCloseFlag = true;return;}// 将回答进行拼接for (MsgDTO msgDTO : responseData.getPayload().getChoices().getText()) {this.answer.append(msgDTO.getContent());}// 对最后一个文本结果进行处理if (2 == responseData.getHeader().getStatus()) {wsCloseFlag = true;}}@Overridepublic void onFailure(@NotNull WebSocket webSocket, @NotNull Throwable t, @Nullable Response response) {super.onFailure(webSocket, t, response);}@Overridepublic void onClosed(@NotNull WebSocket webSocket, int code, @NotNull String reason) {super.onClosed(webSocket, code, reason);}
}

🍀 component 包

📌 XfXhStreamClient
/*** @author 狐狸半面添* @create 2023-09-15 1:10*/
@Component
@Slf4j
public class XfXhStreamClient {@Resourceprivate XfXhConfig xfXhConfig;@Value("${xfxh.QPS}")private int connectionTokenCount;/*** 获取令牌*/public static int GET_TOKEN_STATUS = 0;/*** 归还令牌*/public static int BACK_TOKEN_STATUS = 1;/*** 操作令牌** @param status 0-获取令牌 1-归还令牌* @return 是否操作成功*/public synchronized boolean operateToken(int status) {if (status == GET_TOKEN_STATUS) {// 获取令牌if (connectionTokenCount != 0) {// 说明还有令牌,将令牌数减一connectionTokenCount -= 1;return true;} else {return false;}} else {// 放回令牌connectionTokenCount += 1;return true;}}/*** 发送消息** @param uid     每个用户的id,用于区分不同用户* @param msgList 发送给大模型的消息,可以包含上下文内容* @return 获取websocket连接,以便于我们在获取完整大模型回复后手动关闭连接*/public WebSocket sendMsg(String uid, List<MsgDTO> msgList, WebSocketListener listener) {// 获取鉴权urlString authUrl = this.getAuthUrl();// 鉴权方法生成失败,直接返回 nullif (authUrl == null) {return null;}OkHttpClient okHttpClient = new OkHttpClient.Builder().build();// 将 https/http 连接替换为 ws/wss 连接String url = authUrl.replace("http://", "ws://").replace("https://", "wss://");Request request = new Request.Builder().url(url).build();// 建立 wss 连接WebSocket webSocket = okHttpClient.newWebSocket(request, listener);// 组装请求参数RequestDTO requestDTO = getRequestParam(uid, msgList);// 发送请求webSocket.send(JSONObject.toJSONString(requestDTO));return webSocket;}/*** 生成鉴权方法,具体实现不用关心,这是讯飞官方定义的鉴权方式** @return 鉴权访问大模型的路径*/public String getAuthUrl() {try {URL url = new URL(xfXhConfig.getHostUrl());// 时间SimpleDateFormat format = new SimpleDateFormat("EEE, dd MMM yyyy HH:mm:ss z", Locale.US);format.setTimeZone(TimeZone.getTimeZone("GMT"));String date = format.format(new Date());// 拼接String preStr = "host: " + url.getHost() + "\n" +"date: " + date + "\n" +"GET " + url.getPath() + " HTTP/1.1";// SHA256加密Mac mac = Mac.getInstance("hmacsha256");SecretKeySpec spec = new SecretKeySpec(xfXhConfig.getApiSecret().getBytes(StandardCharsets.UTF_8), "hmacsha256");mac.init(spec);byte[] hexDigits = mac.doFinal(preStr.getBytes(StandardCharsets.UTF_8));// Base64加密String sha = Base64.getEncoder().encodeToString(hexDigits);// 拼接String authorizationOrigin = String.format("api_key=\"%s\", algorithm=\"%s\", headers=\"%s\", signature=\"%s\"", xfXhConfig.getApiKey(), "hmac-sha256", "host date request-line", sha);// 拼接地址HttpUrl httpUrl = Objects.requireNonNull(HttpUrl.parse("https://" + url.getHost() + url.getPath())).newBuilder().addQueryParameter("authorization", Base64.getEncoder().encodeToString(authorizationOrigin.getBytes(StandardCharsets.UTF_8))).addQueryParameter("date", date).addQueryParameter("host", url.getHost()).build();return httpUrl.toString();} catch (Exception e) {log.error("鉴权方法中发生错误:" + e.getMessage());return null;}}/*** 获取请求参数** @param uid     每个用户的id,用于区分不同用户* @param msgList 发送给大模型的消息,可以包含上下文内容* @return 请求DTO,该 DTO 转 json 字符串后生成的格式参考 resources/demo-json/request.json*/public RequestDTO getRequestParam(String uid, List<MsgDTO> msgList) {RequestDTO requestDTO = new RequestDTO();requestDTO.setHeader(new RequestDTO.HeaderDTO(xfXhConfig.getAppId(), uid));requestDTO.setParameter(new RequestDTO.ParameterDTO(new RequestDTO.ParameterDTO.ChatDTO(xfXhConfig.getDomain(), xfXhConfig.getTemperature(), xfXhConfig.getMaxTokens())));requestDTO.setPayload(new RequestDTO.PayloadDTO(new RequestDTO.PayloadDTO.MessageDTO(msgList)));return requestDTO;}}

🍀 controller 包

📌 TestController
/*** @author 狐狸半面添* @create 2023-09-20 1:42*/
@RestController
@RequestMapping("/test")
@Slf4j
public class TestController {@Resourceprivate XfXhStreamClient xfXhStreamClient;@Resourceprivate XfXhConfig xfXhConfig;/*** 发送问题** @param question 问题* @return 星火大模型的回答*/@GetMapping("/sendQuestion")public String sendQuestion(@RequestParam("question") String question) {// 如果是无效字符串,则不对大模型进行请求if (StrUtil.isBlank(question)) {return "无效问题,请重新输入";}// 获取连接令牌if (!xfXhStreamClient.operateToken(XfXhStreamClient.GET_TOKEN_STATUS)) {return "当前大模型连接数过多,请稍后再试";}// 创建消息对象MsgDTO msgDTO = MsgDTO.createUserMsg(question);// 创建监听器XfXhWebSocketListener listener = new XfXhWebSocketListener();// 发送问题给大模型,生成 websocket 连接WebSocket webSocket = xfXhStreamClient.sendMsg(UUID.randomUUID().toString().substring(0, 10), Collections.singletonList(msgDTO), listener);if (webSocket == null) {// 归还令牌xfXhStreamClient.operateToken(XfXhStreamClient.BACK_TOKEN_STATUS);return "系统内部错误,请联系管理员";}try {int count = 0;// 为了避免死循环,设置循环次数来定义超时时长int maxCount = xfXhConfig.getMaxResponseTime() * 5;while (count <= maxCount) {Thread.sleep(200);if (listener.isWsCloseFlag()) {break;}count++;}if (count > maxCount) {return "大模型响应超时,请联系管理员";}// 响应大模型的答案return listener.getAnswer().toString();} catch (InterruptedException e) {log.error("错误:" + e.getMessage());return "系统内部错误,请联系管理员";} finally {// 关闭 websocket 连接webSocket.close(1000, "");// 归还令牌xfXhStreamClient.operateToken(XfXhStreamClient.BACK_TOKEN_STATUS);}}
}

🍀 XfXhApplication 启动类

@SpringBootApplication
public class XfXhApplication {public static void main(String[] args) {SpringApplication.run(XfXhApplication.class, args);}
}

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

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

相关文章

element中使用el-steps 进度条效果demo(整理)

<template><div class"margin-top20"><!-- align-center 不要居中就去掉 --><!-- process-status 这几个参数值&#xff1a;改变颜色 wait / process / finish / error / --><!-- active 到第几个是绿色 --><el-steps :space&qu…

深度解读F5:从企业级负载均衡到云原生应用服务

上世纪九十年代&#xff0c;Internet 的快速发展催生了大量在线网站&#xff0c;Web 访问量迅速提升。在互联网泡沫破灭以前&#xff0c;这个领域基本是围绕如何对 Web 网站进行负载均衡与优化。因而在早期&#xff0c;也会有“Web 交换机”的说法。从1997年 F5 发布了 BIG-IP …

黑马JVM总结(二十五)

&#xff08;1&#xff09;字节码指令-cinit 构造方法可以分为两类&#xff0c;一类是cinit 一类init cinit是整个类的构造方法 putstatic&#xff1a;进行static变量的赋值&#xff0c;是到常量池里找到名字一个叫做i的变量 &#xff08;2&#xff09;字节码指令-init in…

JimuReport积木报表 v1.6.2 版本正式发布—开源免费的低代码报表

项目介绍 一款免费的数据可视化报表&#xff0c;含报表和大屏设计&#xff0c;像搭建积木一样在线设计报表&#xff01;功能涵盖&#xff0c;数据报表、打印设计、图表报表、大屏设计等&#xff01; Web 版报表设计器&#xff0c;类似于excel操作风格&#xff0c;通过拖拽完成报…

mysql面试题2:说一说MySQL的架构设计?一条 MySQL 语句执行的步骤?

该文章专注于面试,面试只要回答关键点即可,不需要对框架有非常深入的回答,如果你想应付面试,是足够了,抓住关键点 面试官:说一说MySQL的架构设计? MySQL的架构设计主要包括以下几个组件: 连接器(Connector):负责与客户端建立连接,并进行身份验证和授权。 查询缓存…

BFS专题7 多终点迷宫问题

题目&#xff1a; 样例&#xff1a; 输入 3 3 0 0 0 1 0 0 0 1 0 输出 0 1 2 -1 2 3 -1 -1 4 思路&#xff1a; 单纯的 BFS 迷宫问题&#xff0c;只是标记一下每个点的 step&#xff0c;注意初始化答案数组都为 -1. 代码详解如下&#xff1a; #include <iostream> #…

常用接口测试工具

首先&#xff0c;什么是接口呢&#xff1f; 接口一般来说有两种&#xff0c;一种是程序内部的接口&#xff0c;一种是系统对外的接口。 系统对外的接口&#xff1a;比如你要从别的网站或服务器上获取资源或信息&#xff0c;别人肯定不会把数据库共享给你&#xff0c;他只能给你…

五、3d场景的卡片展示的创建

在我们3d的开发中&#xff0c;对某一些建筑和物体进行解释说明是非常常见的现象&#xff0c;那么就不得不说卡片的展示了&#xff0c;卡片展示很友好的说明了当前物体的状态&#xff0c;一目了然&#xff0c;下面就是效果图。 它主要有两个方法来实现&#xff0c;大量的图片建议…

最近很火的AIGC人工智能之AI赋能运营(巧用ChatGPT轻松上手新媒体)

前言 「作者主页」&#xff1a;雪碧有白泡泡 「个人网站」&#xff1a;雪碧的个人网站 「推荐专栏」&#xff1a; ★java一站式服务 ★ ★ React从入门到精通★ ★前端炫酷代码分享 ★ ★ 从0到英雄&#xff0c;vue成神之路★ ★ uniapp-从构建到提升★ ★ 从0到英雄&#xff…

阿里云服务器技术创新、网络技术和数据中心技术说明

阿里云服务器技术创新、网络技术创新、数据中心技术创新和智能运维&#xff1a;云服务器方升架构、自研硬件、自研存储硬件AliFlash和异构计算加速平台&#xff0c;以及全自研网络系统技术创新和数据中心巴拿马电源、液冷技术等技术创新说明&#xff0c;阿里云百科分享阿里云服…

算法-堆/多路归并-查找和最小的 K 对数字

算法-堆/多路归并-查找和最小的 K 对数字 1 题目概述 1.1 题目出处 https://leetcode.cn/problems/find-k-pairs-with-smallest-sums/description/?envTypestudy-plan-v2&envIdtop-interview-150 1.2 题目描述 2 优先级队列构建大顶堆 2.1 思路 将两个数字的和放入大…

Spring面试题23:Spring支持哪些事务管理类型?Spring框架的事务管理有哪些优点?你更倾向用哪种事务管理类型?

该文章专注于面试,面试只要回答关键点即可,不需要对框架有非常深入的回答,如果你想应付面试,是足够了,抓住关键点 面试官:Spring支持哪些事务管理类型? Spring 支持以下几种事务管理类型: 编程式事务管理:通过在代码中显式地使用事务管理 API(如 TransactionTempla…

英飞凌 Tricore 架构中断系统详解

本文以TC3系列MCU为例&#xff0c;先来了解中断源是如何产生的&#xff0c;再看一下CPU是如何处理中断源的。 AURIX TC3XX的中断路由模块 Interrupt Router (IR) 在TC3中&#xff0c;中断既可以被CPU处理&#xff0c;也可以被DMA处理&#xff0c;所以手册中不再把中断称为中断…

Spring 学习(九)整合 Mybatis

1. 整合 Mybatis 步骤 导入相关 jar 包 <dependencies><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.12</version><scope>test</scope></dependency><dependency>…

软件测试/测试开发丨岗位内推-58同城岗位开放~

58同城-测试工程师 岗位职责 1.参与需求分析、设计评审&#xff0c;制定测试计划&#xff0c;设计测试用例&#xff0c;搭建测试环境&#xff1b; 2.执行各级别和类型的测试&#xff0c;参与互联网测试的全流程&#xff1b; 3.借助最前沿的研发技术和理念&#xff0c;通过测…

短视频矩阵源码saas版开发---技术成品打磨

短视频矩阵源码saas版开发---技术成品打磨 企业要不要做抖音&#xff1f; 我的答案是要&#xff0c;而且要快&#xff0c;原因有以下几点&#xff1a; 1. 抖音高速增长未停&#xff0c;流量红利还未完全释放完&#xff0c;仍然有增长空间&#xff1b; 2. 抖音变现能力越来越…

《家的温暖,国庆团圆》

目录 &#x1f4d6; 引言 &#x1f4dd; 假日放假表 &#x1f365; 中秋节 &#x1f4da; 中秋节的由来 中秋节的仪式 &#x1f4da; 赏月 &#x1f4da; 吃月饼 &#x1f4da; 猜灯谜 &#x1f4da; 品茶赏花 &#x1f4da; 舞狮龙 &#x1f4da; 中秋节的感触 &am…

C++项目笔记--基于TensorRT搭建一个YoloV5服务器

目录 1--项目描述 2--项目地址 3--编译运行 4--测试结果 5--补充说明 1--项目描述 ① 基于 C/S 模型来构建 TCP 服务器和 TCP 客户端。 ② 使用 Epoll 来监控服务器和客户端之间的连接。 ③ 服务器和客户端约定使用相同的数据传输协议&#xff0c;头部分别使用 4 个字节来…

OpenCV查找和绘制轮廓:findContours和drawContours

1 任务描述&#xff1a; 绘制图中粗线矩形的2个边界&#xff0c;并找到其边界的中心线 图1 原始图像 2.函数原型 findContours( InputOutputArray image, OutputArrayOfArrays contours, OutputArray hierarchy, int mode, …

【C++】string 之 find、rfind、replace、compare函数的学习

前言 上篇文章&#xff0c;我们学习了assign、at、append这三个函数 今天&#xff0c;我们来学习find、 函数 find函数 引入 我们都知道&#xff0c;find函数可以是string类中&#xff0c;用于查找字符或者字符串的函数 也可以是&#xff0c;<algorithm>头文件中&am…