java調(diào)用chatgpt接口來(lái)實(shí)現(xiàn)專屬于自己的人工智能助手
前言
今天突然突發(fā)奇想,就想要用java來(lái)調(diào)用chatget的接口,實(shí)現(xiàn)自己的聊天機(jī)器人,但是網(wǎng)上找文章,屬實(shí)是少的可憐(可能是不讓發(fā)吧)。找到了一些文章,但是基本都是通過(guò)調(diào)用別人的庫(kù)來(lái)完成的,導(dǎo)入其他的jar還有不低的學(xué)習(xí)成本,于是就自己使用HttpClient5寫了一個(gè),在這里講解一下思路。
導(dǎo)包
對(duì)于http調(diào)用,我使用的是比較流行的httpclient5,然后直接創(chuàng)建了一個(gè)springboot項(xiàng)目,方便以后對(duì)外提供接口。
<parent> <artifactId>spring-boot-starter-parent</artifactId> <groupId>org.springframework.boot</groupId> <version>2.5.3</version> </parent> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> </dependency> <!-- https://mvnrepository.com/artifact/org.apache.httpcomponents.client5/httpclient5 --> <dependency> <groupId>org.apache.httpcomponents.client5</groupId> <artifactId>httpclient5</artifactId> <version>5.2.1</version> </dependency> </dependencies>
基本說(shuō)明
在編寫代碼之前,這個(gè)先給出HttpClient的Api文檔 api文檔
我們?cè)诰帉懘a之前需要了解官方提供的接口如何進(jìn)行訪問(wèn)以及返回的結(jié)果是什么
請(qǐng)求參數(shù)
官方文檔地址為文檔,請(qǐng)求參數(shù)必須填寫的內(nèi)容如下
{ "model": "gpt-3.5-turbo", "messages": [{"role": "user", "content": "Hello!"}] }
一個(gè)是model,一個(gè)是messages。model根據(jù)自己的情況來(lái)選擇,聊天的話就是gpt-3.5-turbo,下面的messages里面包含n個(gè)對(duì)象,每個(gè)對(duì)象有role和content,role表示角色,content表示內(nèi)容。
下面為官方文檔中的解釋
簡(jiǎn)單理解就是我們要問(wèn)問(wèn)題,role就是user。如果要實(shí)現(xiàn)連續(xù)對(duì)話,那么就將返回的返回內(nèi)容設(shè)置到messages中,role設(shè)置為返回的role。
響應(yīng)參數(shù)
下面直接給出響應(yīng)的內(nèi)容
{ 'id': 'chatcmpl-6p9XYPYSTTRi0xEviKjjilqrWU2Ve', 'object': 'chat.completion', 'created': 1677649420, 'model': 'gpt-3.5-turbo', 'usage': {'prompt_tokens': 56, 'completion_tokens': 31, 'total_tokens': 87}, 'choices': [ { 'message': { 'role': 'assistant', 'content': 'The 2020 World Series was played in Arlington, Texas at the Globe Life Field, which was the new home stadium for the Texas Rangers.'}, 'finish_reason': 'stop', 'index': 0 } ] }
我們問(wèn)問(wèn)題的答案就在choices.message下的content中,而role就代表了chatGpt扮演的角色??吹竭@我們就應(yīng)該知道該干嘛了吧肯定是創(chuàng)建對(duì)應(yīng)的VO類啊。
創(chuàng)建請(qǐng)求和響應(yīng)的VO類
下面5個(gè)類就對(duì)應(yīng)了我們發(fā)送和接收的各種信息
ChatGptMessage類
@Data @NoArgsConstructor @AllArgsConstructor public class ChatGptMessage { String role; String content; }
ChatGptRequestParameter 類
@Data @NoArgsConstructor @AllArgsConstructor public class ChatGptRequestParameter { String model = "gpt-3.5-turbo"; List<ChatGptMessage> messages = new ArrayList<>(); public void addMessages(ChatGptMessage message) { this.messages.add(message); } }
ChatGptResponseParameter 類
@Data @NoArgsConstructor @AllArgsConstructor public class ChatGptResponseParameter { String id; String object; String created; String model; Usage usage; List<Choices> choices; }
Choices 類
@Data @NoArgsConstructor @AllArgsConstructor public class Choices { ChatGptMessage message; String finish_reason; Integer index; }
Usage 類
@Data @NoArgsConstructor @AllArgsConstructor public class Usage { String prompt_tokens; String completion_tokens; String total_tokens; }
代碼編寫
不說(shuō)廢話,首先創(chuàng)建一個(gè)CustomChatGpt類
public class CustomChatGpt { }
然后定義一些成員屬性
/** * 自己chatGpt的ApiKey */ private String apiKey; /** * 使用的模型 */ private String model = "gpt-3.5-turbo-0301"; /** * 對(duì)應(yīng)的請(qǐng)求接口 */ private String url = "https://api.openai.com/v1/chat/completions"; /** * 默認(rèn)編碼 */ private Charset charset = StandardCharsets.UTF_8; /** * 創(chuàng)建一個(gè)ChatGptRequestParameter,用于攜帶請(qǐng)求參數(shù) */ private ChatGptRequestParameter chatGptRequestParameter = new ChatGptRequestParameter();
提供一個(gè)ApiKey的構(gòu)造器,創(chuàng)建該對(duì)象必須要傳入ApiKey
public CustomChatGpt(String apiKey) { this.apiKey = apiKey; }
定義一個(gè)響應(yīng)超時(shí)時(shí)間
/** * 響應(yīng)超時(shí)時(shí)間,毫秒 */ private int responseTimeout = 10000; public void setResponseTimeout(int responseTimeout) { this.responseTimeout = responseTimeout; }
編寫一個(gè)getAnswer方法,要求傳入一個(gè)CloseableHttpClient和一個(gè)問(wèn)題
public String getAnswer(CloseableHttpClient client, String question) { }
繼續(xù)實(shí)現(xiàn)方法,下面會(huì)完成一些參數(shù)的創(chuàng)建和設(shè)置
// 創(chuàng)建一個(gè)HttpPost HttpPost httpPost = new HttpPost(url); // 創(chuàng)建一個(gè)ObjectMapper,用于解析和創(chuàng)建json ObjectMapper objectMapper = new ObjectMapper(); // 設(shè)置請(qǐng)求參數(shù) chatGptRequestParameter.addMessages(new ChatGptMessage("user", question)); HttpEntity httpEntity = null; try { // 對(duì)象轉(zhuǎn)換為json字符串 httpEntity = new StringEntity(objectMapper.writeValueAsString(chatGptRequestParameter), charset); } catch (JsonProcessingException e) { System.out.println(question + "->json轉(zhuǎn)換異常"); return null; } httpPost.setEntity(httpEntity);
下面會(huì)完成一些配置的設(shè)置
// 設(shè)置請(qǐng)求頭 httpPost.setHeader(HttpHeaders.CONTENT_TYPE, "application/json"); // 設(shè)置登錄憑證 httpPost.setHeader(HttpHeaders.AUTHORIZATION, "Bearer " + apiKey); // 用于設(shè)置超時(shí)時(shí)間 RequestConfig config = RequestConfig .custom() .setResponseTimeout(responseTimeout, TimeUnit.MILLISECONDS) .build(); httpPost.setConfig(config);
下面代碼會(huì)提交請(qǐng)求,解析響應(yīng),最后返回對(duì)應(yīng)問(wèn)題的答案
try { // 提交請(qǐng)求 return client.execute(httpPost, response -> { // 得到返回的內(nèi)容 String resStr = EntityUtils.toString(response.getEntity(), charset); // 轉(zhuǎn)換為對(duì)象 ChatGptResponseParameter responseParameter = objectMapper.readValue(resStr, ChatGptResponseParameter.class); String ans = ""; // 遍歷所有的Choices(一般都只有一個(gè)) for (Choices choice : responseParameter.getChoices()) { ChatGptMessage message = choice.getMessage(); chatGptRequestParameter.addMessages(new ChatGptMessage(message.getRole(), message.getContent())); String s = message.getContent().replaceAll("\n+", "\n"); ans += s; } // 返回信息 return ans; }); } catch (IOException e) { e.printStackTrace(); } // 發(fā)生異常,移除剛剛添加的ChatGptMessage chatGptRequestParameter.getMessages().remove(chatGptRequestParameter.getMessages().size()-1); return "您當(dāng)前的網(wǎng)絡(luò)無(wú)法訪問(wèn)";
下面給出這個(gè)類的完整代碼
import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.databind.ObjectMapper; import com.ttpfx.vo.ChatGptMessage; import com.ttpfx.vo.ChatGptRequestParameter; import com.ttpfx.vo.ChatGptResponseParameter; import com.ttpfx.vo.Choices; import org.apache.hc.client5.http.classic.methods.HttpPost; import org.apache.hc.client5.http.config.RequestConfig; import org.apache.hc.client5.http.impl.classic.CloseableHttpClient; import org.apache.hc.core5.http.HttpEntity; import org.apache.hc.core5.http.HttpHeaders; import org.apache.hc.core5.http.io.entity.EntityUtils; import org.apache.hc.core5.http.io.entity.StringEntity; import java.io.IOException; import java.nio.charset.Charset; import java.nio.charset.StandardCharsets; import java.util.concurrent.TimeUnit; /** * @author ttpfx * @date 2023/3/23 */ public class CustomChatGpt { /** * 自己chatGpt的ApiKey */ private String apiKey; /** * 使用的模型 */ private String model = "gpt-3.5-turbo-0301"; /** * 對(duì)應(yīng)的請(qǐng)求接口 */ private String url = "https://api.openai.com/v1/chat/completions"; /** * 默認(rèn)編碼 */ private Charset charset = StandardCharsets.UTF_8; /** * 創(chuàng)建一個(gè)ChatGptRequestParameter,用于攜帶請(qǐng)求參數(shù) */ private ChatGptRequestParameter chatGptRequestParameter = new ChatGptRequestParameter(); /** * 相應(yīng)超時(shí)時(shí)間,毫秒 */ private int responseTimeout = 1000; public void setResponseTimeout(int responseTimeout) { this.responseTimeout = responseTimeout; } public CustomChatGpt(String apiKey) { this.apiKey = apiKey; } public String getAnswer(CloseableHttpClient client, String question) { // 創(chuàng)建一個(gè)HttpPost HttpPost httpPost = new HttpPost(url); // 創(chuàng)建一個(gè)ObjectMapper,用于解析和創(chuàng)建json ObjectMapper objectMapper = new ObjectMapper(); // 設(shè)置請(qǐng)求參數(shù) chatGptRequestParameter.addMessages(new ChatGptMessage("user", question)); HttpEntity httpEntity = null; try { // 對(duì)象轉(zhuǎn)換為json字符串 httpEntity = new StringEntity(objectMapper.writeValueAsString(chatGptRequestParameter), charset); } catch (JsonProcessingException e) { System.out.println(question + "->json轉(zhuǎn)換異常"); return null; } httpPost.setEntity(httpEntity); // 設(shè)置請(qǐng)求頭 httpPost.setHeader(HttpHeaders.CONTENT_TYPE, "application/json"); // 設(shè)置登錄憑證 httpPost.setHeader(HttpHeaders.AUTHORIZATION, "Bearer " + apiKey); // 用于設(shè)置超時(shí)時(shí)間 RequestConfig config = RequestConfig .custom() .setResponseTimeout(responseTimeout, TimeUnit.MILLISECONDS) .build(); httpPost.setConfig(config); try { // 提交請(qǐng)求 return client.execute(httpPost, response -> { // 得到返回的內(nèi)容 String resStr = EntityUtils.toString(response.getEntity(), charset); // 轉(zhuǎn)換為對(duì)象 ChatGptResponseParameter responseParameter = objectMapper.readValue(resStr, ChatGptResponseParameter.class); String ans = ""; // 遍歷所有的Choices(一般都只有一個(gè)) for (Choices choice : responseParameter.getChoices()) { ChatGptMessage message = choice.getMessage(); chatGptRequestParameter.addMessages(new ChatGptMessage(message.getRole(), message.getContent())); String s = message.getContent().replaceAll("\n+", "\n"); ans += s; } // 返回信息 return ans; }); } catch (IOException e) { e.printStackTrace(); } // 發(fā)生異常,移除剛剛添加的ChatGptMessage chatGptRequestParameter.getMessages().remove(chatGptRequestParameter.getMessages().size()-1); return "您當(dāng)前的網(wǎng)絡(luò)無(wú)法訪問(wèn)"; } }
使用
下面就是測(cè)試代碼,我們只需要傳入一個(gè)CloseableHttpClient 和 question 即可
public class Test { public static void main(String[] args) throws IOException { CloseableHttpClient httpClient = HttpClients.createDefault(); String apiKey = "自己的ApiKey"; CustomChatGpt customChatGpt = new CustomChatGpt(apiKey); // 根據(jù)自己的網(wǎng)絡(luò)設(shè)置吧 customChatGpt.setResponseTimeout(20000); while (true) { System.out.print("\n請(qǐng)輸入問(wèn)題(q退出):"); String question = new Scanner(System.in).nextLine(); if ("q".equals(question)) break; long start = System.currentTimeMillis(); String answer = customChatGpt.getAnswer(httpClient, question); long end = System.currentTimeMillis(); System.out.println("該回答花費(fèi)時(shí)間為:" + (end - start) / 1000.0 + "秒"); System.out.println(answer); } httpClient.close(); } }
最后說(shuō)明
對(duì)于ApiKey,只能說(shuō)難者不會(huì),會(huì)者不難,這個(gè)沒(méi)辦法教。
如果代碼無(wú)法運(yùn)行,或者運(yùn)行速度及其緩慢,請(qǐng)使用代理,在HttpClient里面可以很輕松的使用代理
String proxyIp = "127.0.0.1"; int proxyPort = 7890; HttpHost httpHost = new HttpHost(proxyIp, proxyPort);
上面就是一個(gè)示例,對(duì)于代理,這里也就無(wú)法繼續(xù)進(jìn)行說(shuō)明了。
如果我們完成了上面的功能,是不是就能夠?qū)ν馓峁┙涌?,然后寫一個(gè)自己的網(wǎng)頁(yè)端的ChatGpt或者弄一個(gè)聊天機(jī)器人呢?當(dāng)然沒(méi)問(wèn)題啊
到此這篇關(guān)于ava調(diào)用chatgpt接口來(lái)實(shí)現(xiàn)專屬于自己的人工智能助手的文章就介紹到這了,更多相關(guān)java調(diào)用chatgpt實(shí)現(xiàn)人工智能助手內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring?@Bean注解深入分析源碼執(zhí)行過(guò)程
隨著SpringBoot的流行,我們現(xiàn)在更多采用基于注解式的配置從而替換掉了基于XML的配置,所以本篇文章我們主要探討基于注解的@Bean以及和其他注解的使用2023-01-01oracle+mybatis-plus+springboot實(shí)現(xiàn)分頁(yè)查詢的實(shí)例
本文主要介紹了oracle+mybatis-plus+springboot實(shí)現(xiàn)分頁(yè)查詢,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08struts2 validation.xml 驗(yàn)證規(guī)則代碼解析
這篇文章主要介紹了struts2 validation.xml 驗(yàn)證規(guī)則代碼解析,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01MyBatis詳細(xì)執(zhí)行流程的全紀(jì)錄
這篇文章主要給大家介紹了關(guān)于MyBatis詳細(xì)執(zhí)行流程的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04springboot默認(rèn)文件緩存(easy-captcha?驗(yàn)證碼)
這篇文章主要介紹了springboot的文件緩存(easy-captcha?驗(yàn)證碼),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-06-06通過(guò)Java實(shí)現(xiàn)bash命令過(guò)程解析
這篇文章主要介紹了通過(guò)Java實(shí)現(xiàn)bash命令過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01java判斷integer是否為空的詳細(xì)過(guò)程
在java編寫過(guò)程中,我們會(huì)使用到各種各樣的表達(dá)式,在使用表達(dá)式的過(guò)程中,有哪些安全問(wèn)題需要我們注意的呢?對(duì)java判斷integer是否為空相關(guān)知識(shí)感興趣的朋友一起來(lái)看看吧2023-02-02