OpenAI模块重构

文章目录

    • 1.common-openai-starter
        • 1.目录结构
        • 2.OpenAiProperties.java 新增apiUrl
        • 3.OpenAIAutoConfiguration.java
        • 4.OpenAiClient.java 使用gson重构
    • 2.common-openai-starter-demo
        • 1.目录结构
        • 2.application.yml 新增api-url
        • 3.OpenAiController.java
        • 4.OpenAiApplication.java
        • 5.测试

1.common-openai-starter

1.目录结构

CleanShot 2025-01-02 at 16.40.21@2x

2.OpenAiProperties.java 新增apiUrl
package com.sunxiansheng.openai.config.properties;import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;/*** Description: OpenAI配置属性类** @Author sun* @Create 2024/12/14 11:44* @Version 1.0*/
@Data
@ConfigurationProperties(prefix = "openai")
public class OpenAiProperties {/*** OpenAI API Key*/private String apiKey;/*** OpenAI API URL 有默认值*/private String apiUrl = "https://api.openai.com/v1/chat/completions";
}
3.OpenAIAutoConfiguration.java
package com.sunxiansheng.openai.config;import com.sunxiansheng.openai.client.OpenAiClient;
import com.sunxiansheng.openai.config.properties.OpenAiProperties;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;/*** Description: OpenAI自动配置类** @Author sun* @Create 2024/12/14 11:39* @Version 1.0*/
@Configuration
@EnableConfigurationProperties({OpenAiProperties.class}) // 启用配置类
public class OpenAIAutoConfiguration {/*** 创建 OpenAiClient** @return*/@Bean@ConditionalOnMissingBeanpublic OpenAiClient openAiClient() {return new OpenAiClient();}
}
4.OpenAiClient.java 使用gson重构
package com.sunxiansheng.openai.client;import com.google.gson.JsonArray;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
import com.sunxiansheng.openai.config.properties.OpenAiProperties;
import okhttp3.*;import javax.annotation.Resource;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.Base64;
import java.util.concurrent.TimeUnit;
import java.util.logging.Level;
import java.util.logging.Logger;/*** Description: OpenAI 客户端类** @Author sun* @Create 2024/12/14 11:56* @Version 1.1*/
public class OpenAiClient {@Resourceprivate OpenAiProperties openAiProperties;private static final OkHttpClient CLIENT = new OkHttpClient.Builder().connectTimeout(120, TimeUnit.SECONDS).readTimeout(120, TimeUnit.SECONDS).writeTimeout(120, TimeUnit.SECONDS).build();private static final Logger LOGGER = Logger.getLogger(OpenAiClient.class.getName());/*** 向 AI 提问通用方法** @param model        使用的 AI 模型,如 "gpt-4o"* @param prompt       提示内容* @param base64Encode 是否对内容进行 Base64 编码* @return AI 的响应内容*/public String askAI(String model, String prompt, boolean base64Encode) {try {// 处理 Base64 编码String encodedPrompt = base64Encode ? encodeBase64(prompt) : prompt;// 构造请求体RequestBody body = RequestBody.create(createJsonRequest(model, encodedPrompt), MediaType.get("application/json; charset=utf-8"));// 构建请求Request request = new Request.Builder().url(openAiProperties.getApiUrl()).header("Authorization", "Bearer " + openAiProperties.getApiKey()).header("Content-Type", "application/json").post(body).build();// 发送请求并获取响应try (Response response = CLIENT.newCall(request).execute()) {if (!response.isSuccessful()) {throw new IOException("Unexpected response: " + response);}// 解析 JSON 响应return parseResponse(response.body().string());}} catch (IOException e) {LOGGER.log(Level.SEVERE, "Error occurred during API request: " + e.getMessage(), e);throw new RuntimeException("API request failed", e);}}/*** 对输入内容进行 Base64 编码** @param prompt 输入内容* @return 编码后的字符串*/private String encodeBase64(String prompt) {return Base64.getEncoder().encodeToString(prompt.getBytes(StandardCharsets.UTF_8));}/*** 构建请求的 JSON 数据** @param model         使用的 AI 模型* @param encodedPrompt 编码后的输入内容* @return 构建好的 JSON 字符串*/private String createJsonRequest(String model, String encodedPrompt) {JsonObject jsonRequest = new JsonObject();jsonRequest.addProperty("model", model);JsonArray messages = new JsonArray();// 添加 system 信息JsonObject systemMessage = new JsonObject();systemMessage.addProperty("role", "system");systemMessage.addProperty("content", "请根据以下内容提供问题的解决方案。使用中文回答,不要使用markdown语法和特殊符号**之类的,注意,内容可能经过 Base64 编码。");messages.add(systemMessage);// 添加 user 信息JsonObject userMessage = new JsonObject();userMessage.addProperty("role", "user");userMessage.addProperty("content", encodedPrompt);messages.add(userMessage);jsonRequest.add("messages", messages);return jsonRequest.toString();}/*** 解析 API 响应内容** @param responseBody 响应的 JSON 内容* @return 解析后的结果*/private String parseResponse(String responseBody) {JsonObject jsonObject = JsonParser.parseString(responseBody).getAsJsonObject();JsonArray choices = jsonObject.getAsJsonArray("choices");if (choices != null && choices.size() > 0) {JsonObject choice = choices.get(0).getAsJsonObject();JsonObject message = choice.getAsJsonObject("message");return message.get("content").getAsString();}throw new RuntimeException("Invalid response: No choices found.");}
}

2.common-openai-starter-demo

1.目录结构

CleanShot 2025-01-02 at 16.47.19@2x

2.application.yml 新增api-url
openai:api-key: guest # 必填,OpenAI API Keyapi-url: https://api.openai.com/v1/chat/completions # 可选,有默认值,为了防止以后 URL 变化
3.OpenAiController.java
package com.sunxiansheng.openai.controller;import com.sunxiansheng.openai.client.OpenAiClient;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;import javax.annotation.Resource;/*** Description: OpenAI 控制器类** @Author sun* @Create 2024/12/14 12:27* @Version 1.0*/
@RestController
public class OpenAiController {@Resourceprivate OpenAiClient openAiClient;@RequestMapping("/ask")public String ask(String question) {String res = openAiClient.askAI("gpt-4o", question, false);return "AI回答:" + res;}
}
4.OpenAiApplication.java
package com.sunxiansheng.openai;import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;/*** Description: OpenAI启动类** @Author sun* @Create 2024/12/14 12:23* @Version 1.0*/
@SpringBootApplication
public class OpenAiApplication {public static void main(String[] args) {SpringApplication.run(OpenAiApplication.class, args);}
}
5.测试

CleanShot 2025-01-02 at 16.48.52@2x

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

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

相关文章

数据标注开源框架 Label Studio

数据标注开源框架 Label Studio Label Studio 是一个开源的、灵活的数据标注平台,旨在帮助开发者和数据科学家轻松创建高质量的训练数据集。它支持多种类型的数据(如文本、图像、音频、视频等)以及复杂的标注任务(如分类、命名实体…

详解:TCP/IP五层(四层)协议模型

一.五层(四层)模型 1.概念 TCP/IP协议模型分为五层:物理层、数据链路层、网络层、传输层和应用层。这五层每一层都依赖于其下一层给它提供的网络去实现需求。 1)物理层:这是最基本的一层,也是最接近硬件…

使用Python进行大模型的测试与部署

随着人工智能技术的飞速发展,大规模模型在各行各业的应用日益广泛。然而,如何有效测试这些模型以确保其稳定性和准确性,成为测试人员的们面临的一大挑战。本文将详细介绍在Python环境下,如何测试大模型,并探讨其部署策…

高并发处理 --- 超卖问题+一人一单解决方案

在高并发场景下,超卖和一人一单是两个典型的并发问题。为了解决这两个问题,我们可以使用乐观锁(CAS)和悲观锁,这两者分别有不同的实现方式和适用场景。下面我们详细介绍如何通过 乐观锁(CAS) 和…

【2024年华为OD机试】(C卷,100分)- 约瑟夫问题 (JavaScriptJava PythonC/C++)

一、问题描述 题目描述 输入一个由随机数组成的数列(数列中每个数均是大于 0 的整数,长度已知),和初始计数值 m。 从数列首位置开始计数,计数到 m 后,将数列该位置数值替换计数值 m,并将数列…

浅谈APP之历史股票通过echarts绘图

浅谈APP之历史股票通过echarts绘图 需求描述 今天我们需要做一个简单的历史股票收盘价格通过echarts进行绘图,效果如下: 业务实现 代码框架 代码框架如下: . 依赖包下载 我们通过网站下载自己需要的涉及的图标,勾选之后进…

【0x0012】HCI_Delete_Stored_Link_Key命令详解

目录 一、命令参数 二、命令格式及参数 2.1. HCI_Delete_Stored_Link_Key 命令格式 2.2. BD_ADDR 2.3. Delete_All 三、生成事件及参数 3.1. HCI_Command_Complete事件 3.2. Status 3.3. Num_Keys_Deleted 四、命令执行流程 4.1. 命令发送阶段 4.2. 控制器处理阶段…

提示词的艺术 ---- AI Prompt 进阶(提示词框架)

提示词的艺术 ---- AI Prompt 进阶(提示词框架) 写在前面 上周发布了一篇《提示词的艺术----AI Prompt撰写指南》,旨在帮助读者理解提示词的作用,以及简单的提示词撰写指南。本篇作为进阶内容,将给出常用的提示词框架…

javaSE.类的继承

在定义不同类的时候,为了方便使用可以将这些共同属性抽象成一个父类,在定义其他子类时可以继承自该父类,减少代码的重复定义,子类可以使用父类中非私有成员. extents 没有可用的无形参构造方法 被构造方法覆盖了 super 需要调用父类的构造方法 super必须是构造主体的第一条语…

统计文本文件中单词频率的 Swift 与 Bash 实现详解

网罗开发 (小红书、快手、视频号同名) 大家好,我是 展菲,目前在上市企业从事人工智能项目研发管理工作,平时热衷于分享各种编程领域的软硬技能知识以及前沿技术,包括iOS、前端、Harmony OS、Java、Python等…

qt QUrl详解

1、概述 QUrl是Qt框架中用于处理URL(统一资源定位符)的类,它提供了构建、解析、编码、解码和处理URL的功能。QUrl支持多种协议,如HTTP、HTTPS、FTP以及文件URL等,并能处理URL的各个组成部分,如协议、主机、…

c++----------------------多态

1.多态 1.1多态的概念 多态(polymorphism)的概念:通俗来说,就是多种形态。多态分为编译时多态(静态多态)和运⾏时多 态(动态多态),这⾥我们重点讲运⾏时多态,编译时多态(静态多态)和运⾏时多态(动态多态)。编译时 多态(静态多态)…

javaSE.类与对象

类与对象 人类,鸟类,鱼类... 例如人,具有不同性格,但根本上都是人。 对象是某一类事物实际存在的每个个体(实例)例如:雷军 A:谁拿走了我的手机? B:是个人(类&#xff0…

Windows cmd常用命令

文章目录 Windows cmd常用命令一、引言二、文件和目录操作1、查看和切换目录2、文件和目录的创建与删除 三、系统信息与网络配置1、系统信息2、网络配置 四、使用示例五、总结 Windows cmd常用命令 一、引言 Windows 命令提示符(cmd)是一个强大的工具&a…

保健食品注册数据库<一键查询保健食品信息>

在保健品市场竞争激烈的情况下,企业要如何保障产品合规、信息公开,并且能够迅速应对市场变化呢?查询保健食品注册信息是关键环节。 当下,查询保健食品注册信息主要有两种途径:一是利用国家保健食品注册数据库进行查询…

无所不搜,吾爱制造

吾爱论坛作为众多软件资源爱好者的宝藏之地,汇聚了许多优秀的软件作品,堪称软件界的“福地”。许多技术大佬在这里分享自己的创作。 而今天要介绍的,正是吾爱作者“buyaobushuo”自制的多功能娱乐软件——太极。这款软件基于flet开发&#x…

【C++】详细讲解继承(下)

本篇来继续说说继承。上篇可移步至【C】详细讲解继承(上) 1.继承与友元 友元关系不能继承 ,也就是说基类友元不能访问派⽣类私有和保护成员。 class Student;//前置声明class Same //基类 { public:friend void Fun(const Same& p, con…

【二叉树】4. 判断一颗二叉树是否是平衡二叉树。5. 对称二叉树。6. 二叉树的构建及遍历 7. 二叉树的分层遍历 。

判断一颗二叉树是否是平衡二叉树。OJ链接 可以在求树高度的过程中判断树是否平衡 对称二叉树。OJ链接 二叉树的构建及遍历。OJ链接 注意:public static int i最好把static去掉 否则当有多个测试用例时 i无法重新为0二叉树的分层遍历 。OJ链接 但此题要求返回List…

代码随想录刷题day14(2)|(链表篇)02.07. 链表相交(疑点)

目录 一、链表理论基础 二、链表相交求解思路 三、相关算法题目 四、疑点 一、链表理论基础 代码随想录 二、链表相交求解思路 链表相交时,是结点的位置,也就是指针相同,不是结点的数值相同; 思路:定义两个指针…

IDE提示:因为在此系统上禁止运行脚本。有关详细信息,请参阅 https:/go.microsoft.com/fwlink/?LinkID=135170

问题情况 不知道为什么我的IDE终端运行命令的时候总提示以下内容: Import-Module : 无法加载文件 D:\Anaconda3\shell\condabin\Conda.psm1,因为在此系统上禁止运行脚本。有关详细信息,请参阅 https:/go.microsoft.com/fwlink/?LinkID1351…