借鉴LangChain思想使用Java实现大模型Function_Call工具开发及调用功能

在这里插入图片描述
关注公众号,回复“工具调用”获取代码。

背景

博主之前研究的是ChatGLM3模型,该模型提供了Openai方式调用工具的代码。但后续转到Qwen1.5模型后,好像不可以直接用Openai接口的方式调用工具了。
然后,博主研究了Qwen-Agent框架,实现了自定义工具:

https://blog.csdn.net/weixin_44455388/article/details/136524354?spm=1001.2014.3001.5501

研究了LangChain框架,实现了自定义工具:

https://blog.csdn.net/weixin_44455388/article/details/136536875?spm=1001.2014.3001.5501

虽然,使用以上框架实现了自定义工具,但是调用工具时,均需要依赖于python环境和以上框架,觉得还是有一定的限制。再加上,博主之前的基于大模型的所有功能(本地知识库、Text2SQL等)均是使用Java调用OpenAI接口实现,没有使用类似langChain这样的python框架。作为倔强的Java程序员,还是想用Java去实现自定义工具。

思路

首先需要封装OpenAIChat对象,对象应包括以下变量:

/**
 * 使用的模型名称.
 */
@Builder.Default
private String model = "gpt-3.5-turbo";

/**
 * 模型的API地址.
 */
@Builder.Default
private String endpointUrl = "http://127.0.0.1:8000/";

/**
 * 模型的最大token数.
 */
@Builder.Default
private int maxToken = 20000;

/**
 * 模型的temperature.
 */
@Builder.Default
private float temperature = 0.9f;

/**
 * 模型的TopP
 */
@Builder.Default
private float topP = 0.78f;

/**
 * 是否使用历史记录.
 */
private boolean withHistory;

/**
 * 历史记录
 */
@Builder.Default
private List<List<?>> history = new ArrayList<>();

OpenAIChat对象中构建工具调用流式问答的方法实现:

/**
 * 工具流式问答
 * @param sessionId 会话ID
 * @param prompt 提示词
 * @param baseTools 工具
 * @return
 */
@Override
public Flux<String> streamChatWithTools(String sessionId, String prompt, List<BaseTool> baseTools) {
    String class2Json = buildClass2Json(new BaseTool());
    String finalPrompt = String.format("你是一个AI助手,我会给你一个工具对象集合,工具对象包括name(工具名)、description(工具描述)、params(工具参数)。" +
            "你可以结合工具对象,从用户的问句中提取到关键词,确定要实现用户的任务应该喧杂哪个工具对象。" +
            "用户的任务是:%S。" +
            "工具对象集合是:%s。" +
            "您的响应结果必须为JSON格式,并且不要返回任何不必要的解释,只提供遵循此格式的符合RFC8259的JSON响应。以下是输出必须遵守的JSON Schema实例:```%s```",
            prompt, JSON.toJSONString(baseTools), class2Json);
    String funcParams = chat(finalPrompt);
    funcParams = JSON.parseObject(funcParams, OpenAIChatResponse.class).getChoices().get(0).getMessage().getContent();
    funcParams = funcParams.substring(funcParams.indexOf("{"), funcParams.lastIndexOf("}")+1);
    String toolResult = LoadFunctions.load(JSON.parseObject(funcParams, BaseTool.class));
    LOG.info("工具调用结果为:"+toolResult);
    String promptFormat = String.format("基于以下内容:{%s}。请回答:{%s}", toolResult, prompt);
    Flux<String> stringFlux = streamChat(sessionId, promptFormat);
    return stringFlux;
}

创建一个自定义工具类,所有的工具都在这里实现,我这里创建了三个工具(查询天气、返回一个UUID、查询手机号的归属地):

public class FunctionCaller {

    /**
     * 查询天气API
     *
     * @param params
     * @return
     */
    public String getWeather(Map<String,Object> params) {
        String cityName = params.get("cityName").toString();
        if (cityName == null || cityName.isEmpty()) {
            throw new IllegalArgumentException("City name must not be null or empty");
        }

        OkHttpClient client = new OkHttpClient.Builder()
                .connectTimeout(60, TimeUnit.SECONDS)
                .writeTimeout(60, TimeUnit.SECONDS)
                .readTimeout(60, TimeUnit.SECONDS)
                .build();

        try {
            Map<String, String> headers = new HashMap<>(16);
            headers.put("Content-Type", "application/json");
            Request.Builder builder = new Request.Builder()
                    .url("https://wttr.in/" + cityName + "?format=j1");
            builder.headers(Headers.of(headers));
            builder.method("GET", null);
            Request request = builder.build();
            Response response = client.newCall(request).execute();
            if (response.isSuccessful()) {
                ResponseBody responseBody = response.body();
                JSONObject jsonObject = JSONObject.parseObject(responseBody.string());
                JSONObject weatherData = new JSONObject();

                // Extract the desired weather data from the JSON response
                JSONArray currentCondition = jsonObject.getJSONArray("current_condition");
                JSONObject firstEntry = currentCondition.getJSONObject(0);
                weatherData.put("temp_C", firstEntry.getString("temp_C"));
                weatherData.put("FeelsLikeC", firstEntry.getString("FeelsLikeC"));
                weatherData.put("humidity", firstEntry.getString("humidity"));
                weatherData.put("weatherDesc", firstEntry.getString("weatherDesc"));
                weatherData.put("observation_time", firstEntry.getString("observation_time"));
                weatherData.put("cityName", params.get("cityName").toString());

                return weatherData.toString();
            } else {
                throw new HttpRuntimeException("Failed.接口访问失败");
            }
        } catch (IOException e) {
            e.printStackTrace();
            return "Error encountered while fetching weather data!";
        }
    }

    /**
     * 随机获取一个uuid
     * @return
     */
    public String getUuid(Map<String,Object> params){
        return UUID.randomUUID().toString();
    }

    /**
     * 查询手机号的归属地
     * @param params
     * @return
     */
    public String getMobileAddress(Map<String,Object> params) {
        String mobileNum = params.get("mobileNum").toString();
        try {
            OkHttpClient client = new OkHttpClient.Builder()
                    .connectTimeout(60, TimeUnit.SECONDS)
                    .writeTimeout(60, TimeUnit.SECONDS)
                    .readTimeout(60, TimeUnit.SECONDS)
                    .build();
            MediaType mediaType = MediaType.parse("application/x-www-form-urlencoded");
            RequestBody body = RequestBody.create(mediaType, "mobile="+mobileNum);
            Request request = new Request.Builder()
                    .url("https://eolink.o.apispace.com/teladress/teladress")
                    .method("POST",body)
                    .addHeader("X-APISpace-Token","v1a524e7ctm4h87ilxxxxxxxxxxxxx")
                    .addHeader("Content-Type","")
                    .build();

            Response response = client.newCall(request).execute();
            if (response.isSuccessful()) {
                ResponseBody responseBody = response.body();
                return responseBody.string();
            } else {
                throw new HttpRuntimeException("Failed.接口访问失败");
            }
        } catch (Exception e) {
            e.printStackTrace();
            return "Error fetching mobile address: " + e.getMessage();
        }
    }
}

然后,采用反射的方式调用这些工具:

/**
 * 通过反射机制调用函数
 * @param methodName
 * @param jsonNode
 * @return
 */
public static Object reflect(String methodName, Map<String,String> jsonNode){
    FunctionCaller functionCaller = new FunctionCaller();
    Method method = ReflectUtil.getMethod(FunctionCaller.class, methodName, new Class[]{Map.class});
    try {
        Object invoke = method.invoke(functionCaller, jsonNode);
        return invoke;
    } catch (IllegalAccessException e) {
        LOG.error("FunctionReflect reflect occur illegal access exception,method name = {},jsonNode = {}",methodName,jsonNode,e);
        throw new RuntimeException(e);
    } catch (InvocationTargetException e) {
        LOG.error("FunctionReflect reflect occur invocation target exception,method name = {},jsonNode = {}",methodName,jsonNode,e);
        throw new RuntimeException(e);
    }
}

调用的时候,将所有的工具集合作为参数,传入OpenAIChat的streamChatWithTools方法:

public static void test2(){
    BaseTool baseTool = new BaseTool();
    baseTool.setName("getWeather");
    Map<String,String> map = new HashMap<>(16);
    map.put("cityName","城市");
    baseTool.setParams(map);
    baseTool.setDescription("查询天气工具");

    BaseTool baseTool1 = new BaseTool();
    baseTool1.setName("getUuid");
    baseTool1.setDescription("获取UUID");
    baseTool1.setParams(null);

    BaseTool baseTool2 = new BaseTool();
    baseTool2.setName("getMobileAddress");
    baseTool2.setDescription("查询手机号归属地");
    Map<String,String> map2 = new HashMap<>(16);
    map2.put("mobileNum","手机号");
    baseTool2.setParams(map2);

    List<BaseTool> baseTools = Arrays.asList(baseTool,baseTool1,baseTool2);

    OpenAIChat openAIChat = OpenAIChat.builder()
            .endpointUrl("http://127.0.0.1:11434/v1/chat/completions")
            .model("qwen:7b-chat-v1.5-q5_K_M")
            .build().init();
    Flux<String> stringFlux = openAIChat.streamChatWithTools("112233","查询北京的天气", baseTools);
    stringFlux.subscribe();
    //System.out.println(s);
}

然后就可以实现工具调用了:
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

如何备考2024年AMC10:吃透2000-2023年1250道真题(限时免费送)

有家长朋友问&#xff0c;有没有适合初中学生参加的奥数类比赛&#xff1f;我推荐AMC10美国数学竞赛&#xff0c;在国内可以方便地参加&#xff0c;而且每年全国各省市参加的初中生越来越多。关于AMC10详细的介绍和常见问题解答&#xff0c;可以联系我获得。 那么如何在AMC10竞…

【智能算法】流向算法(FDA)原理及实现

目录 1.背景2.算法原理2.1算法思想2.2算法过程 3.结果展示4.参考文献 1.背景 2021年&#xff0c;H Karami等人受到水流运动规律启发&#xff0c;提出了流向算法&#xff08;Flow Direction Algorithm&#xff0c; FDA&#xff09;。 2.算法原理 2.1算法思想 FDA受到了流入排…

Jenkins用户角色权限管理

Jenkins作为一款强大的自动化构建与持续集成工具&#xff0c;用户角色权限管理是其功能体系中不可或缺的一环。有效的权限管理能确保项目的安全稳定&#xff0c;避免敏感信息泄露。 1、安装插件&#xff1a;Role-based Authorization Strategy 系统管理 > 插件管理 > 可…

第十一章:位运算符与位运算

文章目录 第十一章&#xff1a;位运算符与位运算1.按位与运算&#xff1a;&2.按位或运算&#xff1a;|3.按位异或运算&#xff1a;^4.取反运算符&#xff1a;~5.左移运算符&#xff1a;<<6.右移运算符&#xff1a;>>总结 第十一章&#xff1a;位运算符与位运算…

StringRedisTemplate与RedisTemplate详解【序列化的方式不同】

spring 封装了 RedisTemplate 对象来进行对redis的各种操作&#xff0c;它支持所有的 redis 原生的 api。在RedisTemplate中提供了几个常用的接口方法的使用&#xff0c;分别是: private ValueOperations<K, V> valueOps; private HashOperations<K, V> hashOps; …

回文数个数-第12届蓝桥杯选拔赛Python真题精选

[导读]&#xff1a;超平老师的Scratch蓝桥杯真题解读系列在推出之后&#xff0c;受到了广大老师和家长的好评&#xff0c;非常感谢各位的认可和厚爱。作为回馈&#xff0c;超平老师计划推出《Python蓝桥杯真题解析100讲》&#xff0c;这是解读系列的第43讲。 回文数个数&#…

Unity 学习日记 12.小球撞击冰块游戏

目录 1.准备场景 2.让小球动起来 3.用鼠标把小球甩出去 4.加入鼠标点击小球的判断 5.小球与冰块的碰撞测试 6.撞击后销毁冰块 ​编辑 7.显示游戏计时 8.显示扔球次数 9.显示剩余冰块个数 10.游戏结束 11.完整代码 下载源码 UnityPackage 最终效果&#xff1a; 1.准…

【氮化镓】位错对氮化镓(GaN)电子能量损失谱(EEL)的影响

本文献《Influence of dislocations on electron energy-loss spectra in gallium nitride》由C. J. Fall等人撰写&#xff0c;发表于2002年。研究团队通过第一性原理计算&#xff0c;探讨了位错对氮化镓&#xff08;GaN&#xff09;电子能量损失谱&#xff08;EEL&#xff09;…

八大排序之堆排序

堆排序算法思想&#xff1a; 堆排序是利用二叉树的原理&#xff0c;模拟二叉树将所求的数据放入存放树中&#xff0c;先将所有数据按照大根堆排列&#xff0c;排列之后再依次的给树按从小到大排列. 例如 2 5 44 21 11 6 1 9 我们将数据按照这样的二叉树的形式列举出来&#…

FPGA时钟资源详解(1)——时钟Buffer的选择

FPGA时钟系列文章总览&#xff1a;FPGA原理与结构&#xff08;14&#xff09;——时钟资源https://ztzhang.blog.csdn.net/article/details/132307564 目录 一、概述 二、时钟Buffer的选择 2.1 BUFG 2.2 BUFR 和 BUFIO 2.2.1 源同步接口的支持 2.2.2 扩展时钟域…

DREAM: A Dynamic Scheduler for Dynamic Real-time Multi-model ML Workloads——论文泛读

ASPLOS 2024 Paper 论文阅读笔记整理 问题 新兴的实时多模型ML&#xff08;RTMM&#xff09;工作负载&#xff0c;如AR/VR和无人机控制&#xff0c;涉及各种粒度的动态行为&#xff1a;任务、模型和模型中的层。这种动态行为给ML系统中的系统软件带来了新的挑战&#xff0c;与…

深度学习中不同学习率调整策略

1、StepLR 功能&#xff1a;固定等间隔调整学习率 主要参数&#xff1a; step_size:调整间隔数 gamma&#xff1a;调整系数 调整方式&#xff1a; l r l r ∗ g a m m a lrlr\ast gamma lrlr∗gamma 2、MultiStepLR 功能&#xff1a;按给定间隔调整学习率 主要参数&#xf…

Linux——磁盘与文件系统管理

目录 磁盘分区的表示 硬盘分区 分区类型 确认系统中的磁盘设备——fdisk 规划硬盘中的分区——fdisk 文件系统 文件系统类型&#xff1a; 在分区中创建文件系统——mkfs&#xff0c;mkswap 挂载文件系统 mount命令 umount命令 查看分区挂载情况 设置启动载入&…

负荷频率控制LFC,自抗扰ADRC控制,麻雀SSA算法优化自抗扰参数,两区域二次调频simulink/matlab

红色曲线为优化结果&#xff0c;蓝色曲线为没有自抗扰和没有优化的结果&#xff01;

Mac系统中使用VSCode安装C#开发环境进行编译调试

VSCode安装插件 C#c# Dev Kit 安装Mac版本 .net .net下载地址 查看安装结果 dotnet --list-sdksdotnet --info配置环境变量 open -e ~/.bash_profile添加如下内容 export DOTNET_ROOT/usr/local/share/dotnet export PATH$PATH:$DOTNET_ROOT终端重新加载配置文件 sourc…

原子激光器(原子激射器)可发射相干原子束 目前仍处于技术研究阶段

原子激光器&#xff08;原子激射器&#xff09;可发射相干原子束 目前仍处于技术研究阶段 原子激光器&#xff0c;也称为原子激射器&#xff0c;是一种能够产生原子激光的器件。原子激光由粒子组成&#xff0c;拥有频率和波长&#xff0c;原子激光器受激发射电磁波&#xff0c;…

顺丰接口接入-主要处理下单接口上电子面单上传问题

概述 最近接到一个需求&#xff0c;需要和顺丰接口对接。由于是第一次对接&#xff0c;就需要把所有的流程全部走一遍&#xff0c;从 注册到 关联API 以及代码测试&#xff0c;电子面单审核&#xff0c;上线&#xff0c;下面就分开来说明把。本来是想着偷懒来着&#xff0c;作…

Days 35 ElfBoard板对Java的支持

Java作为一种功能强大且广泛应用的编程语言&#xff0c;具有广泛的适应性和实用性。在ELF 1开发板上集成Java支持&#xff0c;无疑将赋予嵌入式开发者更广阔的选择空间&#xff0c;今天就为各位小伙伴详细解析如何在ELF 1开发板上成功部署和运行Java环境。 1.拷贝两个压缩包到E…

FME学习之旅---day14

我们付出一些成本&#xff0c;时间的或者其他&#xff0c;最终总能收获一些什么。 【FME-HOW-TO系列】13 通过重新采样修改栅格像元大小 除了使用RasterResampler转换器进行重采样的操作外&#xff0c;还需要了解不同的插值方法&#xff0c;各方法大概的不同。 可以参考ArcG…

计算机网络(二)物理层

物理层 一、通信基础1.奈氏准则、香农定理2.编码与调制3.电路交换、报文交换、分组交换 二、 传输介质、设备1.导向性传输介质&#xff1a;1.1双绞线1.2 同轴电缆1.3光纤 2.非导向性传输介质&#xff1a; 一、通信基础 信道带宽&#xff1a;信道能通过的最高频率和最低频率之差…